From 5b16199a610ebc2a4ea923ffe2177985da279a8a Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 5 Feb 2024 09:49:55 -0500
Subject: [PATCH 01/83] build(deps): bump github.com/cert-manager/cert-manager
(#6158)
Bumps [github.com/cert-manager/cert-manager](https://github.com/cert-manager/cert-manager) from 1.13.3 to 1.14.1.
- [Release notes](https://github.com/cert-manager/cert-manager/releases)
- [Commits](https://github.com/cert-manager/cert-manager/compare/v1.13.3...v1.14.1)
---
updated-dependencies:
- dependency-name: github.com/cert-manager/cert-manager
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
go.mod | 24 ++++++++++++------------
go.sum | 56 +++++++++++++++++++++++++++-----------------------------
2 files changed, 39 insertions(+), 41 deletions(-)
diff --git a/go.mod b/go.mod
index eef75cd5b4c..325523c784e 100644
--- a/go.mod
+++ b/go.mod
@@ -8,7 +8,7 @@ require (
github.com/ahmetb/gen-crd-api-reference-docs v0.3.0
github.com/alecthomas/kingpin/v2 v2.4.0
github.com/bombsimon/logrusr/v4 v4.1.0
- github.com/cert-manager/cert-manager v1.13.3
+ github.com/cert-manager/cert-manager v1.14.1
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc
github.com/distribution/reference v0.5.0
github.com/envoyproxy/go-control-plane v0.12.1-0.20240111020705-5401a878d8bb
@@ -31,7 +31,7 @@ require (
go.uber.org/automaxprocs v1.5.3
golang.org/x/oauth2 v0.16.0
gonum.org/v1/plot v0.14.0
- google.golang.org/genproto/googleapis/rpc v0.0.0-20231106174013-bbf56f31fb17
+ google.golang.org/genproto/googleapis/rpc v0.0.0-20240102182953-50ed04b92917
google.golang.org/grpc v1.61.0
google.golang.org/protobuf v1.32.0
gopkg.in/yaml.v3 v3.0.1
@@ -40,6 +40,7 @@ require (
k8s.io/apimachinery v0.29.1
k8s.io/client-go v0.29.1
k8s.io/klog/v2 v2.120.1
+ k8s.io/utils v0.0.0-20240102154912-e7106e64919e
sigs.k8s.io/controller-runtime v0.17.0
sigs.k8s.io/controller-tools v0.14.0
sigs.k8s.io/gateway-api v1.0.0
@@ -67,9 +68,9 @@ require (
github.com/go-errors/errors v1.4.2 // indirect
github.com/go-fonts/liberation v0.3.1 // indirect
github.com/go-latex/latex v0.0.0-20230307184459-12ec69307ad9 // indirect
- github.com/go-openapi/jsonpointer v0.20.0 // indirect
- github.com/go-openapi/jsonreference v0.20.2 // indirect
- github.com/go-openapi/swag v0.22.4 // indirect
+ github.com/go-openapi/jsonpointer v0.20.2 // indirect
+ github.com/go-openapi/jsonreference v0.20.4 // indirect
+ github.com/go-openapi/swag v0.22.7 // indirect
github.com/go-pdf/fpdf v0.8.0 // indirect
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect
github.com/gobuffalo/flect v1.0.2 // indirect
@@ -114,32 +115,31 @@ require (
github.com/spf13/jwalterweatherman v1.1.0 // indirect
github.com/spf13/pflag v1.0.5 // indirect
github.com/spf13/viper v1.15.0 // indirect
- github.com/stretchr/objx v0.5.0 // indirect
+ github.com/stretchr/objx v0.5.1 // indirect
github.com/subosito/gotenv v1.4.2 // indirect
github.com/tsaarni/x500dn v1.0.0 // indirect
github.com/xhit/go-str2duration/v2 v2.1.0 // indirect
golang.org/x/crypto v0.18.0 // indirect
- golang.org/x/exp v0.0.0-20231006140011-7918f672742d // indirect
+ golang.org/x/exp v0.0.0-20231226003508-02704c960a9b // indirect
golang.org/x/image v0.11.0 // indirect
golang.org/x/mod v0.14.0 // indirect
golang.org/x/net v0.20.0 // indirect
golang.org/x/sys v0.16.0 // indirect
golang.org/x/term v0.16.0 // indirect
golang.org/x/text v0.14.0 // indirect
- golang.org/x/time v0.3.0 // indirect
+ golang.org/x/time v0.5.0 // indirect
golang.org/x/tools v0.16.1 // indirect
gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect
google.golang.org/appengine v1.6.8 // indirect
- google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17 // indirect
- google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17 // indirect
+ google.golang.org/genproto v0.0.0-20240102182953-50ed04b92917 // indirect
+ google.golang.org/genproto/googleapis/api v0.0.0-20240102182953-50ed04b92917 // indirect
gopkg.in/inf.v0 v0.9.1 // indirect
gopkg.in/ini.v1 v1.67.0 // indirect
gopkg.in/yaml.v2 v2.4.0 // indirect
k8s.io/component-base v0.29.1 // indirect
k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01 // indirect
k8s.io/klog v1.0.0 // indirect
- k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 // indirect
- k8s.io/utils v0.0.0-20230726121419-3b25d923346b // indirect
+ k8s.io/kube-openapi v0.0.0-20240103051144-eec4567ac022 // indirect
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect
sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect
)
diff --git a/go.sum b/go.sum
index b567c9bc7db..f0b7e04b19a 100644
--- a/go.sum
+++ b/go.sum
@@ -68,8 +68,8 @@ github.com/campoy/embedmd v1.0.0/go.mod h1:oxyr9RCiSXg0M3VJ3ks0UGfp98BpSSGr0kpiX
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/census-instrumentation/opencensus-proto v0.4.1 h1:iKLQ0xPNFxR/2hzXZMrBo8f1j86j5WHzznCCQxV/b8g=
github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw=
-github.com/cert-manager/cert-manager v1.13.3 h1:3R4G0RI7K0OkTZhWlVOC5SGZMYa2NwqmQJoyKydrz/M=
-github.com/cert-manager/cert-manager v1.13.3/go.mod h1:BM2+Pt/NmSv1Zr25/MHv6BgIEF9IUxA1xAjp80qkxgc=
+github.com/cert-manager/cert-manager v1.14.1 h1:i5sJHfEucqpAfVjkCe3n4sO5S+6YBaN2Yu18+l/1ZMw=
+github.com/cert-manager/cert-manager v1.14.1/go.mod h1:pik7K6jXfgh++lfVJ/i1HzEnDluSUtTVLXSHikj8Lho=
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/chigopher/pathlib v0.19.1 h1:RoLlUJc0CqBGwq239cilyhxPNLXTK+HXoASGyGznx5A=
@@ -85,7 +85,6 @@ github.com/cncf/xds/go v0.0.0-20231109132714-523115ebc101 h1:7To3pQ+pZo0i3dsWEbi
github.com/cncf/xds/go v0.0.0-20231109132714-523115ebc101/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/coreos/go-systemd/v22 v22.3.3-0.20220203105225-a9a7ef127534/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
-github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
@@ -137,14 +136,12 @@ github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ=
github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
github.com/go-logr/zapr v1.3.0 h1:XGdV8XW8zdwFiwOA2Dryh1gj2KRQyOOoNmBy4EplIcQ=
github.com/go-logr/zapr v1.3.0/go.mod h1:YKepepNBd1u/oyhd/yQmtjVXmm9uML4IXUgMOwR8/Gg=
-github.com/go-openapi/jsonpointer v0.19.6/go.mod h1:osyAmYz/mB/C3I+WsTTSgw1ONzaLJoLCyoi6/zppojs=
-github.com/go-openapi/jsonpointer v0.20.0 h1:ESKJdU9ASRfaPNOPRx12IUyA1vn3R9GiE3KYD14BXdQ=
-github.com/go-openapi/jsonpointer v0.20.0/go.mod h1:6PGzBjjIIumbLYysB73Klnms1mwnU4G3YHOECG3CedA=
-github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE=
-github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
-github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
-github.com/go-openapi/swag v0.22.4 h1:QLMzNJnMGPRNDCbySlcj1x01tzU8/9LTTL9hZZZogBU=
-github.com/go-openapi/swag v0.22.4/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
+github.com/go-openapi/jsonpointer v0.20.2 h1:mQc3nmndL8ZBzStEo3JYF8wzmeWffDH4VbXz58sAx6Q=
+github.com/go-openapi/jsonpointer v0.20.2/go.mod h1:bHen+N0u1KEO3YlmqOjTT9Adn1RfD91Ar825/PuiRVs=
+github.com/go-openapi/jsonreference v0.20.4 h1:bKlDxQxQJgwpUSgOENiMPzCTBVuc7vTdXSSgNeAhojU=
+github.com/go-openapi/jsonreference v0.20.4/go.mod h1:5pZJyJP2MnYCpoeoMAql78cCHauHj0V9Lhc506VOpw4=
+github.com/go-openapi/swag v0.22.7 h1:JWrc1uc/P9cSomxfnsFSVWoE1FW6bNbrVPmpQYpCcR8=
+github.com/go-openapi/swag v0.22.7/go.mod h1:Gl91UqO+btAM0plGGxHqJcQZ1ZTy6jbmridBTsDy8A0=
github.com/go-pdf/fpdf v0.8.0 h1:IJKpdaagnWUeSkUFUjTcSzTppFxmv8ucGQyNPQWxYOQ=
github.com/go-pdf/fpdf v0.8.0/go.mod h1:gfqhcNwXrsd3XYKte9a7vM3smvU/jB4ZRDrmWSxpfdc=
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
@@ -270,7 +267,6 @@ github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxv
github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg=
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
-github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
@@ -337,8 +333,8 @@ github.com/prometheus/common v0.46.0/go.mod h1:Tp0qkxpb9Jsg54QMe+EAmqXkSV7Evdy1B
github.com/prometheus/procfs v0.12.0 h1:jluTpSng7V9hY0O2R9DzzJHYb2xULk9VTR1V1R/k6Bo=
github.com/prometheus/procfs v0.12.0/go.mod h1:pcuDEFsWDnvcgNzo4EEweacyhjeA9Zk3cnaOZAZEfOo=
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
-github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M=
-github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA=
+github.com/rogpeppe/go-internal v1.12.0 h1:exVL4IDcn6na9z1rAb56Vxr+CgyK3nn3O+epU5NdKM8=
+github.com/rogpeppe/go-internal v1.12.0/go.mod h1:E+RYuTGaKKdloAfM02xzb0FW3Paa99yedzYV+kq4uf4=
github.com/rs/xid v1.4.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
github.com/rs/zerolog v1.29.0 h1:Zes4hju04hjbvkVkOhdl2HpZa+0PmVwigmo8XoORE5w=
github.com/rs/zerolog v1.29.0/go.mod h1:NILgTygv/Uej1ra5XxGf82ZFSLk58MFGAUS2o6usyD0=
@@ -364,8 +360,9 @@ github.com/spf13/viper v1.15.0/go.mod h1:fFcTBJxvhhzSJiZy8n+PeW6t8l+KeT/uTARa0jH
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
-github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
+github.com/stretchr/objx v0.5.1 h1:4VhoImhV/Bm0ToFkXFi8hXNXwpDRZ/ynw3amt82mzq0=
+github.com/stretchr/objx v0.5.1/go.mod h1:/iHQpkQwBD6DLUmQ4pE+s1TXdob1mORJ4/UFdrifcy0=
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
@@ -375,6 +372,7 @@ github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
+github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk=
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8=
@@ -430,8 +428,8 @@ golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u0
golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4=
golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM=
golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU=
-golang.org/x/exp v0.0.0-20231006140011-7918f672742d h1:jtJma62tbqLibJ5sFQz8bKtEM8rJBtfilJ2qTU199MI=
-golang.org/x/exp v0.0.0-20231006140011-7918f672742d/go.mod h1:ldy0pHrwJyGW56pPQzzkH36rKxoZW1tw7ZJpeKx+hdo=
+golang.org/x/exp v0.0.0-20231226003508-02704c960a9b h1:kLiC65FbiHWFAOu+lxwNPujcsl8VYyTYYEZnsOO1WK4=
+golang.org/x/exp v0.0.0-20231226003508-02704c960a9b/go.mod h1:iRJReGqOEeBhDZGkGbynYwcHlctCvnjTYIamk7uXpHI=
golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
golang.org/x/image v0.11.0 h1:ds2RoQvBvYTiJkwpSFDwCcDFNX7DqjL2WsUgTNk0Ooo=
@@ -587,8 +585,8 @@ golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
-golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4=
-golang.org/x/time v0.3.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
+golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk=
+golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
@@ -719,12 +717,12 @@ google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6D
google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
-google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17 h1:wpZ8pe2x1Q3f2KyT5f8oP/fa9rHAKgFPr/HZdNuS+PQ=
-google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17/go.mod h1:J7XzRzVy1+IPwWHZUzoD0IccYZIrXILAQpc+Qy9CMhY=
-google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17 h1:JpwMPBpFN3uKhdaekDpiNlImDdkUAyiJ6ez/uxGaUSo=
-google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17/go.mod h1:0xJLfVdJqpAPl8tDg1ujOCGzx6LFLttXT5NhllGOXY4=
-google.golang.org/genproto/googleapis/rpc v0.0.0-20231106174013-bbf56f31fb17 h1:Jyp0Hsi0bmHXG6k9eATXoYtjd6e2UzZ1SCn/wIupY14=
-google.golang.org/genproto/googleapis/rpc v0.0.0-20231106174013-bbf56f31fb17/go.mod h1:oQ5rr10WTTMvP4A36n8JpR1OrO1BEiV4f78CneXZxkA=
+google.golang.org/genproto v0.0.0-20240102182953-50ed04b92917 h1:nz5NESFLZbJGPFxDT/HCn+V1mZ8JGNoY4nUpmW/Y2eg=
+google.golang.org/genproto v0.0.0-20240102182953-50ed04b92917/go.mod h1:pZqR+glSb11aJ+JQcczCvgf47+duRuzNSKqE8YAQnV0=
+google.golang.org/genproto/googleapis/api v0.0.0-20240102182953-50ed04b92917 h1:rcS6EyEaoCO52hQDupoSfrxI3R6C2Tq741is7X8OvnM=
+google.golang.org/genproto/googleapis/api v0.0.0-20240102182953-50ed04b92917/go.mod h1:CmlNWB9lSezaYELKS5Ym1r44VrrbPUa7JTvw+6MbpJ0=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20240102182953-50ed04b92917 h1:6G8oQ016D88m1xAKljMlBOOGWDZkes4kMhgGFlf8WcQ=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20240102182953-50ed04b92917/go.mod h1:xtjpI3tXFPP051KaWnhvxkiubL/6dJ18vLVf7q2pTOU=
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
@@ -804,10 +802,10 @@ k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
k8s.io/klog/v2 v2.2.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y=
k8s.io/klog/v2 v2.120.1 h1:QXU6cPEOIslTGvZaXvFWiP9VKyeet3sawzTOvdXb4Vw=
k8s.io/klog/v2 v2.120.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE=
-k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 h1:aVUu9fTY98ivBPKR9Y5w/AuzbMm96cd3YHRTU83I780=
-k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00/go.mod h1:AsvuZPBlUDVuCdzJ87iajxtXuR9oktsTctW/R9wwouA=
-k8s.io/utils v0.0.0-20230726121419-3b25d923346b h1:sgn3ZU783SCgtaSJjpcVVlRqd6GSnlTLKgpAAttJvpI=
-k8s.io/utils v0.0.0-20230726121419-3b25d923346b/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
+k8s.io/kube-openapi v0.0.0-20240103051144-eec4567ac022 h1:avRdiaB03v88Mfvum2S3BBwkNuTlmuar4LlfO9Hajko=
+k8s.io/kube-openapi v0.0.0-20240103051144-eec4567ac022/go.mod h1:sIV51WBTkZrlGOJMCDZDA1IaPBUDTulPpD4y7oe038k=
+k8s.io/utils v0.0.0-20240102154912-e7106e64919e h1:eQ/4ljkx21sObifjzXwlPKpdGLrCfRziVtos3ofG/sQ=
+k8s.io/utils v0.0.0-20240102154912-e7106e64919e/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
rsc.io/pdf v0.1.1 h1:k1MczvYDUvJBe93bYd7wrZLLUEcLZAuF824/I4e5Xr4=
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
From 9f248c22ab3b4cfc5d3a4094da6b3f8992cc30f6 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 5 Feb 2024 09:50:44 -0500
Subject: [PATCH 02/83] build(deps): bump github/codeql-action from 3.23.2 to
3.24.0 (#6156)
Bumps [github/codeql-action](https://github.com/github/codeql-action) from 3.23.2 to 3.24.0.
- [Release notes](https://github.com/github/codeql-action/releases)
- [Changelog](https://github.com/github/codeql-action/blob/main/CHANGELOG.md)
- [Commits](https://github.com/github/codeql-action/compare/b7bf0a3ed3ecfa44160715d7c442788f65f0f923...e8893c57a1f3a2b659b6b55564fdfdbbd2982911)
---
updated-dependencies:
- dependency-name: github/codeql-action
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/codeql-analysis.yml | 6 +++---
.github/workflows/openssf-scorecard.yaml | 2 +-
.github/workflows/trivy-scan.yaml | 2 +-
3 files changed, 5 insertions(+), 5 deletions(-)
diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml
index da8eda043d0..5c682d12a76 100644
--- a/.github/workflows/codeql-analysis.yml
+++ b/.github/workflows/codeql-analysis.yml
@@ -41,11 +41,11 @@ jobs:
cache: false
# Initializes the CodeQL tools for scanning.
- name: Initialize CodeQL
- uses: github/codeql-action/init@b7bf0a3ed3ecfa44160715d7c442788f65f0f923 # v3.23.2
+ uses: github/codeql-action/init@e8893c57a1f3a2b659b6b55564fdfdbbd2982911 # v3.24.0
with:
languages: go
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
- name: Autobuild
- uses: github/codeql-action/autobuild@b7bf0a3ed3ecfa44160715d7c442788f65f0f923 # v3.23.2
+ uses: github/codeql-action/autobuild@e8893c57a1f3a2b659b6b55564fdfdbbd2982911 # v3.24.0
- name: Perform CodeQL Analysis
- uses: github/codeql-action/analyze@b7bf0a3ed3ecfa44160715d7c442788f65f0f923 # v3.23.2
+ uses: github/codeql-action/analyze@e8893c57a1f3a2b659b6b55564fdfdbbd2982911 # v3.24.0
diff --git a/.github/workflows/openssf-scorecard.yaml b/.github/workflows/openssf-scorecard.yaml
index fdd6348df0c..eadc61b9dd5 100644
--- a/.github/workflows/openssf-scorecard.yaml
+++ b/.github/workflows/openssf-scorecard.yaml
@@ -37,6 +37,6 @@ jobs:
name: SARIF file
path: results.sarif
- name: "Upload to code-scanning"
- uses: github/codeql-action/upload-sarif@b7bf0a3ed3ecfa44160715d7c442788f65f0f923 # v3.23.2
+ uses: github/codeql-action/upload-sarif@e8893c57a1f3a2b659b6b55564fdfdbbd2982911 # v3.24.0
with:
sarif_file: results.sarif
diff --git a/.github/workflows/trivy-scan.yaml b/.github/workflows/trivy-scan.yaml
index 3197dceeb34..6b7421caea3 100644
--- a/.github/workflows/trivy-scan.yaml
+++ b/.github/workflows/trivy-scan.yaml
@@ -35,6 +35,6 @@ jobs:
output: 'trivy-results.sarif'
ignore-unfixed: true
severity: 'HIGH,CRITICAL'
- - uses: github/codeql-action/upload-sarif@b7bf0a3ed3ecfa44160715d7c442788f65f0f923 # v3.23.2
+ - uses: github/codeql-action/upload-sarif@e8893c57a1f3a2b659b6b55564fdfdbbd2982911 # v3.24.0
with:
sarif_file: 'trivy-results.sarif'
From aeb04fc69da12b408c2b5f6f865adf621d0abe17 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 5 Feb 2024 09:51:25 -0500
Subject: [PATCH 03/83] build(deps): bump necojackarc/auto-request-review from
0.12.0 to 0.13.0 (#6157)
Bumps [necojackarc/auto-request-review](https://github.com/necojackarc/auto-request-review) from 0.12.0 to 0.13.0.
- [Release notes](https://github.com/necojackarc/auto-request-review/releases)
- [Commits](https://github.com/necojackarc/auto-request-review/compare/6a51cebffe2c084705d9a7b394abd802e0119633...e89da1a8cd7c8c16d9de9c6e763290b6b0e3d424)
---
updated-dependencies:
- dependency-name: necojackarc/auto-request-review
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/request-reviews.yaml | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/.github/workflows/request-reviews.yaml b/.github/workflows/request-reviews.yaml
index e1edc0c37ae..467d23c7d73 100644
--- a/.github/workflows/request-reviews.yaml
+++ b/.github/workflows/request-reviews.yaml
@@ -11,7 +11,7 @@ jobs:
request-reviews:
runs-on: ubuntu-latest
steps:
- - uses: necojackarc/auto-request-review@6a51cebffe2c084705d9a7b394abd802e0119633 # v0.12.0
+ - uses: necojackarc/auto-request-review@e89da1a8cd7c8c16d9de9c6e763290b6b0e3d424 # v0.13.0
with:
token: ${{ secrets.PAT_FOR_AUTO_REQUEST_REVIEW }}
config: .github/reviewers.yaml
From 3fee38490b04ec200b001938af05d6f43adffef9 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 5 Feb 2024 12:59:30 -0500
Subject: [PATCH 04/83] build(deps): bump codecov/codecov-action from 3.1.5 to
4.0.1 (#6155)
* build(deps): bump codecov/codecov-action from 3.1.5 to 4.0.1
Bumps [codecov/codecov-action](https://github.com/codecov/codecov-action) from 3.1.5 to 4.0.1.
- [Release notes](https://github.com/codecov/codecov-action/releases)
- [Changelog](https://github.com/codecov/codecov-action/blob/main/CHANGELOG.md)
- [Commits](https://github.com/codecov/codecov-action/compare/4fe8c5f003fae66aa5ebb77cfd3e7bfbbda0b6b0...e0b68c6749509c5f83f984dd99a76a1c1a231044)
---
updated-dependencies:
- dependency-name: codecov/codecov-action
dependency-type: direct:production
update-type: version-update:semver-major
...
Signed-off-by: dependabot[bot]
* codecov token is now required
Signed-off-by: Sunjay Bhatia
---------
Signed-off-by: dependabot[bot]
Signed-off-by: Sunjay Bhatia
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: Sunjay Bhatia
---
.github/workflows/prbuild.yaml | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/.github/workflows/prbuild.yaml b/.github/workflows/prbuild.yaml
index 5033d735a62..623391e3add 100644
--- a/.github/workflows/prbuild.yaml
+++ b/.github/workflows/prbuild.yaml
@@ -289,8 +289,9 @@ jobs:
make check-coverage
- name: codeCoverage
if: ${{ success() }}
- uses: codecov/codecov-action@4fe8c5f003fae66aa5ebb77cfd3e7bfbbda0b6b0 # v3.1.5
+ uses: codecov/codecov-action@e0b68c6749509c5f83f984dd99a76a1c1a231044 # v4.0.1
with:
+ token: ${{ secrets.CODECOV_TOKEN }}
files: coverage.out
- uses: act10ns/slack@ed1309ab9862e57e9e583e51c7889486b9a00b0f # v2.0.0
with:
From 621e897020fc3d2d58dc87b101faf59da627b91c Mon Sep 17 00:00:00 2001
From: Lubron
Date: Mon, 5 Feb 2024 12:12:34 -0800
Subject: [PATCH 05/83] Add anti-affinity for envoy deployed by provisioner
(#6148)
Signed-off-by: lubronzhan
Signed-off-by: Lubron Zhan
---
.../unreleased/6148-lubronzhan-minor.md | 3 +++
examples/deployment/03-envoy-deployment.yaml | 13 +++++------
examples/render/contour-deployment.yaml | 13 +++++------
.../objects/dataplane/dataplane.go | 15 +++++++++++--
.../objects/dataplane/dataplane_test.go | 22 +++++++++++++++++++
site/content/resources/upgrading.md | 16 ++++++++++----
6 files changed, 62 insertions(+), 20 deletions(-)
create mode 100644 changelogs/unreleased/6148-lubronzhan-minor.md
diff --git a/changelogs/unreleased/6148-lubronzhan-minor.md b/changelogs/unreleased/6148-lubronzhan-minor.md
new file mode 100644
index 00000000000..3124fff004c
--- /dev/null
+++ b/changelogs/unreleased/6148-lubronzhan-minor.md
@@ -0,0 +1,3 @@
+## Add anti-affinity rule for envoy deployed by provisioner
+
+The envoy deployment created by the gateway provisioner now includes a default anti-affinity rule. The anti-affinity rule in the [example envoy deployment manifest](https://github.com/projectcontour/contour/blob/main/examples/deployment/03-envoy-deployment.yaml) is also updated to `preferredDuringSchedulingIgnoredDuringExecution` to be consistent with the contour deployment and the gateway provisioner anti-affinity rule.
diff --git a/examples/deployment/03-envoy-deployment.yaml b/examples/deployment/03-envoy-deployment.yaml
index 5236e57fbb1..a05c9a4dace 100644
--- a/examples/deployment/03-envoy-deployment.yaml
+++ b/examples/deployment/03-envoy-deployment.yaml
@@ -28,14 +28,13 @@ spec:
spec:
affinity:
podAntiAffinity:
- requiredDuringSchedulingIgnoredDuringExecution:
- - labelSelector:
- matchExpressions:
- - key: app
- operator: In
- values:
- - envoy
+ preferredDuringSchedulingIgnoredDuringExecution:
+ - podAffinityTerm:
+ labelSelector:
+ matchLabels:
+ app: envoy
topologyKey: "kubernetes.io/hostname"
+ weight: 100
containers:
- command:
- /bin/contour
diff --git a/examples/render/contour-deployment.yaml b/examples/render/contour-deployment.yaml
index 5085f15db8c..d79153ad41b 100644
--- a/examples/render/contour-deployment.yaml
+++ b/examples/render/contour-deployment.yaml
@@ -9149,14 +9149,13 @@ spec:
spec:
affinity:
podAntiAffinity:
- requiredDuringSchedulingIgnoredDuringExecution:
- - labelSelector:
- matchExpressions:
- - key: app
- operator: In
- values:
- - envoy
+ preferredDuringSchedulingIgnoredDuringExecution:
+ - podAffinityTerm:
+ labelSelector:
+ matchLabels:
+ app: envoy
topologyKey: "kubernetes.io/hostname"
+ weight: 100
containers:
- command:
- /bin/contour
diff --git a/internal/provisioner/objects/dataplane/dataplane.go b/internal/provisioner/objects/dataplane/dataplane.go
index a34e02fcc30..792eacaffbf 100644
--- a/internal/provisioner/objects/dataplane/dataplane.go
+++ b/internal/provisioner/objects/dataplane/dataplane.go
@@ -431,8 +431,19 @@ func desiredDeployment(contour *model.Contour, contourImage, envoyImage string)
Labels: envoyPodLabels(contour),
},
Spec: corev1.PodSpec{
- // TODO anti-affinity
- Affinity: nil,
+ Affinity: &corev1.Affinity{
+ PodAntiAffinity: &corev1.PodAntiAffinity{
+ PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{
+ {
+ Weight: int32(100),
+ PodAffinityTerm: corev1.PodAffinityTerm{
+ LabelSelector: EnvoyPodSelector(contour),
+ TopologyKey: "kubernetes.io/hostname",
+ },
+ },
+ },
+ },
+ },
Containers: containers,
InitContainers: initContainers,
Volumes: []corev1.Volume{
diff --git a/internal/provisioner/objects/dataplane/dataplane_test.go b/internal/provisioner/objects/dataplane/dataplane_test.go
index 09d2579ad49..928a9b7e631 100644
--- a/internal/provisioner/objects/dataplane/dataplane_test.go
+++ b/internal/provisioner/objects/dataplane/dataplane_test.go
@@ -260,6 +260,27 @@ func checkDaemonSetHasMetricsPort(t *testing.T, ds *appsv1.DaemonSet, port int32
t.Errorf("container has unexpected metrics port %d", port)
}
+func checkEnvoyDeploymentHasAffinity(t *testing.T, d *appsv1.Deployment, contour *model.Contour) {
+ t.Helper()
+ if apiequality.Semantic.DeepEqual(*d.Spec.Template.Spec.Affinity,
+ corev1.Affinity{
+ PodAntiAffinity: &corev1.PodAntiAffinity{
+ PreferredDuringSchedulingIgnoredDuringExecution: []corev1.WeightedPodAffinityTerm{
+ {
+ Weight: int32(100),
+ PodAffinityTerm: corev1.PodAffinityTerm{
+ LabelSelector: EnvoyPodSelector(contour),
+ TopologyKey: "kubernetes.io/hostname",
+ },
+ },
+ },
+ },
+ }) {
+ return
+ }
+ t.Errorf("container has unexpected affinity %v", d.Spec.Template.Spec.Affinity)
+}
+
func TestDesiredDaemonSet(t *testing.T) {
name := "ds-test"
cntr := model.Default(fmt.Sprintf("%s-ns", name), name)
@@ -360,6 +381,7 @@ func TestDesiredDeployment(t *testing.T) {
testEnvoyImage := "docker.io/envoyproxy/envoy:test"
deploy := desiredDeployment(cntr, testContourImage, testEnvoyImage)
checkDeploymentHasStrategy(t, deploy, cntr.Spec.EnvoyDeploymentStrategy)
+ checkEnvoyDeploymentHasAffinity(t, deploy, cntr)
}
func TestNodePlacementDaemonSet(t *testing.T) {
diff --git a/site/content/resources/upgrading.md b/site/content/resources/upgrading.md
index d6f5ebc6451..1841b7f3d85 100644
--- a/site/content/resources/upgrading.md
+++ b/site/content/resources/upgrading.md
@@ -15,6 +15,14 @@ Contour currently only tests sequential upgrades, i.e. without skipping any mino
This approach is recommended for users in order to minimize downtime and avoid any potential issues.
If you choose to skip versions while upgrading, please note that this may lead to additional downtime.
+# Known issues
+
+1. Envoy pod stuck in pending state
+
+ If Envoy is deployed with a Deployment and the number of envoy instances is not less than number of kubernetes nodes in the clusters, during rolling upgrade, new envoy pod will be stuck in pending stage because old envoy pod is occupying host port.
+
+ Workaround: Delete the envoy instance of older version manually. This will cause a little bit of downtime but it's pretty short.
+
# The easy way to upgrade
If the following are true for you:
@@ -525,7 +533,7 @@ If your version of Contour is older than v1.22.0, please upgrade to v1.22.0 firs
1. Update the Contour RBAC resources:
```bash
- $ kubectl apply -f examples/contour/02-rbac.yaml
+ $ kubectl apply -f examples/contour/02-rbac.yaml
$ kubectl apply -f examples/contour/02-role-contour.yaml
```
@@ -591,7 +599,7 @@ If your version of Contour is older than v1.21.3, please upgrade to v1.21.3 firs
1. Update the Contour RBAC resources:
```bash
- $ kubectl apply -f examples/contour/02-rbac.yaml
+ $ kubectl apply -f examples/contour/02-rbac.yaml
$ kubectl apply -f examples/contour/02-role-contour.yaml
```
@@ -790,7 +798,7 @@ If your version of Contour is older than v1.21.0, please upgrade to v1.21.0 firs
1. Update the Contour RBAC resources:
```bash
- $ kubectl apply -f examples/contour/02-rbac.yaml
+ $ kubectl apply -f examples/contour/02-rbac.yaml
$ kubectl apply -f examples/contour/02-role-contour.yaml
```
@@ -856,7 +864,7 @@ If your version of Contour is older than v1.20.2, please upgrade to v1.20.2 firs
1. Update the Contour RBAC resources:
```bash
- $ kubectl apply -f examples/contour/02-rbac.yaml
+ $ kubectl apply -f examples/contour/02-rbac.yaml
$ kubectl apply -f examples/contour/02-role-contour.yaml
```
From 080940778e4995aa310e8d5e9caab5a0f2747c08 Mon Sep 17 00:00:00 2001
From: Sunjay Bhatia <5337253+sunjayBhatia@users.noreply.github.com>
Date: Mon, 5 Feb 2024 16:03:29 -0500
Subject: [PATCH 06/83] Bump kind/kubectl and node images (#6160)
Signed-off-by: Sunjay Bhatia
---
.github/workflows/prbuild.yaml | 12 ++++++------
hack/actions/install-kubernetes-toolchain.sh | 4 ++--
test/scripts/make-kind-cluster.sh | 2 +-
3 files changed, 9 insertions(+), 9 deletions(-)
diff --git a/.github/workflows/prbuild.yaml b/.github/workflows/prbuild.yaml
index 623391e3add..05e72a9f3d9 100644
--- a/.github/workflows/prbuild.yaml
+++ b/.github/workflows/prbuild.yaml
@@ -140,11 +140,11 @@ jobs:
# image to use) for each kubernetes_version value.
include:
- kubernetes_version: "kubernetes:latest"
- node_image: "docker.io/kindest/node:v1.29.0@sha256:eaa1450915475849a73a9227b8f201df25e55e268e5d619312131292e324d570"
+ node_image: "docker.io/kindest/node:v1.29.1@sha256:a0cc28af37cf39b019e2b448c54d1a3f789de32536cb5a5db61a49623e527144"
- kubernetes_version: "kubernetes:n-1"
- node_image: "docker.io/kindest/node:v1.28.0@sha256:b7a4cad12c197af3ba43202d3efe03246b3f0793f162afb40a33c923952d5b31"
+ node_image: "docker.io/kindest/node:v1.28.6@sha256:b7e1cf6b2b729f604133c667a6be8aab6f4dde5bb042c1891ae248d9154f665b"
- kubernetes_version: "kubernetes:n-2"
- node_image: "docker.io/kindest/node:v1.27.3@sha256:3966ac761ae0136263ffdb6cfd4db23ef8a83cba8a463690e98317add2c9ba72"
+ node_image: "docker.io/kindest/node:v1.27.10@sha256:3700c811144e24a6c6181065265f69b9bf0b437c45741017182d7c82b908918f"
- config_type: "ConfigmapConfiguration"
use_config_crd: "false"
- config_type: "ContourConfiguration"
@@ -205,11 +205,11 @@ jobs:
# image to use) for each kubernetes_version value.
include:
- kubernetes_version: "kubernetes:latest"
- node_image: "docker.io/kindest/node:v1.29.0@sha256:eaa1450915475849a73a9227b8f201df25e55e268e5d619312131292e324d570"
+ node_image: "docker.io/kindest/node:v1.29.1@sha256:a0cc28af37cf39b019e2b448c54d1a3f789de32536cb5a5db61a49623e527144"
- kubernetes_version: "kubernetes:n-1"
- node_image: "docker.io/kindest/node:v1.28.0@sha256:b7a4cad12c197af3ba43202d3efe03246b3f0793f162afb40a33c923952d5b31"
+ node_image: "docker.io/kindest/node:v1.28.6@sha256:b7e1cf6b2b729f604133c667a6be8aab6f4dde5bb042c1891ae248d9154f665b"
- kubernetes_version: "kubernetes:n-2"
- node_image: "docker.io/kindest/node:v1.27.3@sha256:3966ac761ae0136263ffdb6cfd4db23ef8a83cba8a463690e98317add2c9ba72"
+ node_image: "docker.io/kindest/node:v1.27.10@sha256:3700c811144e24a6c6181065265f69b9bf0b437c45741017182d7c82b908918f"
steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1
with:
diff --git a/hack/actions/install-kubernetes-toolchain.sh b/hack/actions/install-kubernetes-toolchain.sh
index 78f4a78ca3e..d909a004286 100755
--- a/hack/actions/install-kubernetes-toolchain.sh
+++ b/hack/actions/install-kubernetes-toolchain.sh
@@ -4,8 +4,8 @@ set -o errexit
set -o nounset
set -o pipefail
-readonly KUBECTL_VERS="v1.29.0"
-readonly KIND_VERS="v0.20.0"
+readonly KUBECTL_VERS="v1.29.1"
+readonly KIND_VERS="v0.21.0"
readonly PROGNAME=$(basename $0)
readonly CURL=${CURL:-curl}
diff --git a/test/scripts/make-kind-cluster.sh b/test/scripts/make-kind-cluster.sh
index dc9fd75f8fc..d6fb9f9d80b 100755
--- a/test/scripts/make-kind-cluster.sh
+++ b/test/scripts/make-kind-cluster.sh
@@ -27,7 +27,7 @@ readonly KUBECTL=${KUBECTL:-kubectl}
readonly MULTINODE_CLUSTER=${MULTINODE_CLUSTER:-"false"}
readonly IPV6_CLUSTER=${IPV6_CLUSTER:-"false"}
readonly SKIP_GATEWAY_API_INSTALL=${SKIP_GATEWAY_API_INSTALL:-"false"}
-readonly NODEIMAGE=${NODEIMAGE:-"kindest/node:v1.29.0@sha256:eaa1450915475849a73a9227b8f201df25e55e268e5d619312131292e324d570"}
+readonly NODEIMAGE=${NODEIMAGE:-"kindest/node:v1.29.1@sha256:a0cc28af37cf39b019e2b448c54d1a3f789de32536cb5a5db61a49623e527144"}
readonly CLUSTERNAME=${CLUSTERNAME:-contour-e2e}
readonly WAITTIME=${WAITTIME:-5m}
From 0160b09078cc68d9873877a4cecb40e17d73657b Mon Sep 17 00:00:00 2001
From: Sunjay Bhatia <5337253+sunjayBhatia@users.noreply.github.com>
Date: Fri, 9 Feb 2024 13:50:02 -0500
Subject: [PATCH 07/83] site: Add troubleshooting doc for common errors (#6161)
This doc should contain steps to diagnose common errors,
focused on mainly HTTP req/responses
Signed-off-by: Sunjay Bhatia
---
.../unreleased/6161-sunjayBhatia-docs.md | 1 +
site/content/docs/main/troubleshooting.md | 30 +++---
.../troubleshooting/common-proxy-errors.md | 96 +++++++++++++++++++
site/data/docs/main-toc.yml | 2 +
4 files changed, 116 insertions(+), 13 deletions(-)
create mode 100644 changelogs/unreleased/6161-sunjayBhatia-docs.md
create mode 100644 site/content/docs/main/troubleshooting/common-proxy-errors.md
diff --git a/changelogs/unreleased/6161-sunjayBhatia-docs.md b/changelogs/unreleased/6161-sunjayBhatia-docs.md
new file mode 100644
index 00000000000..f2ce37e8298
--- /dev/null
+++ b/changelogs/unreleased/6161-sunjayBhatia-docs.md
@@ -0,0 +1 @@
+Add troubleshooting guide for general app traffic errors.
diff --git a/site/content/docs/main/troubleshooting.md b/site/content/docs/main/troubleshooting.md
index c2dfc570b2e..28461bd8641 100644
--- a/site/content/docs/main/troubleshooting.md
+++ b/site/content/docs/main/troubleshooting.md
@@ -2,22 +2,25 @@
If you encounter issues, follow the guides below for help. For topics not covered here, you can [file an issue][0], or talk to us on the [#contour channel][1] on Kubernetes Slack.
-### [Envoy Administration Access][2]
+### [Troubleshooting Common Proxy Errors][2]
+A guide on how to investigate common errors with Contour and Envoy.
+
+### [Envoy Administration Access][3]
Review the linked steps to learn how to access the administration interface for your Envoy instance.
-### [Contour Debug Logging][3]
+### [Contour Debug Logging][4]
Learn how to enable debug logging to diagnose issues between Contour and the Kubernetes API.
-### [Envoy Debug Logging][4]
+### [Envoy Debug Logging][5]
Learn how to enable debug logging to diagnose TLS connection issues.
-### [Visualize the Contour Graph][5]
+### [Visualize the Contour Graph][6]
Learn how to visualize Contour's internal object graph in [DOT][9] format, or as a png file.
-### [Show Contour xDS Resources][6]
+### [Show Contour xDS Resources][7]
Review the linked steps to view the [xDS][10] resource data exchanged by Contour and Envoy.
-### [Profiling Contour][7]
+### [Profiling Contour][8]
Learn how to profile Contour by using [net/http/pprof][11] handlers.
### [Envoy container stuck in unready/draining state][12]
@@ -25,13 +28,14 @@ Read the linked document if you have Envoy containers stuck in an unready/draini
[0]: {{< param github_url >}}/issues
[1]: {{< param slack_url >}}
-[2]: /docs/{{< param latest_version >}}/troubleshooting/envoy-admin-interface/
-[3]: /docs/{{< param latest_version >}}/troubleshooting/contour-debug-log/
-[4]: /docs/{{< param latest_version >}}/troubleshooting/envoy-debug-log/
-[5]: /docs/{{< param latest_version >}}/troubleshooting/contour-graph/
-[6]: /docs/{{< param latest_version >}}/troubleshooting/contour-xds-resources/
-[7]: /docs/{{< param latest_version >}}/troubleshooting/profiling-contour/
+[2]: /docs/{{< param version >}}/troubleshooting/common-proxy-errors/
+[3]: /docs/{{< param version >}}/troubleshooting/envoy-admin-interface/
+[4]: /docs/{{< param version >}}/troubleshooting/contour-debug-log/
+[5]: /docs/{{< param version >}}/troubleshooting/envoy-debug-log/
+[6]: /docs/{{< param version >}}/troubleshooting/contour-graph/
+[7]: /docs/{{< param version >}}/troubleshooting/contour-xds-resources/
+[8]: /docs/{{< param version >}}/troubleshooting/profiling-contour/
[9]: https://en.wikipedia.org/wiki/Dot
[10]: https://www.envoyproxy.io/docs/envoy/latest/api-docs/xds_protocol
[11]: https://golang.org/pkg/net/http/pprof/
-[12]: /docs/{{< param latest_version >}}/troubleshooting/envoy-container-draining/
\ No newline at end of file
+[12]: /docs/{{< param version >}}/troubleshooting/envoy-container-draining/
diff --git a/site/content/docs/main/troubleshooting/common-proxy-errors.md b/site/content/docs/main/troubleshooting/common-proxy-errors.md
new file mode 100644
index 00000000000..e05f153242d
--- /dev/null
+++ b/site/content/docs/main/troubleshooting/common-proxy-errors.md
@@ -0,0 +1,96 @@
+# Troubleshooting Common Proxy Errors
+
+## Unexpected HTTP errors
+
+Here are some steps to take in investigating common HTTP errors that users may encounter.
+We'll include example error cases to debug with these steps.
+
+1. Inspect the HTTP response in detail (possibly via `curl -v`).
+
+ Here we're looking to validate if the error response is coming from the backend app, Envoy, or possibly another proxy in front of Envoy.
+ If the response has the `server: envoy` header set, the request at least made it to the Envoy proxy so we can likely rule out anything before it.
+ The error may originate from Envoy itself or the backend app.
+ Look for headers or a response body that may originate from the backend app to verify if the error is in fact just the intended app behavior.
+ In the example below, we can see the response looks like it originates from Envoy, based on the `server: envoy` header and response body string.
+
+ ```
+ curl -vvv example.projectcontour.io
+ ...
+ > GET / HTTP/1.1
+ > Host: example.projectcontour.io
+ ...
+ >
+ < HTTP/1.1 503 Service Unavailable
+ < content-length: 91
+ < content-type: text/plain
+ < vary: Accept-Encoding
+ < date: Tue, 06 Feb 2024 03:44:30 GMT
+ < server: envoy
+ <
+ * Connection #0 to host example.projectcontour.io left intact
+ upstream connect error or disconnect/reset before headers. reset reason: connection failure
+ ```
+
+1. Look at the Envoy pod logs for the access logs corresponding to the erroring request/response.
+
+ The exact fields/field ordering present in the access log may vary if you have [configured a custom access log string or JSON access logs][0].
+ For example for a Contour installation using the [default Envoy access log format][1] we would want to inspect:
+ * `%REQ(:METHOD)%`, `%REQ(X-ENVOY-ORIGINAL-PATH?:PATH)%`, `%REQ(:AUTHORITY)%`, `%PROTOCOL%`: Ensure these are sensible values based on your configured route and HTTP request
+ * `%RESPONSE_FLAGS%`: See the [documentation on Envoy response flags][2] and below how to interpret a few of them in a Contour context:
+ * `UF`: Likely that Envoy could not connect to the upstream
+ * `UH`: Upstream Service has no health/ready pods
+ * `NR`: No configured route matching the request
+ * `%DURATION%`: Can correlate this with any configured timeouts
+ * `%RESP(X-ENVOY-UPSTREAM-SERVICE-TIME)%`: Can correlate this with any configured timeouts. If `-` then this is a hint the request was never forwarded to the upstream.
+ * `%UPSTREAM_HOST%`: This is the IP of the upstream pod that was selected to proxy the request to and can be used to verify the exact upstream instance that might be erroring.
+
+ For example in this access log:
+
+ ```
+ [2024-02-06T15:18:17.437Z] "GET / HTTP/1.1" 503 UF 0 91 1998 - "103.67.2.26" "curl/8.4.0" "d70640ec-2feb-46f8-9f63-24c44142c42e" "example.projectcontour.io" "10.244.8.27:8080"
+ ```
+
+ We can see the `UF` response flag as the cause of the `503` response code.
+ We also see the `-` for upstream request time.
+ It is likely in this case that Envoy was not able to establish a connection to the upstream.
+ That is further supported by the request duration of `1998` which is approximately the default upstream connection timeout of `2s`.
+
+1. Inspect Envoy metrics
+
+ This method of debugging can be useful especially for deployments that service a large volume of traffic.
+ In this case, access logs are possibly not suitable to use, as the volume of logs may be too large to pinpoint an exact erroring request.
+
+ Metrics from individual Envoy instances can be viewed manually or scraped using Envoy's prometheus endpoints and graphed using common visualization tools.
+ See the `/stats/prometheus` endpoint of the [Envoy admin interface][3].
+
+ Metrics that may be useful to inspect:
+ * [Listener metrics][4]
+ * `downstream_cx_total`
+ * `ssl.connection_error`
+ * [HTTP metrics][5]
+ * `downstream_cx_total`
+ * `downstream_cx_protocol_error`
+ * `downstream_rq_total`
+ * `downstream_rq_rx_reset`
+ * `downstream_rq_tx_reset`
+ * `downstream_rq_timeout`
+ * `downstream_rq_5xx` (and other status code groups)
+ * [Upstream metrics][6]
+ * `upstream_cx_total`
+ * `upstream_cx_connect_fail`
+ * `upstream_cx_connect_timeout`
+ * `upstream_rq_total`
+ * `upstream_rq_timeout`
+
+1. Send a direct request to the backend app to narrow down where the error may be originating.
+
+ This can be done via a port-forward to send a request to the app directly, skipping over the Envoy proxy.
+ If this sort of request succeeds, we know the issue likely originates from Contour configuration or the Envoy proxy rather than the app itself.
+
+[0]: /docs/{{< param latest_version >}}/config/access-logging/
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#default-format-string
+[2]: https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#config-access-log-format-response-flags
+[3]: /docs/{{< param latest_version >}}/guides/prometheus/#envoy-metrics
+[4]: https://www.envoyproxy.io/docs/envoy/latest/configuration/listeners/stats
+[5]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_conn_man/stats
+[6]: https://www.envoyproxy.io/docs/envoy/latest/configuration/upstream/cluster_manager/cluster_stats
diff --git a/site/data/docs/main-toc.yml b/site/data/docs/main-toc.yml
index 920070276ce..151db52d41a 100644
--- a/site/data/docs/main-toc.yml
+++ b/site/data/docs/main-toc.yml
@@ -101,6 +101,8 @@ toc:
url: /guides/resource-limits/
- title: Troubleshooting
subfolderitems:
+ - page: Troubleshooting Common Proxy Errors
+ url: /troubleshooting/common-proxy-errors
- page: Envoy Administration Access
url: /troubleshooting/envoy-admin-interface
- page: Contour Debug Logging
From 39a7d2002389792660c4ee7ffb2806b53e0c334d Mon Sep 17 00:00:00 2001
From: Lubron
Date: Fri, 9 Feb 2024 12:29:33 -0800
Subject: [PATCH 08/83] add disabled-features flag to ContourDeployment for
provisioner (#6152)
Fixes #5276.
Signed-off-by: Lubron Zhan
---
apis/projectcontour/v1/httpproxy.go | 3 +
apis/projectcontour/v1/httpproxy_helpers.go | 22 ---
.../v1alpha1/contourdeployment.go | 8 ++
.../unreleased/6152-lubronzhan-minor.md | 7 +
examples/contour/01-crds.yaml | 14 ++
examples/render/contour-deployment.yaml | 14 ++
.../render/contour-gateway-provisioner.yaml | 14 ++
examples/render/contour-gateway.yaml | 14 ++
examples/render/contour.yaml | 14 ++
internal/provisioner/controller/gateway.go | 5 +-
internal/provisioner/model/model.go | 23 ++-
.../provisioner/model/model_test.go | 41 +++++-
.../objects/deployment/deployment.go | 8 +-
.../objects/deployment/deployment_test.go | 40 +++++-
.../objects/rbac/clusterrole/cluster_role.go | 4 +-
.../rbac/clusterrole/cluster_role_test.go | 134 +++++++++++++++++-
internal/provisioner/objects/rbac/rbac.go | 7 +-
.../provisioner/objects/rbac/role/role.go | 2 +-
.../objects/rbac/role/role_test.go | 119 ++++++++++++++++
.../provisioner/objects/rbac/util/util.go | 45 +++++-
.../objects/rbac/util/util_test.go | 71 ++++++++++
.../docs/main/config/api-reference.html | 24 ++++
test/e2e/gatewayapi_predicates.go | 25 ++++
test/e2e/provisioner/provisioner_test.go | 128 ++++++++++++++++-
24 files changed, 733 insertions(+), 53 deletions(-)
delete mode 100644 apis/projectcontour/v1/httpproxy_helpers.go
create mode 100644 changelogs/unreleased/6152-lubronzhan-minor.md
rename apis/projectcontour/v1/httpproxy_helpers_test.go => internal/provisioner/model/model_test.go (52%)
create mode 100644 internal/provisioner/objects/rbac/util/util_test.go
diff --git a/apis/projectcontour/v1/httpproxy.go b/apis/projectcontour/v1/httpproxy.go
index b0a999fd921..1de37a7248e 100644
--- a/apis/projectcontour/v1/httpproxy.go
+++ b/apis/projectcontour/v1/httpproxy.go
@@ -1514,3 +1514,6 @@ type SlowStartPolicy struct {
// +kubebuilder:validation:Maximum=100
MinimumWeightPercent uint32 `json:"minWeightPercent"`
}
+
+// +kubebuilder:validation:Enum=grpcroutes;tlsroutes;extensionservices;backendtlspolicies
+type Feature string
diff --git a/apis/projectcontour/v1/httpproxy_helpers.go b/apis/projectcontour/v1/httpproxy_helpers.go
deleted file mode 100644
index 2716fa01738..00000000000
--- a/apis/projectcontour/v1/httpproxy_helpers.go
+++ /dev/null
@@ -1,22 +0,0 @@
-// Copyright Project Contour Authors
-// Licensed under the Apache License, Version 2.0 (the "License");
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-//
-// http://www.apache.org/licenses/LICENSE-2.0
-//
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an "AS IS" BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-
-package v1
-
-func NamespacesToStrings(ns []Namespace) []string {
- res := make([]string, len(ns))
- for i, n := range ns {
- res[i] = string(n)
- }
- return res
-}
diff --git a/apis/projectcontour/v1alpha1/contourdeployment.go b/apis/projectcontour/v1alpha1/contourdeployment.go
index 049a098df42..da534f384fa 100644
--- a/apis/projectcontour/v1alpha1/contourdeployment.go
+++ b/apis/projectcontour/v1alpha1/contourdeployment.go
@@ -131,6 +131,14 @@ type ContourSettings struct {
// +kubebuilder:validation:MinItems=1
// +kubebuilder:validation:MaxItems=42
WatchNamespaces []contour_api_v1.Namespace `json:"watchNamespaces,omitempty"`
+
+ // DisabledFeatures defines an array of resources that will be ignored by
+ // contour reconciler.
+ // +optional
+ // +kubebuilder:validation:Type=array
+ // +kubebuilder:validation:MinItems=1
+ // +kubebuilder:validation:MaxItems=42
+ DisabledFeatures []contour_api_v1.Feature `json:"disabledFeatures,omitempty"`
}
// DeploymentSettings contains settings for Deployment resources.
diff --git a/changelogs/unreleased/6152-lubronzhan-minor.md b/changelogs/unreleased/6152-lubronzhan-minor.md
new file mode 100644
index 00000000000..116189f08cc
--- /dev/null
+++ b/changelogs/unreleased/6152-lubronzhan-minor.md
@@ -0,0 +1,7 @@
+## Add DisabledFeatures to ContourDeployment for gateway provisioner
+
+A new flag DisabledFeatures is added to ContourDeployment so that user can configure contour which is deployed by the provisioner to skip reconciling CRDs which are specified inside the flag.
+
+Accepted values are `grpcroutes|tlsroutes|extensionservices|backendtlspolicies`.
+
+
diff --git a/examples/contour/01-crds.yaml b/examples/contour/01-crds.yaml
index 9de3a3bf2bb..6695ac9b884 100644
--- a/examples/contour/01-crds.yaml
+++ b/examples/contour/01-crds.yaml
@@ -1479,6 +1479,20 @@ spec:
type: string
type: object
type: object
+ disabledFeatures:
+ description: |-
+ DisabledFeatures defines an array of resources that will be ignored by
+ contour reconciler.
+ items:
+ enum:
+ - grpcroutes
+ - tlsroutes
+ - extensionservices
+ - backendtlspolicies
+ type: string
+ maxItems: 42
+ minItems: 1
+ type: array
kubernetesLogLevel:
description: |-
KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
diff --git a/examples/render/contour-deployment.yaml b/examples/render/contour-deployment.yaml
index d79153ad41b..f401e80ddeb 100644
--- a/examples/render/contour-deployment.yaml
+++ b/examples/render/contour-deployment.yaml
@@ -1698,6 +1698,20 @@ spec:
type: string
type: object
type: object
+ disabledFeatures:
+ description: |-
+ DisabledFeatures defines an array of resources that will be ignored by
+ contour reconciler.
+ items:
+ enum:
+ - grpcroutes
+ - tlsroutes
+ - extensionservices
+ - backendtlspolicies
+ type: string
+ maxItems: 42
+ minItems: 1
+ type: array
kubernetesLogLevel:
description: |-
KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
diff --git a/examples/render/contour-gateway-provisioner.yaml b/examples/render/contour-gateway-provisioner.yaml
index 9d6a8e51877..79a2d1a5830 100644
--- a/examples/render/contour-gateway-provisioner.yaml
+++ b/examples/render/contour-gateway-provisioner.yaml
@@ -1490,6 +1490,20 @@ spec:
type: string
type: object
type: object
+ disabledFeatures:
+ description: |-
+ DisabledFeatures defines an array of resources that will be ignored by
+ contour reconciler.
+ items:
+ enum:
+ - grpcroutes
+ - tlsroutes
+ - extensionservices
+ - backendtlspolicies
+ type: string
+ maxItems: 42
+ minItems: 1
+ type: array
kubernetesLogLevel:
description: |-
KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
diff --git a/examples/render/contour-gateway.yaml b/examples/render/contour-gateway.yaml
index 9bc2b73bdcc..ab80e2eaf68 100644
--- a/examples/render/contour-gateway.yaml
+++ b/examples/render/contour-gateway.yaml
@@ -1701,6 +1701,20 @@ spec:
type: string
type: object
type: object
+ disabledFeatures:
+ description: |-
+ DisabledFeatures defines an array of resources that will be ignored by
+ contour reconciler.
+ items:
+ enum:
+ - grpcroutes
+ - tlsroutes
+ - extensionservices
+ - backendtlspolicies
+ type: string
+ maxItems: 42
+ minItems: 1
+ type: array
kubernetesLogLevel:
description: |-
KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
diff --git a/examples/render/contour.yaml b/examples/render/contour.yaml
index 4b8212d47e4..73c2bf28dd4 100644
--- a/examples/render/contour.yaml
+++ b/examples/render/contour.yaml
@@ -1698,6 +1698,20 @@ spec:
type: string
type: object
type: object
+ disabledFeatures:
+ description: |-
+ DisabledFeatures defines an array of resources that will be ignored by
+ contour reconciler.
+ items:
+ enum:
+ - grpcroutes
+ - tlsroutes
+ - extensionservices
+ - backendtlspolicies
+ type: string
+ maxItems: 42
+ minItems: 1
+ type: array
kubernetesLogLevel:
description: |-
KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
diff --git a/internal/provisioner/controller/gateway.go b/internal/provisioner/controller/gateway.go
index f83c74f396f..c258fab9a44 100644
--- a/internal/provisioner/controller/gateway.go
+++ b/internal/provisioner/controller/gateway.go
@@ -17,7 +17,6 @@ import (
"context"
"fmt"
- contour_api_v1 "github.com/projectcontour/contour/apis/projectcontour/v1"
contour_api_v1alpha1 "github.com/projectcontour/contour/apis/projectcontour/v1alpha1"
"github.com/projectcontour/contour/internal/gatewayapi"
"github.com/projectcontour/contour/internal/provisioner/model"
@@ -262,7 +261,9 @@ func (r *gatewayReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ct
contourModel.Spec.KubernetesLogLevel = contourParams.KubernetesLogLevel
- contourModel.Spec.WatchNamespaces = contour_api_v1.NamespacesToStrings(contourParams.WatchNamespaces)
+ contourModel.Spec.WatchNamespaces = contourParams.WatchNamespaces
+
+ contourModel.Spec.DisabledFeatures = contourParams.DisabledFeatures
if contourParams.Deployment != nil &&
contourParams.Deployment.Strategy != nil {
diff --git a/internal/provisioner/model/model.go b/internal/provisioner/model/model.go
index 35541eb57bc..d6258ceb93a 100644
--- a/internal/provisioner/model/model.go
+++ b/internal/provisioner/model/model.go
@@ -14,6 +14,7 @@
package model
import (
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
contourv1alpha1 "github.com/projectcontour/contour/apis/projectcontour/v1alpha1"
"github.com/projectcontour/contour/internal/ref"
@@ -253,7 +254,27 @@ type ContourSpec struct {
// WatchNamespaces is an array of namespaces. Setting it will instruct the contour instance
// to only watch these set of namespaces
// default is nil, contour will watch resource of all namespaces
- WatchNamespaces []string
+ WatchNamespaces []contourv1.Namespace
+
+ // DisabledFeatures defines an array of resources that will be ignored by
+ // contour reconciler.
+ DisabledFeatures []contourv1.Feature
+}
+
+func NamespacesToStrings(ns []contourv1.Namespace) []string {
+ res := make([]string, len(ns))
+ for i, n := range ns {
+ res[i] = string(n)
+ }
+ return res
+}
+
+func FeaturesToStrings(fs []contourv1.Feature) []string {
+ res := make([]string, len(fs))
+ for i := range fs {
+ res[i] = string(fs[i])
+ }
+ return res
}
// WorkloadType is the type of Kubernetes workload to use for a component.
diff --git a/apis/projectcontour/v1/httpproxy_helpers_test.go b/internal/provisioner/model/model_test.go
similarity index 52%
rename from apis/projectcontour/v1/httpproxy_helpers_test.go
rename to internal/provisioner/model/model_test.go
index bacfaa55d4f..61800672bd9 100644
--- a/apis/projectcontour/v1/httpproxy_helpers_test.go
+++ b/internal/provisioner/model/model_test.go
@@ -11,27 +11,29 @@
// See the License for the specific language governing permissions and
// limitations under the License.
-package v1
+package model
import (
"reflect"
"testing"
+
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
)
func TestNamespacesToStrings(t *testing.T) {
testCases := []struct {
description string
- namespaces []Namespace
+ namespaces []contourv1.Namespace
expectStrings []string
}{
{
- description: "namespace 1",
- namespaces: []Namespace{},
+ description: "no namespaces",
+ namespaces: []contourv1.Namespace{},
expectStrings: []string{},
},
{
- description: "namespace 2",
- namespaces: []Namespace{"ns1", "ns2"},
+ description: "2 namespaces",
+ namespaces: []contourv1.Namespace{"ns1", "ns2"},
expectStrings: []string{"ns1", "ns2"},
},
}
@@ -44,3 +46,30 @@ func TestNamespacesToStrings(t *testing.T) {
})
}
}
+
+func TestFeaturesToStrings(t *testing.T) {
+ testCases := []struct {
+ description string
+ features []contourv1.Feature
+ expectStrings []string
+ }{
+ {
+ description: "no features",
+ features: []contourv1.Feature{},
+ expectStrings: []string{},
+ },
+ {
+ description: "2 features",
+ features: []contourv1.Feature{"tlsroutes", "grpcroutes"},
+ expectStrings: []string{"tlsroutes", "grpcroutes"},
+ },
+ }
+
+ for _, tc := range testCases {
+ t.Run(tc.description, func(t *testing.T) {
+ if !reflect.DeepEqual(FeaturesToStrings(tc.features), tc.expectStrings) {
+ t.Errorf("expect converted strings %v is the same as %v", FeaturesToStrings(tc.features), tc.expectStrings)
+ }
+ })
+ }
+}
diff --git a/internal/provisioner/objects/deployment/deployment.go b/internal/provisioner/objects/deployment/deployment.go
index bfc56116bcb..ec1700c6eb5 100644
--- a/internal/provisioner/objects/deployment/deployment.go
+++ b/internal/provisioner/objects/deployment/deployment.go
@@ -103,13 +103,17 @@ func DesiredDeployment(contour *model.Contour, image string) *appsv1.Deployment
}
if !contour.WatchAllNamespaces() {
- ns := contour.Spec.WatchNamespaces
- if !slices.Contains(contour.Spec.WatchNamespaces, contour.Namespace) {
+ ns := model.NamespacesToStrings(contour.Spec.WatchNamespaces)
+ if !slices.Contains(ns, contour.Namespace) {
ns = append(ns, contour.Namespace)
}
args = append(args, fmt.Sprintf("--watch-namespaces=%s", strings.Join(ns, ",")))
}
+ if contour.Spec.DisabledFeatures != nil && len(contour.Spec.DisabledFeatures) > 0 {
+ args = append(args, fmt.Sprintf("--disable-feature=%s", strings.Join(model.FeaturesToStrings(contour.Spec.DisabledFeatures), ",")))
+ }
+
// Pass the insecure/secure flags to Contour if using non-default ports.
for _, port := range contour.Spec.NetworkPublishing.Envoy.Ports {
switch {
diff --git a/internal/provisioner/objects/deployment/deployment_test.go b/internal/provisioner/objects/deployment/deployment_test.go
index 71d0c81be6f..9cf62aa29ed 100644
--- a/internal/provisioner/objects/deployment/deployment_test.go
+++ b/internal/provisioner/objects/deployment/deployment_test.go
@@ -18,6 +18,7 @@ import (
"strings"
"testing"
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
"github.com/projectcontour/contour/apis/projectcontour/v1alpha1"
"github.com/projectcontour/contour/internal/provisioner/model"
@@ -216,15 +217,15 @@ func TestDesiredDeployment(t *testing.T) {
func TestDesiredDeploymentWhenSettingWatchNamespaces(t *testing.T) {
testCases := []struct {
description string
- namespaces []string
+ namespaces []contourv1.Namespace
}{
{
description: "several valid namespaces",
- namespaces: []string{"ns1", "ns2"},
+ namespaces: []contourv1.Namespace{"ns1", "ns2"},
},
{
description: "single valid namespace",
- namespaces: []string{"ns1"},
+ namespaces: []contourv1.Namespace{"ns1"},
},
}
@@ -238,7 +239,7 @@ func TestDesiredDeploymentWhenSettingWatchNamespaces(t *testing.T) {
cntr.Spec.WatchNamespaces = tc.namespaces
deploy := DesiredDeployment(cntr, "ghcr.io/projectcontour/contour:test")
container := checkDeploymentHasContainer(t, deploy, contourContainerName, true)
- arg := fmt.Sprintf("--watch-namespaces=%s", strings.Join(append(tc.namespaces, cntr.Namespace), ","))
+ arg := fmt.Sprintf("--watch-namespaces=%s", strings.Join(append(model.NamespacesToStrings(tc.namespaces), cntr.Namespace), ","))
checkContainerHasArg(t, container, arg)
})
}
@@ -270,3 +271,34 @@ func TestNodePlacementDeployment(t *testing.T) {
checkDeploymentHasNodeSelector(t, deploy, selectors)
checkDeploymentHasTolerations(t, deploy, tolerations)
}
+
+func TestDesiredDeploymentWhenSettingDisabledFeature(t *testing.T) {
+ testCases := []struct {
+ description string
+ disabledFeatures []contourv1.Feature
+ }{
+ {
+ description: "disable 2 featuers",
+ disabledFeatures: []contourv1.Feature{"tlsroutes", "grpcroutes"},
+ },
+ {
+ description: "disable single feature",
+ disabledFeatures: []contourv1.Feature{"tlsroutes"},
+ },
+ }
+
+ for _, tc := range testCases {
+ t.Run(tc.description, func(t *testing.T) {
+ name := "deploy-test"
+ cntr := model.Default(fmt.Sprintf("%s-ns", name), name)
+ icName := "test-ic"
+ cntr.Spec.IngressClassName = &icName
+ cntr.Spec.DisabledFeatures = tc.disabledFeatures
+ // Change the Contour watch namespaces flag
+ deploy := DesiredDeployment(cntr, "ghcr.io/projectcontour/contour:test")
+ container := checkDeploymentHasContainer(t, deploy, contourContainerName, true)
+ arg := fmt.Sprintf("--disable-feature=%s", strings.Join(model.FeaturesToStrings(tc.disabledFeatures), ","))
+ checkContainerHasArg(t, container, arg)
+ })
+ }
+}
diff --git a/internal/provisioner/objects/rbac/clusterrole/cluster_role.go b/internal/provisioner/objects/rbac/clusterrole/cluster_role.go
index 6371a9202f2..700ca27c31f 100644
--- a/internal/provisioner/objects/rbac/clusterrole/cluster_role.go
+++ b/internal/provisioner/objects/rbac/clusterrole/cluster_role.go
@@ -58,8 +58,8 @@ func desiredClusterRole(name string, contour *model.Contour, clusterScopedResour
return role
}
- // add basic rules to role
- role.Rules = append(role.Rules, util.NamespacedResourcePolicyRules()...)
+ // add other rules for namespacedResources, so that we can associated them with ClusterRole later
+ role.Rules = append(role.Rules, util.NamespacedResourcePolicyRules(contour.Spec.DisabledFeatures)...)
return role
}
diff --git a/internal/provisioner/objects/rbac/clusterrole/cluster_role_test.go b/internal/provisioner/objects/rbac/clusterrole/cluster_role_test.go
index 2e48d3edd98..56fa4a8809f 100644
--- a/internal/provisioner/objects/rbac/clusterrole/cluster_role_test.go
+++ b/internal/provisioner/objects/rbac/clusterrole/cluster_role_test.go
@@ -19,9 +19,14 @@ import (
"testing"
"github.com/projectcontour/contour/internal/provisioner/model"
+ "github.com/projectcontour/contour/internal/provisioner/objects/rbac/util"
+ "github.com/projectcontour/contour/internal/provisioner/slice"
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
rbacv1 "k8s.io/api/rbac/v1"
apiequality "k8s.io/apimachinery/pkg/api/equality"
+ "k8s.io/utils/diff"
+ gatewayv1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
)
func checkClusterRoleName(t *testing.T, cr *rbacv1.ClusterRole, expected string) {
@@ -82,7 +87,6 @@ func TestDesiredClusterRole(t *testing.T) {
model.GatewayAPIOwningGatewayNameLabel: cntr.Name,
}
checkClusterRoleLabels(t, cr, ownerLabels)
- fmt.Println(cr.Rules)
if tc.clusterScopeOnly != clusterRoleRulesContainOnlyClusterScopeRules(cr) {
t.Errorf("expect clusterScopeOnly to be %v, but clusterRoleRulesContainOnlyClusterScopeRules shows %v",
tc.clusterScopeOnly, clusterRoleRulesContainOnlyClusterScopeRules(cr))
@@ -90,3 +94,131 @@ func TestDesiredClusterRole(t *testing.T) {
})
}
}
+
+func TestDesiredClusterRoleFilterResources(t *testing.T) {
+ filterNamespacedGatewayResources := func(policyRules []rbacv1.PolicyRule) [][]string {
+ gatewayResources := [][]string{}
+ for _, rule := range policyRules {
+ for _, apigroup := range rule.APIGroups {
+ // gatewayclass is in isolate rule
+ if apigroup == gatewayv1alpha2.GroupName && rule.Resources[0] != "gatewayclasses" && rule.Resources[0] != "gatewayclasses/status" {
+ gatewayResources = append(gatewayResources, rule.Resources)
+ break
+ }
+ }
+ }
+ return gatewayResources
+ }
+
+ filterContourResources := func(policyRules []rbacv1.PolicyRule) [][]string {
+ contourResources := [][]string{}
+ for _, rule := range policyRules {
+ for _, apigroup := range rule.APIGroups {
+ if apigroup == contourv1.GroupName {
+ contourResources = append(contourResources, rule.Resources)
+ break
+ }
+ }
+ }
+ return contourResources
+ }
+
+ tests := []struct {
+ description string
+ disabledFeatures []contourv1.Feature
+ clusterScopedResourceOnly bool
+ expectedGateway [][]string
+ expectedContour [][]string
+ }{
+ {
+ description: "empty disabled features",
+ disabledFeatures: nil,
+ clusterScopedResourceOnly: false,
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+ {
+ description: "disable tlsroutes feature",
+ disabledFeatures: []contourv1.Feature{"tlsroutes"},
+ clusterScopedResourceOnly: false,
+ expectedGateway: [][]string{
+ removeFromStringArray(util.GatewayGroupNamespacedResource, "tlsroutes"),
+ removeFromStringArray(util.GatewayGroupNamespacedResourceStatus, "tlsroutes/status"),
+ },
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+
+ {
+ description: "disable extensionservices feature",
+ disabledFeatures: []contourv1.Feature{"extensionservices"},
+ clusterScopedResourceOnly: false,
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{
+ removeFromStringArray(util.ContourGroupNamespacedResource, "extensionservices"),
+ removeFromStringArray(util.ContourGroupNamespacedResourceStatus, "extensionservices/status"),
+ },
+ },
+ {
+ description: "disable non-existent features",
+ disabledFeatures: []contourv1.Feature{"abc", "efg"},
+ clusterScopedResourceOnly: false,
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+ {
+ description: "disable both gateway and contour features",
+ disabledFeatures: []contourv1.Feature{"grpcroutes", "tlsroutes", "extensionservices", "backendtlspolicies"},
+ clusterScopedResourceOnly: false,
+ expectedGateway: [][]string{
+ removeFromStringArray(util.GatewayGroupNamespacedResource, "tlsroutes", "grpcroutes", "backendtlspolicies"),
+ removeFromStringArray(util.GatewayGroupNamespacedResourceStatus, "tlsroutes/status", "grpcroutes/status", "backendtlspolicies/status"),
+ },
+ expectedContour: [][]string{
+ removeFromStringArray(util.ContourGroupNamespacedResource, "extensionservices"),
+ removeFromStringArray(util.ContourGroupNamespacedResourceStatus, "extensionservices/status"),
+ },
+ },
+ {
+ description: "empty disabled features but with clusterScoped only",
+ disabledFeatures: nil,
+ clusterScopedResourceOnly: true,
+ expectedGateway: [][]string{},
+ expectedContour: [][]string{},
+ },
+ }
+
+ cntrName := "test-filteredresources"
+ cntr := model.Default(fmt.Sprintf("%s-ns", cntrName), cntrName)
+
+ for _, tt := range tests {
+ t.Run(tt.description, func(t *testing.T) {
+ cntrLocal := cntr
+
+ // set the disableFeatures
+ cntrLocal.Spec.DisabledFeatures = tt.disabledFeatures
+
+ cr := desiredClusterRole(cntrName, cntrLocal, tt.clusterScopedResourceOnly)
+
+ // fetch gateway resources
+ gatewayResources := filterNamespacedGatewayResources(cr.Rules)
+ contourResources := filterContourResources(cr.Rules)
+ if !apiequality.Semantic.DeepEqual(gatewayResources, tt.expectedGateway) {
+ t.Errorf("filtered gateway resources didn't match: %v", diff.ObjectReflectDiff(gatewayResources, tt.expectedGateway))
+ }
+
+ if !apiequality.Semantic.DeepEqual(contourResources, tt.expectedContour) {
+ t.Errorf("filtered contour resources didn't match: %v", diff.ObjectReflectDiff(contourResources, tt.expectedContour))
+ }
+ })
+ }
+}
+
+func removeFromStringArray(arr []string, s ...string) []string {
+ res := []string{}
+ for _, a := range arr {
+ if !slice.ContainsString(s, a) {
+ res = append(res, a)
+ }
+ }
+ return res
+}
diff --git a/internal/provisioner/objects/rbac/rbac.go b/internal/provisioner/objects/rbac/rbac.go
index 30b731a7e1d..271698c90ff 100644
--- a/internal/provisioner/objects/rbac/rbac.go
+++ b/internal/provisioner/objects/rbac/rbac.go
@@ -17,6 +17,7 @@ import (
"context"
"fmt"
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
"github.com/projectcontour/contour/internal/provisioner/model"
"github.com/projectcontour/contour/internal/provisioner/objects"
"github.com/projectcontour/contour/internal/provisioner/objects/rbac/clusterrole"
@@ -81,7 +82,7 @@ func ensureContourRBAC(ctx context.Context, cli client.Client, contour *model.Co
}
// includes contour's namespace if it's not inside watchNamespaces
- ns := contour.Spec.WatchNamespaces
+ ns := model.NamespacesToStrings(contour.Spec.WatchNamespaces)
if !slice.ContainsString(ns, contour.Namespace) {
ns = append(ns, contour.Namespace)
}
@@ -178,12 +179,12 @@ func EnsureRBACDeleted(ctx context.Context, cli client.Client, contour *model.Co
return nil
}
-func validateNamespacesExist(ctx context.Context, cli client.Client, ns []string) error {
+func validateNamespacesExist(ctx context.Context, cli client.Client, ns []contourv1.Namespace) error {
errs := []error{}
for _, n := range ns {
namespace := &corev1.Namespace{}
// Check if the namespace exists
- err := cli.Get(ctx, types.NamespacedName{Name: n}, namespace)
+ err := cli.Get(ctx, types.NamespacedName{Name: string(n)}, namespace)
if err != nil {
if apierrors.IsNotFound(err) {
errs = append(errs, fmt.Errorf("failed to find namespace %s in watchNamespace. Please make sure it exist", n))
diff --git a/internal/provisioner/objects/rbac/role/role.go b/internal/provisioner/objects/rbac/role/role.go
index f4dbd6b0a8d..83ab82085ce 100644
--- a/internal/provisioner/objects/rbac/role/role.go
+++ b/internal/provisioner/objects/rbac/role/role.go
@@ -108,7 +108,7 @@ func desiredRoleForResourceInNamespace(name, namespace string, contour *model.Co
Labels: contour.CommonLabels(),
Annotations: contour.CommonAnnotations(),
},
- Rules: util.NamespacedResourcePolicyRules(),
+ Rules: util.NamespacedResourcePolicyRules(contour.Spec.DisabledFeatures),
}
}
diff --git a/internal/provisioner/objects/rbac/role/role_test.go b/internal/provisioner/objects/rbac/role/role_test.go
index a42fa7b8712..8db45e5f41d 100644
--- a/internal/provisioner/objects/rbac/role/role_test.go
+++ b/internal/provisioner/objects/rbac/role/role_test.go
@@ -17,10 +17,15 @@ import (
"fmt"
"testing"
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
"github.com/projectcontour/contour/internal/provisioner/model"
+ "github.com/projectcontour/contour/internal/provisioner/objects/rbac/util"
+ "github.com/projectcontour/contour/internal/provisioner/slice"
+ gatewayv1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
rbacv1 "k8s.io/api/rbac/v1"
apiequality "k8s.io/apimachinery/pkg/api/equality"
+ "k8s.io/utils/diff"
)
func checkRoleName(t *testing.T, role *rbacv1.Role, expected string) {
@@ -95,3 +100,117 @@ func TestDesiredRoleForContourInNamespace(t *testing.T) {
})
}
}
+
+func TestDesiredRoleFilterResources(t *testing.T) {
+ filterNamespacedGatewayResources := func(policyRules []rbacv1.PolicyRule) [][]string {
+ gatewayResources := [][]string{}
+ for _, rule := range policyRules {
+ for _, apigroup := range rule.APIGroups {
+ if apigroup == gatewayv1alpha2.GroupName {
+ gatewayResources = append(gatewayResources, rule.Resources)
+ break
+ }
+ }
+ }
+ return gatewayResources
+ }
+
+ filterContourResources := func(policyRules []rbacv1.PolicyRule) [][]string {
+ contourResources := [][]string{}
+ for _, rule := range policyRules {
+ for _, apigroup := range rule.APIGroups {
+ if apigroup == contourv1.GroupName {
+ contourResources = append(contourResources, rule.Resources)
+ break
+ }
+ }
+ }
+ return contourResources
+ }
+
+ tests := []struct {
+ description string
+ disabledFeatures []contourv1.Feature
+ expectedGateway [][]string
+ expectedContour [][]string
+ }{
+ {
+ description: "empty disabled features",
+ disabledFeatures: nil,
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+ {
+ description: "disable tlsroutes feature",
+ disabledFeatures: []contourv1.Feature{"tlsroutes"},
+ expectedGateway: [][]string{
+ removeFromStringArray(util.GatewayGroupNamespacedResource, "tlsroutes"),
+ removeFromStringArray(util.GatewayGroupNamespacedResourceStatus, "tlsroutes/status"),
+ },
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+
+ {
+ description: "disable extensionservices feature",
+ disabledFeatures: []contourv1.Feature{"extensionservices"},
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{
+ removeFromStringArray(util.ContourGroupNamespacedResource, "extensionservices"),
+ removeFromStringArray(util.ContourGroupNamespacedResourceStatus, "extensionservices/status"),
+ },
+ },
+ {
+ description: "disable non-existent features",
+ disabledFeatures: []contourv1.Feature{"abc", "efg"},
+ expectedGateway: [][]string{util.GatewayGroupNamespacedResource, util.GatewayGroupNamespacedResourceStatus},
+ expectedContour: [][]string{util.ContourGroupNamespacedResource, util.ContourGroupNamespacedResourceStatus},
+ },
+ {
+ description: "disable both gateway and contour features",
+ disabledFeatures: []contourv1.Feature{"grpcroutes", "tlsroutes", "backendtlspolicies", "extensionservices"},
+ expectedGateway: [][]string{
+ removeFromStringArray(util.GatewayGroupNamespacedResource, "tlsroutes", "grpcroutes", "backendtlspolicies"),
+ removeFromStringArray(util.GatewayGroupNamespacedResourceStatus, "tlsroutes/status", "grpcroutes/status", "backendtlspolicies/status"),
+ },
+ expectedContour: [][]string{
+ removeFromStringArray(util.ContourGroupNamespacedResource, "extensionservices"),
+ removeFromStringArray(util.ContourGroupNamespacedResourceStatus, "extensionservices/status"),
+ },
+ },
+ }
+
+ cntrName := "test-filteredresources"
+ cntr := model.Default(fmt.Sprintf("%s-ns", cntrName), cntrName)
+
+ for _, tt := range tests {
+ t.Run(tt.description, func(t *testing.T) {
+ cntrLocal := cntr
+
+ // set the disableFeatures
+ cntrLocal.Spec.DisabledFeatures = tt.disabledFeatures
+
+ cr := desiredRoleForResourceInNamespace(cntrName, "test", cntrLocal)
+
+ // fetch gateway resources
+ gatewayResources := filterNamespacedGatewayResources(cr.Rules)
+ contourResources := filterContourResources(cr.Rules)
+ if !apiequality.Semantic.DeepEqual(gatewayResources, tt.expectedGateway) {
+ t.Errorf("filtered gateway resources didn't match: %v", diff.ObjectReflectDiff(gatewayResources, tt.expectedGateway))
+ }
+
+ if !apiequality.Semantic.DeepEqual(contourResources, tt.expectedContour) {
+ t.Errorf("filtered contour resources didn't match: %v", diff.ObjectReflectDiff(contourResources, tt.expectedContour))
+ }
+ })
+ }
+}
+
+func removeFromStringArray(arr []string, s ...string) []string {
+ res := []string{}
+ for _, a := range arr {
+ if !slice.ContainsString(s, a) {
+ res = append(res, a)
+ }
+ }
+ return res
+}
diff --git a/internal/provisioner/objects/rbac/util/util.go b/internal/provisioner/objects/rbac/util/util.go
index 698cb828144..1fb781f0d6d 100644
--- a/internal/provisioner/objects/rbac/util/util.go
+++ b/internal/provisioner/objects/rbac/util/util.go
@@ -14,6 +14,11 @@
package util
import (
+ "strings"
+
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
+ "github.com/projectcontour/contour/internal/provisioner/model"
+ "github.com/projectcontour/contour/internal/provisioner/slice"
corev1 "k8s.io/api/core/v1"
discoveryv1 "k8s.io/api/discovery/v1"
networkingv1 "k8s.io/api/networking/v1"
@@ -23,6 +28,13 @@ import (
const contourV1GroupName = "projectcontour.io"
+var (
+ GatewayGroupNamespacedResource = []string{"gateways", "httproutes", "tlsroutes", "grpcroutes", "tcproutes", "referencegrants", "backendtlspolicies"}
+ GatewayGroupNamespacedResourceStatus = []string{"gateways/status", "httproutes/status", "tlsroutes/status", "grpcroutes/status", "tcproutes/status", "backendtlspolicies/status"}
+ ContourGroupNamespacedResource = []string{"httpproxies", "tlscertificatedelegations", "extensionservices", "contourconfigurations"}
+ ContourGroupNamespacedResourceStatus = []string{"httpproxies/status", "extensionservices/status", "contourconfigurations/status"}
+)
+
var (
createGetUpdate = []string{"create", "get", "update"}
getListWatch = []string{"get", "list", "watch"}
@@ -39,8 +51,9 @@ func PolicyRuleFor(apiGroup string, verbs []string, resources ...string) rbacv1.
}
// NamespacedResourcePolicyRules returns a set of policy rules for resources that are
-// namespaced-scoped
-func NamespacedResourcePolicyRules() []rbacv1.PolicyRule {
+// namespaced-scoped. If resourcesToSkip is not empty, skip creating RBAC for those
+// CRDs.
+func NamespacedResourcePolicyRules(resourcesToSkip []contourv1.Feature) []rbacv1.PolicyRule {
return []rbacv1.PolicyRule{
// Core Contour-watched resources.
PolicyRuleFor(corev1.GroupName, getListWatch, "secrets", "endpoints", "services", "configmaps"),
@@ -50,16 +63,16 @@ func NamespacedResourcePolicyRules() []rbacv1.PolicyRule {
// Gateway API resources.
// Note, ReferenceGrant does not currently have a .status field so it's omitted from the status rule.
- PolicyRuleFor(gatewayv1alpha2.GroupName, getListWatch, "gateways", "httproutes", "tlsroutes", "grpcroutes", "tcproutes", "referencegrants", "backendtlspolicies"),
- PolicyRuleFor(gatewayv1alpha2.GroupName, update, "gateways/status", "httproutes/status", "tlsroutes/status", "grpcroutes/status", "tcproutes/status", "backendtlspolicies/status"),
+ PolicyRuleFor(gatewayv1alpha2.GroupName, getListWatch, filterResources(resourcesToSkip, GatewayGroupNamespacedResource...)...),
+ PolicyRuleFor(gatewayv1alpha2.GroupName, update, filterResources(resourcesToSkip, GatewayGroupNamespacedResourceStatus...)...),
// Ingress resources.
PolicyRuleFor(networkingv1.GroupName, getListWatch, "ingresses"),
PolicyRuleFor(networkingv1.GroupName, createGetUpdate, "ingresses/status"),
// Contour CRDs.
- PolicyRuleFor(contourV1GroupName, getListWatch, "httpproxies", "tlscertificatedelegations", "extensionservices", "contourconfigurations"),
- PolicyRuleFor(contourV1GroupName, createGetUpdate, "httpproxies/status", "extensionservices/status", "contourconfigurations/status"),
+ PolicyRuleFor(contourV1GroupName, getListWatch, filterResources(resourcesToSkip, ContourGroupNamespacedResource...)...),
+ PolicyRuleFor(contourV1GroupName, createGetUpdate, filterResources(resourcesToSkip, ContourGroupNamespacedResourceStatus...)...),
}
}
@@ -75,3 +88,23 @@ func ClusterScopedResourcePolicyRules() []rbacv1.PolicyRule {
PolicyRuleFor(corev1.GroupName, getListWatch, "namespaces"),
}
}
+
+func filterResources(resourcesToSkip []contourv1.Feature, resources ...string) []string {
+ if len(resourcesToSkip) == 0 {
+ return resources
+ }
+ filteredResources := []string{}
+ rts := model.FeaturesToStrings(resourcesToSkip)
+ for _, resource := range resources {
+ resourceCopy := resource
+ // handle status resources by splitting and using the first part
+ if strings.Contains(resourceCopy, "/") {
+ parts := strings.Split(resourceCopy, "/")
+ resourceCopy = parts[0]
+ }
+ if !slice.ContainsString(rts, resourceCopy) {
+ filteredResources = append(filteredResources, resource)
+ }
+ }
+ return filteredResources
+}
diff --git a/internal/provisioner/objects/rbac/util/util_test.go b/internal/provisioner/objects/rbac/util/util_test.go
new file mode 100644
index 00000000000..c6218761e07
--- /dev/null
+++ b/internal/provisioner/objects/rbac/util/util_test.go
@@ -0,0 +1,71 @@
+// Copyright Project Contour Authors
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package util
+
+import (
+ "reflect"
+ "testing"
+
+ contourv1 "github.com/projectcontour/contour/apis/projectcontour/v1"
+)
+
+func TestFilterResources(t *testing.T) {
+ testCases := []struct {
+ description string
+ disabledFeatures []contourv1.Feature
+ resourceList []string
+ expectedList []string
+ }{
+ {
+ description: "empty disabled features",
+ resourceList: []string{"httpproxies", "tlscertificatedelegations", "extensionservices", "contourconfigurations"},
+ disabledFeatures: nil,
+ expectedList: []string{"httpproxies", "tlscertificatedelegations", "extensionservices", "contourconfigurations"},
+ },
+ {
+ description: "disable extensionservices",
+ resourceList: []string{"httpproxies", "tlscertificatedelegations", "extensionservices", "contourconfigurations"},
+ disabledFeatures: []contourv1.Feature{"extensionservices"},
+ expectedList: []string{"httpproxies", "tlscertificatedelegations", "contourconfigurations"},
+ },
+ {
+ description: "disable extensionservices, filter status",
+ resourceList: []string{"httpproxies/status", "extensionservices/status", "contourconfigurations/status"},
+ disabledFeatures: []contourv1.Feature{"extensionservices"},
+ expectedList: []string{"httpproxies/status", "contourconfigurations/status"},
+ },
+ {
+ description: "disable tlsroutes",
+ resourceList: []string{"gateways", "httproutes", "tlsroutes", "grpcroutes", "tcproutes", "referencegrants"},
+ disabledFeatures: []contourv1.Feature{"tlsroutes"},
+ expectedList: []string{"gateways", "httproutes", "grpcroutes", "tcproutes", "referencegrants"},
+ },
+ {
+ description: "disable non-existence abc",
+ resourceList: []string{"gateways", "httproutes", "tlsroutes", "grpcroutes", "tcproutes", "referencegrants"},
+ disabledFeatures: []contourv1.Feature{"abc"},
+ expectedList: []string{"gateways", "httproutes", "tlsroutes", "grpcroutes", "tcproutes", "referencegrants"},
+ },
+ }
+
+ for _, tc := range testCases {
+ t.Run(tc.description, func(t *testing.T) {
+ f := filterResources(tc.disabledFeatures, tc.resourceList...)
+ if !reflect.DeepEqual(tc.expectedList, f) {
+ t.Errorf("expect filtered list to be %v, but is %v",
+ tc.expectedList, f)
+ }
+ })
+ }
+}
diff --git a/site/content/docs/main/config/api-reference.html b/site/content/docs/main/config/api-reference.html
index dde8bb108e2..522e89d16a8 100644
--- a/site/content/docs/main/config/api-reference.html
+++ b/site/content/docs/main/config/api-reference.html
@@ -1176,6 +1176,14 @@ ExtensionServiceReferenc
+Feature
+(string
alias)
+
+(Appears on:
+ContourSettings)
+
+
+
GenericKeyDescriptor
@@ -6284,6 +6292,22 @@
ContourSettings
to only watch this subset of namespaces.
+
+
+disabledFeatures
+
+
+
+[]Feature
+
+
+ |
+
+(Optional)
+ DisabledFeatures defines an array of resources that will be ignored by
+contour reconciler.
+ |
+
CustomTag
diff --git a/test/e2e/gatewayapi_predicates.go b/test/e2e/gatewayapi_predicates.go
index aa1946ad860..67556328ce0 100644
--- a/test/e2e/gatewayapi_predicates.go
+++ b/test/e2e/gatewayapi_predicates.go
@@ -145,6 +145,31 @@ func TCPRouteAccepted(route *gatewayapi_v1alpha2.TCPRoute) bool {
return false
}
+// TLSRouteIgnoredByContour returns true if the route has an empty .status.parents.conditions list
+func TLSRouteIgnoredByContour(route *gatewayapi_v1alpha2.TLSRoute) bool {
+ if route == nil {
+ return false
+ }
+
+ return len(route.Status.Parents) == 0
+}
+
+// TLSRouteAccepted returns true if the route has a .status.conditions
+// entry of "Accepted: true".
+func TLSRouteAccepted(route *gatewayapi_v1alpha2.TLSRoute) bool {
+ if route == nil {
+ return false
+ }
+
+ for _, gw := range route.Status.Parents {
+ if conditionExists(gw.Conditions, string(gatewayapi_v1alpha2.RouteConditionAccepted), metav1.ConditionTrue) {
+ return true
+ }
+ }
+
+ return false
+}
+
// BackendTLSPolicyAccepted returns true if the backend TLS policy has a .status.conditions
// entry of "Accepted: true".
func BackendTLSPolicyAccepted(btp *gatewayapi_v1alpha2.BackendTLSPolicy) bool {
diff --git a/test/e2e/provisioner/provisioner_test.go b/test/e2e/provisioner/provisioner_test.go
index f1c24740885..9390eb16bab 100644
--- a/test/e2e/provisioner/provisioner_test.go
+++ b/test/e2e/provisioner/provisioner_test.go
@@ -34,6 +34,7 @@ import (
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/utils/ptr"
"sigs.k8s.io/controller-runtime/pkg/client"
gatewayapi_v1 "sigs.k8s.io/gateway-api/apis/v1"
gatewayapi_v1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
@@ -540,7 +541,7 @@ var _ = Describe("Gateway provisioner", func() {
params := &contour_api_v1alpha1.ContourDeployment{
ObjectMeta: metav1.ObjectMeta{
Namespace: namespace,
- Name: "contour-params-with-watch-namespaces",
+ Name: objectTestName,
},
Spec: contour_api_v1alpha1.ContourDeploymentSpec{
RuntimeSettings: contourDeploymentRuntimeSettings(),
@@ -669,6 +670,7 @@ var _ = Describe("Gateway provisioner", func() {
// Root proxy in non-watched namespace should fail
By(fmt.Sprintf("Expect namespace %s not to be watched by contour", t.namespace))
hr, ok := f.CreateHTTPRouteAndWaitFor(route, e2e.HTTPRouteIgnoredByContour)
+ require.True(f.T(), ok, fmt.Sprintf("httproute's is %v", hr))
By(fmt.Sprintf("Expect httproute under namespace %s is not accepted for a period of time", t.namespace))
require.Never(f.T(), func() bool {
@@ -677,12 +679,132 @@ var _ = Describe("Gateway provisioner", func() {
return false
}
return e2e.HTTPRouteAccepted(hr)
- }, 10*time.Second, time.Second, hr)
- require.True(f.T(), ok, fmt.Sprintf("httproute's is %v", hr))
+ }, 20*time.Second, time.Second, hr)
}
}
})
}, "testns-1", "testns-2", "testns-3")
+ f.NamespacedTest("gateway-with-envoy-with-disabled-features", func(namespace string) {
+ objectTestName := "contour-params-with-disabled-features"
+ BeforeEach(func() {
+ By("create gatewayclass that reference contourDeployment with disabled-features value")
+ gatewayClass := &gatewayapi_v1beta1.GatewayClass{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: objectTestName,
+ },
+ Spec: gatewayapi_v1beta1.GatewayClassSpec{
+ ControllerName: gatewayapi_v1beta1.GatewayController("projectcontour.io/gateway-controller"),
+ ParametersRef: &gatewayapi_v1beta1.ParametersReference{
+ Group: "projectcontour.io",
+ Kind: "ContourDeployment",
+ Namespace: ref.To(gatewayapi_v1beta1.Namespace(namespace)),
+ Name: objectTestName,
+ },
+ },
+ }
+ _, ok := f.CreateGatewayClassAndWaitFor(gatewayClass, e2e.GatewayClassNotAccepted)
+ require.True(f.T(), ok)
+
+ // Now create the ContourDeployment to match the parametersRef.
+ params := &contour_api_v1alpha1.ContourDeployment{
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: namespace,
+ Name: objectTestName,
+ },
+ Spec: contour_api_v1alpha1.ContourDeploymentSpec{
+ RuntimeSettings: contourDeploymentRuntimeSettings(),
+ Contour: &contour_api_v1alpha1.ContourSettings{
+ DisabledFeatures: []contour_api_v1.Feature{"tlsroutes"},
+ },
+ },
+ }
+ require.NoError(f.T(), f.Client.Create(context.Background(), params))
+
+ // Now the GatewayClass should be accepted.
+ require.Eventually(f.T(), func() bool {
+ gc := &gatewayapi_v1beta1.GatewayClass{}
+ if err := f.Client.Get(context.Background(), k8s.NamespacedNameOf(gatewayClass), gc); err != nil {
+ return false
+ }
+
+ return e2e.GatewayClassAccepted(gc)
+ }, time.Minute, time.Second)
+ })
+ AfterEach(func() {
+ require.NoError(f.T(), f.DeleteGatewayClass(&gatewayapi_v1beta1.GatewayClass{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: objectTestName,
+ },
+ }, false))
+ })
+ Specify("A gateway can be provisioned that ignore CRDs in disabledFeatures", func() {
+ By("Deploy gateway that referencing above gatewayclass")
+ gateway := &gatewayapi_v1beta1.Gateway{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "tlsroute",
+ Namespace: namespace,
+ },
+ Spec: gatewayapi_v1beta1.GatewaySpec{
+ GatewayClassName: gatewayapi_v1beta1.ObjectName(objectTestName),
+ Listeners: []gatewayapi_v1beta1.Listener{
+ {
+ Name: "https",
+ Protocol: gatewayapi_v1.TLSProtocolType,
+ Port: gatewayapi_v1beta1.PortNumber(443),
+ TLS: &gatewayapi_v1beta1.GatewayTLSConfig{
+ Mode: ptr.To(gatewayapi_v1.TLSModePassthrough),
+ },
+ AllowedRoutes: &gatewayapi_v1beta1.AllowedRoutes{
+ Namespaces: &gatewayapi_v1beta1.RouteNamespaces{
+ From: ref.To(gatewayapi_v1.NamespacesFromSame),
+ },
+ },
+ },
+ },
+ },
+ }
+
+ gateway, ok := f.CreateGatewayAndWaitFor(gateway, func(gw *gatewayapi_v1beta1.Gateway) bool {
+ return e2e.GatewayProgrammed(gw) && e2e.GatewayHasAddress(gw)
+ })
+ require.True(f.T(), ok, fmt.Sprintf("gateway is %v", gateway))
+
+ By("Skip reconciling the TLSRoute if disabledFeatures includes it")
+ f.Fixtures.EchoSecure.Deploy(namespace, "echo-secure", nil)
+ route := &gatewayapi_v1alpha2.TLSRoute{
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: namespace,
+ Name: "tlsroute-1",
+ },
+ Spec: gatewayapi_v1alpha2.TLSRouteSpec{
+ Hostnames: []gatewayapi_v1alpha2.Hostname{"provisioner.projectcontour.io"},
+ CommonRouteSpec: gatewayapi_v1beta1.CommonRouteSpec{
+ ParentRefs: []gatewayapi_v1alpha2.ParentReference{
+ {
+ Namespace: ref.To(gatewayapi_v1alpha2.Namespace(gateway.Namespace)),
+ Name: gatewayapi_v1alpha2.ObjectName(gateway.Name),
+ },
+ },
+ },
+ Rules: []gatewayapi_v1alpha2.TLSRouteRule{
+ {
+ BackendRefs: gatewayapi.TLSRouteBackendRef("echo-secure", 443, ref.To(int32(1))),
+ },
+ },
+ },
+ }
+ tr, ok := f.CreateTLSRouteAndWaitFor(route, e2e.TLSRouteIgnoredByContour)
+ require.True(f.T(), ok, fmt.Sprintf("tlsroute's is %v", tr))
+ By("Expect tlsroute not to be accepted")
+ require.Never(f.T(), func() bool {
+ tr = &gatewayapi_v1alpha2.TLSRoute{}
+ if err := f.Client.Get(context.Background(), k8s.NamespacedNameOf(tr), tr); err != nil {
+ return false
+ }
+ return e2e.TLSRouteAccepted(tr)
+ }, 20*time.Second, time.Second, tr)
+ })
+ })
})
func contourDeploymentRuntimeSettings() *contour_api_v1alpha1.ContourConfigurationSpec {
From 7d6831b911b2286488b2f1099d772e1d83787a8e Mon Sep 17 00:00:00 2001
From: Christian Ang
Date: Mon, 12 Feb 2024 06:09:22 -0800
Subject: [PATCH 09/83] Add Accepted condition to BackendTLSPolicy (#6151)
Signed-off-by: Christian Ang
---
.../unreleased/6151-christianang-small.md | 1 +
internal/dag/gatewayapi_processor.go | 192 +++---
internal/dag/status_test.go | 568 ++++++++++++++++++
internal/status/backendtlspolicyconditions.go | 151 +++++
.../status/backendtlspolicyconditions_test.go | 135 +++++
internal/status/cache.go | 61 +-
test/e2e/gateway/backend_tls_policy_test.go | 3 +-
test/e2e/gatewayapi_predicates.go | 14 +-
8 files changed, 1042 insertions(+), 83 deletions(-)
create mode 100644 changelogs/unreleased/6151-christianang-small.md
create mode 100644 internal/status/backendtlspolicyconditions.go
create mode 100644 internal/status/backendtlspolicyconditions_test.go
diff --git a/changelogs/unreleased/6151-christianang-small.md b/changelogs/unreleased/6151-christianang-small.md
new file mode 100644
index 00000000000..8a7c85b38bf
--- /dev/null
+++ b/changelogs/unreleased/6151-christianang-small.md
@@ -0,0 +1 @@
+For Gateway API, add "Accepted" condition to BackendTLSPolicy. If the condition is true the BackendTLSPolicy was accepted by the Gateway and if false a reason will be stated on the policy as to why it wasn't accepted.
diff --git a/internal/dag/gatewayapi_processor.go b/internal/dag/gatewayapi_processor.go
index 5c78b3993a3..0a2f8cd6957 100644
--- a/internal/dag/gatewayapi_processor.go
+++ b/internal/dag/gatewayapi_processor.go
@@ -262,7 +262,7 @@ func (p *GatewayAPIProcessor) processRoute(
switch route := route.(type) {
case *gatewayapi_v1beta1.HTTPRoute:
- p.computeHTTPRouteForListener(route, routeParentStatus, listener, hosts)
+ p.computeHTTPRouteForListener(route, routeParentStatus, routeParentRef, listener, hosts)
case *gatewayapi_v1alpha2.TLSRoute:
p.computeTLSRouteForListener(route, routeParentStatus, listener, hosts)
case *gatewayapi_v1alpha2.GRPCRoute:
@@ -1147,6 +1147,7 @@ func parseHTTPRouteTimeouts(httpRouteTimeouts *gatewayapi_v1.HTTPRouteTimeouts)
func (p *GatewayAPIProcessor) computeHTTPRouteForListener(
route *gatewayapi_v1beta1.HTTPRoute,
routeAccessor *status.RouteParentStatusUpdate,
+ routeParentRef gatewayapi_v1beta1.ParentReference,
listener *listenerInfo,
hosts sets.Set[string],
) {
@@ -1407,7 +1408,7 @@ func (p *GatewayAPIProcessor) computeHTTPRouteForListener(
)
} else {
// Get clusters from rule backendRefs
- clusters, totalWeight, ok := p.httpClusters(route.Namespace, rule.BackendRefs, routeAccessor)
+ clusters, totalWeight, ok := p.httpClusters(route.Namespace, rule.BackendRefs, routeAccessor, routeParentRef)
if !ok {
continue
}
@@ -1963,7 +1964,7 @@ func gatewayQueryParamMatchConditions(matches []gatewayapi_v1beta1.HTTPQueryPara
}
// httpClusters builds clusters from backendRef.
-func (p *GatewayAPIProcessor) httpClusters(routeNamespace string, backendRefs []gatewayapi_v1beta1.HTTPBackendRef, routeAccessor *status.RouteParentStatusUpdate) ([]*Cluster, uint32, bool) {
+func (p *GatewayAPIProcessor) httpClusters(routeNamespace string, backendRefs []gatewayapi_v1beta1.HTTPBackendRef, routeAccessor *status.RouteParentStatusUpdate, routeParentRef gatewayapi_v1beta1.ParentReference) ([]*Cluster, uint32, bool) {
totalWeight := uint32(0)
if len(backendRefs) == 0 {
@@ -1981,73 +1982,9 @@ func (p *GatewayAPIProcessor) httpClusters(routeNamespace string, backendRefs []
continue
}
- var upstreamValidation *PeerValidationContext
- var backendRefGroup gatewayapi_v1alpha2.Group
- if backendRef.Group != nil {
- backendRefGroup = *backendRef.Group
- }
-
- var backendRefKind gatewayapi_v1alpha2.Kind
- if backendRef.Kind != nil {
- backendRefKind = *backendRef.Kind
- }
-
- var backendNamespace *gatewayapi_v1.Namespace
- if backendRef.Namespace != nil && *backendRef.Namespace != "" {
- backendNamespace = backendRef.Namespace
- } else {
- backendNamespace = ptr.To(gatewayapi_v1.Namespace(routeNamespace))
- }
-
- policyTargetRef := gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
- PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
- Group: backendRefGroup,
- Kind: backendRefKind,
- Name: backendRef.Name,
- Namespace: backendNamespace,
- },
- SectionName: ptr.To(gatewayapi_v1alpha2.SectionName(service.Weighted.ServicePort.Name)),
- }
-
- var upstreamTLS *UpstreamTLS
- // Check to see if there is any BackendTLSPolicy matching this service and service port
- backendTLSPolicy, found := p.source.LookupBackendTLSPolicyByTargetRef(policyTargetRef)
- if found {
- var caSecrets []*Secret
- for _, certRef := range backendTLSPolicy.Spec.TLS.CACertRefs {
- switch certRef.Kind {
- case "Secret":
- caSecret, err := p.source.LookupCASecret(types.NamespacedName{
- Name: string(certRef.Name),
- Namespace: backendTLSPolicy.Namespace,
- }, backendTLSPolicy.Namespace)
- if err != nil {
- continue
- }
- caSecrets = append(caSecrets, caSecret)
- case "ConfigMap":
- caSecret, err := p.source.LookupCAConfigMap(types.NamespacedName{
- Name: string(certRef.Name),
- Namespace: backendTLSPolicy.Namespace,
- })
- if err != nil {
- continue
- }
- caSecrets = append(caSecrets, caSecret)
- default:
- continue
- }
- }
-
- if len(caSecrets) != 0 {
- upstreamValidation = &PeerValidationContext{
- CACertificates: caSecrets,
- SubjectNames: []string{string(backendTLSPolicy.Spec.TLS.Hostname)},
- }
-
- service.Protocol = "tls"
- upstreamTLS = p.UpstreamTLS
- }
+ upstreamValidation, upstreamTLS := p.computeBackendTLSPolicies(routeNamespace, backendRef, service, routeParentRef)
+ if upstreamValidation != nil {
+ service.Protocol = "tls"
}
var clusterRequestHeaderPolicy *HeadersPolicy
@@ -2118,6 +2055,121 @@ func (p *GatewayAPIProcessor) httpClusters(routeNamespace string, backendRefs []
return clusters, totalWeight, true
}
+// computeBackendTLSPolicies returns the upstreamValidation and upstreamTLS
+// fields for the cluster that is being calculated if there is an associated
+// BackendTLSPolicy for the service being referenced.
+//
+// If no BackendTLSPolicy is found or the BackendTLSPolicy is invalid then nil
+// is returned for both fields.
+func (p *GatewayAPIProcessor) computeBackendTLSPolicies(routeNamespace string, backendRef gatewayapi_v1beta1.HTTPBackendRef, service *Service, routeParentRef gatewayapi_v1beta1.ParentReference) (*PeerValidationContext, *UpstreamTLS) {
+ var upstreamValidation *PeerValidationContext
+ var upstreamTLS *UpstreamTLS
+
+ var backendRefGroup gatewayapi_v1alpha2.Group
+ if backendRef.Group != nil {
+ backendRefGroup = *backendRef.Group
+ }
+
+ var backendRefKind gatewayapi_v1alpha2.Kind
+ if backendRef.Kind != nil {
+ backendRefKind = *backendRef.Kind
+ }
+
+ var backendNamespace *gatewayapi_v1.Namespace
+ if backendRef.Namespace != nil && *backendRef.Namespace != "" {
+ backendNamespace = backendRef.Namespace
+ } else {
+ backendNamespace = ptr.To(gatewayapi_v1.Namespace(routeNamespace))
+ }
+
+ policyTargetRef := gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Group: backendRefGroup,
+ Kind: backendRefKind,
+ Name: backendRef.Name,
+ Namespace: backendNamespace,
+ },
+ SectionName: ptr.To(gatewayapi_v1alpha2.SectionName(service.Weighted.ServicePort.Name)),
+ }
+
+ // Check to see if there is any BackendTLSPolicy matching this service and service port
+ backendTLSPolicy, found := p.source.LookupBackendTLSPolicyByTargetRef(policyTargetRef)
+ if found {
+ backendTLSPolicyAccessor, commit := p.dag.StatusCache.BackendTLSPolicyConditionsAccessor(
+ k8s.NamespacedNameOf(backendTLSPolicy),
+ backendTLSPolicy.GetGeneration(),
+ backendTLSPolicy,
+ )
+ defer commit()
+ backendTLSPolicyAncestorStatus := backendTLSPolicyAccessor.StatusUpdateFor(routeParentRef)
+
+ if backendTLSPolicy.Spec.TLS.WellKnownCACerts != nil && *backendTLSPolicy.Spec.TLS.WellKnownCACerts != "" {
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, "BackendTLSPolicy.Spec.TLS.WellKnownCACerts is unsupported.")
+ return nil, nil
+ }
+
+ if err := gatewayapi.IsValidHostname(string(backendTLSPolicy.Spec.TLS.Hostname)); err != nil {
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, fmt.Sprintf("BackendTLSPolicy.Spec.TLS.Hostname %q is invalid. Hostname must be a valid RFC 1123 fully qualified domain name. Wildcard domains and numeric IP addresses are not allowed", backendTLSPolicy.Spec.TLS.Hostname))
+ return nil, nil
+ }
+
+ if strings.Contains(string(backendTLSPolicy.Spec.TLS.Hostname), "*") {
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, fmt.Sprintf("BackendTLSPolicy.Spec.TLS.Hostname %q is invalid. Hostname must be a valid RFC 1123 fully qualified domain name. Wildcard domains and numeric IP addresses are not allowed", backendTLSPolicy.Spec.TLS.Hostname))
+ return nil, nil
+ }
+
+ var isInvalidCertChain bool
+ var caSecrets []*Secret
+ for _, certRef := range backendTLSPolicy.Spec.TLS.CACertRefs {
+ switch certRef.Kind {
+ case "Secret":
+ caSecret, err := p.source.LookupCASecret(types.NamespacedName{
+ Name: string(certRef.Name),
+ Namespace: backendTLSPolicy.Namespace,
+ }, backendTLSPolicy.Namespace)
+ if err != nil {
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, fmt.Sprintf("Could not find CACertRef Secret: %s/%s", backendTLSPolicy.Namespace, certRef.Name))
+ isInvalidCertChain = true
+ continue
+ }
+ caSecrets = append(caSecrets, caSecret)
+ case "ConfigMap":
+ caSecret, err := p.source.LookupCAConfigMap(types.NamespacedName{
+ Name: string(certRef.Name),
+ Namespace: backendTLSPolicy.Namespace,
+ })
+ if err != nil {
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, fmt.Sprintf("Could not find CACertRef ConfigMap: %s/%s", backendTLSPolicy.Namespace, certRef.Name))
+ isInvalidCertChain = true
+ continue
+ }
+ caSecrets = append(caSecrets, caSecret)
+ default:
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionFalse, gatewayapi_v1alpha2.PolicyReasonInvalid, fmt.Sprintf("BackendTLSPolicy.Spec.TLS.CACertRef.Kind %q is unsupported. Only ConfigMap or Secret Kind is supported.", certRef.Kind))
+ isInvalidCertChain = true
+ continue
+ }
+ }
+
+ if isInvalidCertChain {
+ return nil, nil
+ }
+
+ if len(caSecrets) != 0 {
+ upstreamValidation = &PeerValidationContext{
+ CACertificates: caSecrets,
+ SubjectNames: []string{string(backendTLSPolicy.Spec.TLS.Hostname)},
+ }
+
+ upstreamTLS = p.UpstreamTLS
+
+ backendTLSPolicyAncestorStatus.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionTrue, gatewayapi_v1alpha2.PolicyReasonAccepted, "Accepted BackendTLSPolicy")
+ }
+ }
+
+ return upstreamValidation, upstreamTLS
+}
+
// grpcClusters builds clusters from backendRef.
func (p *GatewayAPIProcessor) grpcClusters(routeNamespace string, backendRefs []gatewayapi_v1alpha2.GRPCBackendRef, routeAccessor *status.RouteParentStatusUpdate, protocolType gatewayapi_v1beta1.ProtocolType) ([]*Cluster, uint32, bool) {
totalWeight := uint32(0)
diff --git a/internal/dag/status_test.go b/internal/dag/status_test.go
index 13d55b31540..29561c6a2dc 100644
--- a/internal/dag/status_test.go
+++ b/internal/dag/status_test.go
@@ -32,6 +32,7 @@ import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
"k8s.io/apimachinery/pkg/util/intstr"
+ "k8s.io/utils/ptr"
gatewayapi_v1 "sigs.k8s.io/gateway-api/apis/v1"
gatewayapi_v1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
gatewayapi_v1beta1 "sigs.k8s.io/gateway-api/apis/v1beta1"
@@ -11390,6 +11391,573 @@ func TestGatewayAPITCPRouteDAGStatus(t *testing.T) {
})
}
+func TestGatewayAPIBackendTLSPolicyDAGStatus(t *testing.T) {
+ type testcase struct {
+ objs []any
+ gateway *gatewayapi_v1beta1.Gateway
+ wantBackendTLSPolicyConditions []*status.BackendTLSPolicyStatusUpdate
+ }
+
+ run := func(t *testing.T, desc string, tc testcase) {
+ t.Helper()
+ t.Run(desc, func(t *testing.T) {
+ t.Helper()
+ builder := Builder{
+ Source: KubernetesCache{
+ RootNamespaces: []string{"roots", "marketing"},
+ FieldLogger: fixture.NewTestLogger(t),
+ gatewayclass: &gatewayapi_v1beta1.GatewayClass{
+ TypeMeta: metav1.TypeMeta{},
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "test-gc",
+ },
+ Spec: gatewayapi_v1beta1.GatewayClassSpec{
+ ControllerName: "projectcontour.io/contour",
+ },
+ Status: gatewayapi_v1beta1.GatewayClassStatus{
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1.GatewayClassConditionStatusAccepted),
+ Status: metav1.ConditionTrue,
+ },
+ },
+ },
+ },
+ gateway: tc.gateway,
+ },
+ Processors: []Processor{
+ &ListenerProcessor{},
+ &IngressProcessor{
+ FieldLogger: fixture.NewTestLogger(t),
+ },
+ &HTTPProxyProcessor{},
+ &GatewayAPIProcessor{
+ FieldLogger: fixture.NewTestLogger(t),
+ },
+ },
+ }
+
+ // Set a default gateway if not defined by a test
+ if tc.gateway == nil {
+ builder.Source.gateway = &gatewayapi_v1beta1.Gateway{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "contour",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1beta1.GatewaySpec{
+ Listeners: []gatewayapi_v1beta1.Listener{{
+ Name: "http",
+ Port: 80,
+ Protocol: gatewayapi_v1.HTTPProtocolType,
+ AllowedRoutes: &gatewayapi_v1beta1.AllowedRoutes{
+ Namespaces: &gatewayapi_v1beta1.RouteNamespaces{
+ From: ref.To(gatewayapi_v1.NamespacesFromAll),
+ },
+ },
+ }},
+ },
+ }
+ }
+
+ for _, o := range tc.objs {
+ builder.Source.Insert(o)
+ }
+
+ dag := builder.Build()
+ gotBackendTLSPolicyUpdates := dag.StatusCache.GetBackendTLSPolicyUpdates()
+
+ ops := []cmp.Option{
+ cmpopts.IgnoreFields(metav1.Condition{}, "LastTransitionTime"),
+ cmpopts.IgnoreFields(status.BackendTLSPolicyStatusUpdate{}, "GatewayRef"),
+ cmpopts.IgnoreFields(status.BackendTLSPolicyStatusUpdate{}, "Generation"),
+ cmpopts.IgnoreFields(status.BackendTLSPolicyStatusUpdate{}, "TransitionTime"),
+ cmpopts.IgnoreFields(status.BackendTLSPolicyStatusUpdate{}, "Resource"),
+ cmpopts.SortSlices(func(i, j metav1.Condition) bool {
+ return i.Message < j.Message
+ }),
+ cmpopts.SortSlices(func(i, j *status.BackendTLSPolicyStatusUpdate) bool {
+ return i.FullName.String() < j.FullName.String()
+ }),
+ }
+
+ // Since we're using a single static GatewayClass,
+ // set the expected controller string here for all
+ // test cases.
+ for _, u := range tc.wantBackendTLSPolicyConditions {
+ u.GatewayController = builder.Source.gatewayclass.Spec.ControllerName
+
+ for _, pas := range u.PolicyAncestorStatuses {
+ pas.ControllerName = builder.Source.gatewayclass.Spec.ControllerName
+ }
+ }
+
+ if diff := cmp.Diff(tc.wantBackendTLSPolicyConditions, gotBackendTLSPolicyUpdates, ops...); diff != "" {
+ t.Fatalf("expected backend tls policy status: %v, got %v", tc.wantBackendTLSPolicyConditions, diff)
+ }
+ })
+ }
+
+ tlsService := &v1.Service{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "tlssvc",
+ Namespace: "projectcontour",
+ },
+ Spec: v1.ServiceSpec{
+ Ports: []v1.ServicePort{makeServicePort("https", "TCP", 443, 8443)},
+ },
+ }
+
+ configMapCert1 := &v1.ConfigMap{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "ca",
+ Namespace: "projectcontour",
+ },
+ Data: map[string]string{
+ CACertificateKey: fixture.CERTIFICATE,
+ },
+ }
+
+ run(t, "simple httproute with backendtlspolicy", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionTrue,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonAccepted),
+ Message: "Accepted BackendTLSPolicy",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with a targetref that cannot be found does not set any conditions", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "nonexistent",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: nil,
+ })
+
+ run(t, "backendtlspolicy with unsupported cacertref", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "Invalid",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.CACertRef.Kind \"Invalid\" is unsupported. Only ConfigMap or Secret Kind is supported.",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with missing configmap certref", testcase{
+ objs: []any{
+ tlsService,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName("missing"),
+ }},
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "Could not find CACertRef ConfigMap: projectcontour/missing",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with missing secret certref", testcase{
+ objs: []any{
+ tlsService,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "Secret",
+ Name: gatewayapi_v1.ObjectName("missing"),
+ }},
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "Could not find CACertRef Secret: projectcontour/missing",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with multiple cacertref that are a mix of valid and invalid", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{
+ {
+ Kind: "Invalid",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ },
+ {
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName("missing"),
+ },
+ {
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ },
+ },
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.CACertRef.Kind \"Invalid\" is unsupported. Only ConfigMap or Secret Kind is supported., Could not find CACertRef ConfigMap: projectcontour/missing",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with wellknowncacerts set", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ WellKnownCACerts: ptr.To(gatewayapi_v1alpha2.WellKnownCACertSystem),
+ Hostname: "example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.WellKnownCACerts is unsupported.",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with malformed hostname", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "-bad-hostname.example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.Hostname \"-bad-hostname.example.com\" is invalid. Hostname must be a valid RFC 1123 fully qualified domain name. Wildcard domains and numeric IP addresses are not allowed",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with unsupported wildcard hostname", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "*.example.com",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.Hostname \"*.example.com\" is invalid. Hostname must be a valid RFC 1123 fully qualified domain name. Wildcard domains and numeric IP addresses are not allowed",
+ },
+ },
+ },
+ },
+ }},
+ })
+
+ run(t, "backendtlspolicy with unsupported numeric ip as hostname", testcase{
+ objs: []any{
+ tlsService,
+ configMapCert1,
+ makeHTTPRoute("basic", "projectcontour", "", makeHTTPRouteRule(gatewayapi_v1.PathMatchPathPrefix, "/", "tlssvc", 443, 1)),
+ &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "basic",
+ Namespace: "projectcontour",
+ },
+ Spec: gatewayapi_v1alpha2.BackendTLSPolicySpec{
+ TargetRef: gatewayapi_v1alpha2.PolicyTargetReferenceWithSectionName{
+ PolicyTargetReference: gatewayapi_v1alpha2.PolicyTargetReference{
+ Kind: "Service",
+ Name: "tlssvc",
+ },
+ },
+ TLS: gatewayapi_v1alpha2.BackendTLSPolicyConfig{
+ CACertRefs: []gatewayapi_v1alpha2.LocalObjectReference{{
+ Kind: "ConfigMap",
+ Name: gatewayapi_v1.ObjectName(configMapCert1.Name),
+ }},
+ Hostname: "127.0.0.1",
+ },
+ },
+ },
+ },
+ wantBackendTLSPolicyConditions: []*status.BackendTLSPolicyStatusUpdate{{
+ FullName: types.NamespacedName{Namespace: "projectcontour", Name: "basic"},
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionFalse,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonInvalid),
+ Message: "BackendTLSPolicy.Spec.TLS.Hostname \"127.0.0.1\" is invalid. Hostname must be a valid RFC 1123 fully qualified domain name. Wildcard domains and numeric IP addresses are not allowed",
+ },
+ },
+ },
+ },
+ }},
+ })
+}
+
func gatewayAcceptedCondition() metav1.Condition {
return metav1.Condition{
Type: string(gatewayapi_v1.GatewayConditionAccepted),
diff --git a/internal/status/backendtlspolicyconditions.go b/internal/status/backendtlspolicyconditions.go
new file mode 100644
index 00000000000..c4ae9ef8d14
--- /dev/null
+++ b/internal/status/backendtlspolicyconditions.go
@@ -0,0 +1,151 @@
+// Copyright Project Contour Authors
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package status
+
+import (
+ "fmt"
+ "time"
+
+ "github.com/projectcontour/contour/internal/gatewayapi"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/types"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ gatewayapi_v1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
+ gatewayapi_v1beta1 "sigs.k8s.io/gateway-api/apis/v1beta1"
+)
+
+// BackendTLSPolicyStatusUpdate represents an atomic update to a
+// BackendTLSPolicy's status.
+type BackendTLSPolicyStatusUpdate struct {
+ FullName types.NamespacedName
+ PolicyAncestorStatuses []*gatewayapi_v1alpha2.PolicyAncestorStatus
+ GatewayRef types.NamespacedName
+ GatewayController gatewayapi_v1beta1.GatewayController
+ Resource client.Object
+ Generation int64
+ TransitionTime metav1.Time
+}
+
+// BackendTLSPolicyAncestorStatusUpdate helps update a specific ancestor ref's
+// PolicyAncestorStatus.
+type BackendTLSPolicyAncestorStatusUpdate struct {
+ *BackendTLSPolicyStatusUpdate
+ ancestorRef gatewayapi_v1beta1.ParentReference
+}
+
+// StatusUpdateFor returns a BackendTLSPolicyAncestorStatusUpdate for the given
+// ancestor ref.
+func (b *BackendTLSPolicyStatusUpdate) StatusUpdateFor(ancestorRef gatewayapi_v1beta1.ParentReference) *BackendTLSPolicyAncestorStatusUpdate {
+ return &BackendTLSPolicyAncestorStatusUpdate{
+ BackendTLSPolicyStatusUpdate: b,
+ ancestorRef: ancestorRef,
+ }
+}
+
+// AddCondition adds a condition with the given properties to the
+// BackendTLSPolicyAncestorStatus.
+func (b *BackendTLSPolicyAncestorStatusUpdate) AddCondition(conditionType gatewayapi_v1alpha2.PolicyConditionType, status metav1.ConditionStatus, reason gatewayapi_v1alpha2.PolicyConditionReason, message string) metav1.Condition {
+ var pas *gatewayapi_v1alpha2.PolicyAncestorStatus
+
+ for _, v := range b.PolicyAncestorStatuses {
+ if v.AncestorRef == b.ancestorRef {
+ pas = v
+ break
+ }
+ }
+
+ if pas == nil {
+ pas = &gatewayapi_v1alpha2.PolicyAncestorStatus{
+ AncestorRef: b.ancestorRef,
+ ControllerName: b.GatewayController,
+ }
+
+ b.PolicyAncestorStatuses = append(b.PolicyAncestorStatuses, pas)
+ }
+
+ idx := -1
+ for i, c := range pas.Conditions {
+ if c.Type == string(conditionType) {
+ idx = i
+ break
+ }
+ }
+
+ if idx > -1 {
+ message = pas.Conditions[idx].Message + ", " + message
+ }
+
+ cond := metav1.Condition{
+ Reason: string(reason),
+ Status: status,
+ Type: string(conditionType),
+ Message: message,
+ LastTransitionTime: metav1.NewTime(time.Now()),
+ ObservedGeneration: b.Generation,
+ }
+
+ if idx > -1 {
+ pas.Conditions[idx] = cond
+ } else {
+ pas.Conditions = append(pas.Conditions, cond)
+ }
+
+ return cond
+}
+
+// ConditionsForAncestorRef returns the list of conditions for a given ancestor
+// if it exists.
+func (b *BackendTLSPolicyStatusUpdate) ConditionsForAncestorRef(ancestorRef gatewayapi_v1beta1.ParentReference) []metav1.Condition {
+ for _, pas := range b.PolicyAncestorStatuses {
+ if pas.AncestorRef == ancestorRef {
+ return pas.Conditions
+ }
+ }
+
+ return nil
+}
+
+func (b *BackendTLSPolicyStatusUpdate) Mutate(obj client.Object) client.Object {
+ o, ok := obj.(*gatewayapi_v1alpha2.BackendTLSPolicy)
+ if !ok {
+ panic(fmt.Sprintf("Unsupported %T object %s/%s in status mutator",
+ obj, b.FullName.Namespace, b.FullName.Name,
+ ))
+ }
+
+ var newPolicyAncestorStatuses []gatewayapi_v1alpha2.PolicyAncestorStatus
+ for _, pas := range b.PolicyAncestorStatuses {
+ for i := range pas.Conditions {
+ cond := &pas.Conditions[i]
+
+ cond.ObservedGeneration = b.Generation
+ cond.LastTransitionTime = b.TransitionTime
+ }
+
+ newPolicyAncestorStatuses = append(newPolicyAncestorStatuses, *pas)
+ }
+
+ btp := o.DeepCopy()
+
+ // Get all the PolicyAncestorStatuses that are for other Gateways.
+ for _, pas := range o.Status.Ancestors {
+ if !gatewayapi.IsRefToGateway(pas.AncestorRef, b.GatewayRef) {
+ newPolicyAncestorStatuses = append(newPolicyAncestorStatuses, pas)
+ }
+ }
+
+ btp.Status.Ancestors = newPolicyAncestorStatuses
+
+ return btp
+}
diff --git a/internal/status/backendtlspolicyconditions_test.go b/internal/status/backendtlspolicyconditions_test.go
new file mode 100644
index 00000000000..b3e6d87487c
--- /dev/null
+++ b/internal/status/backendtlspolicyconditions_test.go
@@ -0,0 +1,135 @@
+// Copyright Project Contour Authors
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package status
+
+import (
+ "testing"
+ "time"
+
+ contour_api_v1 "github.com/projectcontour/contour/apis/projectcontour/v1"
+ "github.com/projectcontour/contour/internal/gatewayapi"
+ "github.com/projectcontour/contour/internal/k8s"
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ gatewayapi_v1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
+)
+
+func TestBackendTLSPolicyAddCondition(t *testing.T) {
+ backendTLSPolicyUpdate := BackendTLSPolicyStatusUpdate{
+ FullName: k8s.NamespacedNameFrom("test/test"),
+ Generation: 7,
+ }
+
+ ancestorRef := gatewayapi.GatewayParentRef("projectcontour", "contour")
+
+ basUpdate := backendTLSPolicyUpdate.StatusUpdateFor(ancestorRef)
+
+ basUpdate.AddCondition(gatewayapi_v1alpha2.PolicyConditionAccepted, metav1.ConditionTrue, gatewayapi_v1alpha2.PolicyReasonAccepted, "Valid BackendTLSPolicy")
+
+ require.Len(t, backendTLSPolicyUpdate.ConditionsForAncestorRef(ancestorRef), 1)
+ got := backendTLSPolicyUpdate.ConditionsForAncestorRef(ancestorRef)[0]
+
+ assert.EqualValues(t, gatewayapi_v1alpha2.PolicyConditionAccepted, got.Type)
+ assert.EqualValues(t, metav1.ConditionTrue, got.Status)
+ assert.EqualValues(t, gatewayapi_v1alpha2.PolicyReasonAccepted, got.Reason)
+ assert.EqualValues(t, "Valid BackendTLSPolicy", got.Message)
+ assert.EqualValues(t, 7, got.ObservedGeneration)
+}
+
+func TestBackendTLSPolicyMutate(t *testing.T) {
+ testTransitionTime := v1.NewTime(time.Now())
+ var testGeneration int64 = 7
+
+ bsu := BackendTLSPolicyStatusUpdate{
+ FullName: k8s.NamespacedNameFrom("test/test"),
+ Generation: testGeneration,
+ TransitionTime: testTransitionTime,
+ PolicyAncestorStatuses: []*gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionTrue,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonAccepted),
+ Message: "Accepted BackendTLSPolicy",
+ },
+ },
+ },
+ },
+ }
+
+ btp := &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "test",
+ Namespace: "test",
+ },
+ Status: gatewayapi_v1alpha2.PolicyStatus{
+ Ancestors: []gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("externalgateway", "some-gateway"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionTrue,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonAccepted),
+ Message: "This was added by some other gateway and should not be removed.",
+ },
+ },
+ },
+ },
+ },
+ }
+
+ wantBackendTLSPolicy := &gatewayapi_v1alpha2.BackendTLSPolicy{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: "test",
+ Namespace: "test",
+ },
+ Status: gatewayapi_v1alpha2.PolicyStatus{
+ Ancestors: []gatewayapi_v1alpha2.PolicyAncestorStatus{
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("projectcontour", "contour"),
+ Conditions: []metav1.Condition{
+ {
+ ObservedGeneration: testGeneration,
+ LastTransitionTime: testTransitionTime,
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionTrue,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonAccepted),
+ Message: "Accepted BackendTLSPolicy",
+ },
+ },
+ },
+ {
+ AncestorRef: gatewayapi.GatewayParentRef("externalgateway", "some-gateway"),
+ Conditions: []metav1.Condition{
+ {
+ Type: string(gatewayapi_v1alpha2.PolicyConditionAccepted),
+ Status: contour_api_v1.ConditionTrue,
+ Reason: string(gatewayapi_v1alpha2.PolicyReasonAccepted),
+ Message: "This was added by some other gateway and should not be removed.",
+ },
+ },
+ },
+ },
+ },
+ }
+
+ btp, ok := bsu.Mutate(btp).(*gatewayapi_v1alpha2.BackendTLSPolicy)
+ require.True(t, ok)
+ assert.Equal(t, wantBackendTLSPolicy, btp, 1)
+}
diff --git a/internal/status/cache.go b/internal/status/cache.go
index 0274427eeb0..b0d86202fc7 100644
--- a/internal/status/cache.go
+++ b/internal/status/cache.go
@@ -37,12 +37,13 @@ const ValidCondition ConditionType = "Valid"
// NewCache creates a new Cache for holding status updates.
func NewCache(gateway types.NamespacedName, gatewayController gatewayapi_v1beta1.GatewayController) Cache {
return Cache{
- gatewayRef: gateway,
- gatewayController: gatewayController,
- proxyUpdates: make(map[types.NamespacedName]*ProxyUpdate),
- gatewayUpdates: make(map[types.NamespacedName]*GatewayStatusUpdate),
- routeUpdates: make(map[types.NamespacedName]*RouteStatusUpdate),
- entries: make(map[string]map[types.NamespacedName]CacheEntry),
+ gatewayRef: gateway,
+ gatewayController: gatewayController,
+ proxyUpdates: make(map[types.NamespacedName]*ProxyUpdate),
+ gatewayUpdates: make(map[types.NamespacedName]*GatewayStatusUpdate),
+ routeUpdates: make(map[types.NamespacedName]*RouteStatusUpdate),
+ backendTLSPolicyUpdates: make(map[types.NamespacedName]*BackendTLSPolicyStatusUpdate),
+ entries: make(map[string]map[types.NamespacedName]CacheEntry),
}
}
@@ -58,9 +59,10 @@ type Cache struct {
gatewayRef types.NamespacedName
gatewayController gatewayapi_v1beta1.GatewayController
- proxyUpdates map[types.NamespacedName]*ProxyUpdate
- gatewayUpdates map[types.NamespacedName]*GatewayStatusUpdate
- routeUpdates map[types.NamespacedName]*RouteStatusUpdate
+ proxyUpdates map[types.NamespacedName]*ProxyUpdate
+ gatewayUpdates map[types.NamespacedName]*GatewayStatusUpdate
+ routeUpdates map[types.NamespacedName]*RouteStatusUpdate
+ backendTLSPolicyUpdates map[types.NamespacedName]*BackendTLSPolicyStatusUpdate
// Map of cache entry maps, keyed on Kind.
entries map[string]map[types.NamespacedName]CacheEntry
@@ -96,6 +98,16 @@ func (c *Cache) Put(obj metav1.Object, e CacheEntry) {
func (c *Cache) GetStatusUpdates() []k8s.StatusUpdate {
var flattened []k8s.StatusUpdate
+ for fullname, backendTLSPolicyUpdate := range c.backendTLSPolicyUpdates {
+ update := k8s.StatusUpdate{
+ NamespacedName: fullname,
+ Resource: backendTLSPolicyUpdate.Resource,
+ Mutator: backendTLSPolicyUpdate,
+ }
+
+ flattened = append(flattened, update)
+ }
+
for fullname, pu := range c.proxyUpdates {
update := k8s.StatusUpdate{
NamespacedName: fullname,
@@ -165,6 +177,15 @@ func (c *Cache) GetRouteUpdates() []*RouteStatusUpdate {
return allUpdates
}
+// GetBackendTLSPolicyUpdates gets the underlying BackendTLSPolicyConditionsUpdate objects from the cache.
+func (c *Cache) GetBackendTLSPolicyUpdates() []*BackendTLSPolicyStatusUpdate {
+ var allUpdates []*BackendTLSPolicyStatusUpdate
+ for _, conditionsUpdate := range c.backendTLSPolicyUpdates {
+ allUpdates = append(allUpdates, conditionsUpdate)
+ }
+ return allUpdates
+}
+
// GatewayStatusAccessor returns a GatewayStatusUpdate that allows a client to build up a list of
// status changes as well as a function to commit the change back to the cache when everything
// is done. The commit function pattern is used so that the GatewayStatusUpdate does not need
@@ -242,3 +263,25 @@ func (c *Cache) RouteConditionsAccessor(nsName types.NamespacedName, generation
c.routeUpdates[pu.FullName] = pu
}
}
+
+// BackendTLSPolicyConditionsAccessor returns a BackendTLSPolicyStatusUpdate that allows a client
+// to build up a list of metav1.Conditions as well as a function to commit the change back to the
+// cache when everything is done. The commit function pattern is used so that the
+// BackendTLSPolicyStatusUpdate does not need to know anything the cache internals.
+func (c *Cache) BackendTLSPolicyConditionsAccessor(nsName types.NamespacedName, generation int64, resource client.Object) (*BackendTLSPolicyStatusUpdate, func()) {
+ pu := &BackendTLSPolicyStatusUpdate{
+ FullName: nsName,
+ GatewayRef: c.gatewayRef,
+ GatewayController: c.gatewayController,
+ Generation: generation,
+ TransitionTime: metav1.NewTime(time.Now()),
+ Resource: resource,
+ }
+
+ return pu, func() {
+ if len(pu.PolicyAncestorStatuses) == 0 {
+ return
+ }
+ c.backendTLSPolicyUpdates[pu.FullName] = pu
+ }
+}
diff --git a/test/e2e/gateway/backend_tls_policy_test.go b/test/e2e/gateway/backend_tls_policy_test.go
index 207c8bc5892..7535792fb96 100644
--- a/test/e2e/gateway/backend_tls_policy_test.go
+++ b/test/e2e/gateway/backend_tls_policy_test.go
@@ -163,7 +163,8 @@ func testBackendTLSPolicy(namespace string, gateway types.NamespacedName) {
},
}
- f.CreateBackendTLSPolicyAndWaitFor(backendTLSPolicy, e2e.BackendTLSPolicyAccepted)
+ _, ok := f.CreateBackendTLSPolicyAndWaitFor(backendTLSPolicy, e2e.BackendTLSPolicyAccepted)
+ assert.Truef(t, ok, "expected policy condition accepted on backend tls policy")
type responseTLSDetails struct {
TLS struct {
diff --git a/test/e2e/gatewayapi_predicates.go b/test/e2e/gatewayapi_predicates.go
index 67556328ce0..5f307b15e61 100644
--- a/test/e2e/gatewayapi_predicates.go
+++ b/test/e2e/gatewayapi_predicates.go
@@ -173,9 +173,17 @@ func TLSRouteAccepted(route *gatewayapi_v1alpha2.TLSRoute) bool {
// BackendTLSPolicyAccepted returns true if the backend TLS policy has a .status.conditions
// entry of "Accepted: true".
func BackendTLSPolicyAccepted(btp *gatewayapi_v1alpha2.BackendTLSPolicy) bool {
- // TODO (christianang): Right now this always returns true if a backendtlspolicy is
- // provided since status conditions are not implemented yet for BackendTLSPolicy
- return btp != nil
+ if btp == nil {
+ return false
+ }
+
+ for _, gw := range btp.Status.Ancestors {
+ if conditionExists(gw.Conditions, string(gatewayapi_v1alpha2.PolicyConditionAccepted), metav1.ConditionTrue) {
+ return true
+ }
+ }
+
+ return false
}
func conditionExists(conditions []metav1.Condition, conditionType string, conditionStatus metav1.ConditionStatus) bool {
From c35f7f0e7b3d0727b92ae6076799fd94f84ed2b5 Mon Sep 17 00:00:00 2001
From: Sunjay Bhatia <5337253+sunjayBhatia@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:09:47 -0500
Subject: [PATCH 10/83] Bump Envoy to 1.29.1 (#6164)
Signed-off-by: Sunjay Bhatia
---
Makefile | 2 +-
changelogs/unreleased/6123-skriss-small.md | 1 -
changelogs/unreleased/6164-sunjayBhatia-small.md | 1 +
cmd/contour/gatewayprovisioner.go | 2 +-
examples/contour/03-envoy.yaml | 2 +-
examples/deployment/03-envoy-deployment.yaml | 2 +-
examples/render/contour-deployment.yaml | 2 +-
examples/render/contour-gateway.yaml | 2 +-
examples/render/contour.yaml | 2 +-
site/content/resources/compatibility-matrix.md | 4 ++--
versions.yaml | 2 +-
11 files changed, 11 insertions(+), 11 deletions(-)
delete mode 100644 changelogs/unreleased/6123-skriss-small.md
create mode 100644 changelogs/unreleased/6164-sunjayBhatia-small.md
diff --git a/Makefile b/Makefile
index 4b933e29d58..a6c1a796440 100644
--- a/Makefile
+++ b/Makefile
@@ -6,7 +6,7 @@ IMAGE := $(REGISTRY)/$(PROJECT)
SRCDIRS := ./cmd ./internal ./apis
LOCAL_BOOTSTRAP_CONFIG = localenvoyconfig.yaml
SECURE_LOCAL_BOOTSTRAP_CONFIG = securelocalenvoyconfig.yaml
-ENVOY_IMAGE = docker.io/envoyproxy/envoy:v1.29.0
+ENVOY_IMAGE = docker.io/envoyproxy/envoy:v1.29.1
GATEWAY_API_VERSION ?= $(shell grep "sigs.k8s.io/gateway-api" go.mod | awk '{print $$2}')
# Used to supply a local Envoy docker container an IP to connect to that is running
diff --git a/changelogs/unreleased/6123-skriss-small.md b/changelogs/unreleased/6123-skriss-small.md
deleted file mode 100644
index ce2dce41d0b..00000000000
--- a/changelogs/unreleased/6123-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Updates Envoy to v1.29.0. See the release notes [here](https://www.envoyproxy.io/docs/envoy/v1.29.0/version_history/v1.29/v1.29.0).
diff --git a/changelogs/unreleased/6164-sunjayBhatia-small.md b/changelogs/unreleased/6164-sunjayBhatia-small.md
new file mode 100644
index 00000000000..3da383b22f4
--- /dev/null
+++ b/changelogs/unreleased/6164-sunjayBhatia-small.md
@@ -0,0 +1 @@
+Updates Envoy to v1.29.1. See the release notes [here](https://www.envoyproxy.io/docs/envoy/v1.29.1/version_history/v1.29/v1.29.1).
diff --git a/cmd/contour/gatewayprovisioner.go b/cmd/contour/gatewayprovisioner.go
index 3d9f03d95fb..ab3e2a5607e 100644
--- a/cmd/contour/gatewayprovisioner.go
+++ b/cmd/contour/gatewayprovisioner.go
@@ -36,7 +36,7 @@ func registerGatewayProvisioner(app *kingpin.Application) (*kingpin.CmdClause, *
provisionerConfig := &gatewayProvisionerConfig{
contourImage: "ghcr.io/projectcontour/contour:main",
- envoyImage: "docker.io/envoyproxy/envoy:v1.29.0",
+ envoyImage: "docker.io/envoyproxy/envoy:v1.29.1",
metricsBindAddress: ":8080",
leaderElection: false,
leaderElectionID: "0d879e31.projectcontour.io",
diff --git a/examples/contour/03-envoy.yaml b/examples/contour/03-envoy.yaml
index 6bcd8fcbb69..cde50e56c24 100644
--- a/examples/contour/03-envoy.yaml
+++ b/examples/contour/03-envoy.yaml
@@ -50,7 +50,7 @@ spec:
- --log-level info
command:
- envoy
- image: docker.io/envoyproxy/envoy:v1.29.0
+ image: docker.io/envoyproxy/envoy:v1.29.1
imagePullPolicy: IfNotPresent
name: envoy
env:
diff --git a/examples/deployment/03-envoy-deployment.yaml b/examples/deployment/03-envoy-deployment.yaml
index a05c9a4dace..c7861ec5244 100644
--- a/examples/deployment/03-envoy-deployment.yaml
+++ b/examples/deployment/03-envoy-deployment.yaml
@@ -62,7 +62,7 @@ spec:
- --log-level info
command:
- envoy
- image: docker.io/envoyproxy/envoy:v1.29.0
+ image: docker.io/envoyproxy/envoy:v1.29.1
imagePullPolicy: IfNotPresent
name: envoy
env:
diff --git a/examples/render/contour-deployment.yaml b/examples/render/contour-deployment.yaml
index f401e80ddeb..790f7744634 100644
--- a/examples/render/contour-deployment.yaml
+++ b/examples/render/contour-deployment.yaml
@@ -9197,7 +9197,7 @@ spec:
- --log-level info
command:
- envoy
- image: docker.io/envoyproxy/envoy:v1.29.0
+ image: docker.io/envoyproxy/envoy:v1.29.1
imagePullPolicy: IfNotPresent
name: envoy
env:
diff --git a/examples/render/contour-gateway.yaml b/examples/render/contour-gateway.yaml
index ab80e2eaf68..a4e9d4efd53 100644
--- a/examples/render/contour-gateway.yaml
+++ b/examples/render/contour-gateway.yaml
@@ -9188,7 +9188,7 @@ spec:
- --log-level info
command:
- envoy
- image: docker.io/envoyproxy/envoy:v1.29.0
+ image: docker.io/envoyproxy/envoy:v1.29.1
imagePullPolicy: IfNotPresent
name: envoy
env:
diff --git a/examples/render/contour.yaml b/examples/render/contour.yaml
index 73c2bf28dd4..e2ab3f43ff3 100644
--- a/examples/render/contour.yaml
+++ b/examples/render/contour.yaml
@@ -9185,7 +9185,7 @@ spec:
- --log-level info
command:
- envoy
- image: docker.io/envoyproxy/envoy:v1.29.0
+ image: docker.io/envoyproxy/envoy:v1.29.1
imagePullPolicy: IfNotPresent
name: envoy
env:
diff --git a/site/content/resources/compatibility-matrix.md b/site/content/resources/compatibility-matrix.md
index 5ab7304f7e9..b6d8f31d584 100644
--- a/site/content/resources/compatibility-matrix.md
+++ b/site/content/resources/compatibility-matrix.md
@@ -10,7 +10,7 @@ These combinations of versions are specifically tested in CI and supported by th
| Contour Version | Envoy Version | Kubernetes Versions | Gateway API Version |
| --------------- | :------------------- | ------------------- | --------------------|
-| main | [1.29.0][46] | 1.29, 1.28, 1.27 | [1.0.0][110] |
+| main | [1.29.1][46] | 1.29, 1.28, 1.27 | [1.0.0][110] |
| 1.27.0 | [1.28.0][45] | 1.28, 1.27, 1.26 | [0.8.1][109] |
| 1.26.1 | [1.27.2][42] | 1.28, 1.27, 1.26 | [0.8.1][109] |
| 1.26.0 | [1.27.0][41] | 1.28, 1.27, 1.26 | [0.8.0][108] |
@@ -174,7 +174,7 @@ __Note:__ This list of extensions was last verified to be complete with Envoy v1
[43]: https://www.envoyproxy.io/docs/envoy/v1.26.6/version_history/v1.26/v1.26.6
[44]: https://www.envoyproxy.io/docs/envoy/v1.25.11/version_history/v1.25/v1.25.11
[45]: https://www.envoyproxy.io/docs/envoy/v1.28.0/version_history/v1.28/v1.28.0
-[46]: https://www.envoyproxy.io/docs/envoy/v1.29.0/version_history/v1.29/v1.29.0
+[46]: https://www.envoyproxy.io/docs/envoy/v1.29.1/version_history/v1.29/v1.29.1
[98]: https://github.com/kubernetes/client-go
[99]: https://github.com/kubernetes/client-go#compatibility-matrix
diff --git a/versions.yaml b/versions.yaml
index 2ca386213ef..2ca19ee939f 100644
--- a/versions.yaml
+++ b/versions.yaml
@@ -7,7 +7,7 @@ versions:
- version: main
supported: "false"
dependencies:
- envoy: "1.29.0"
+ envoy: "1.29.1"
kubernetes:
- "1.29"
- "1.28"
From ca4108f0d49e32f1b899af411bc2213a0e844e9b Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:11:33 -0500
Subject: [PATCH 11/83] build(deps): bump sigs.k8s.io/controller-runtime from
0.17.0 to 0.17.1 (#6173)
Bumps [sigs.k8s.io/controller-runtime](https://github.com/kubernetes-sigs/controller-runtime) from 0.17.0 to 0.17.1.
- [Release notes](https://github.com/kubernetes-sigs/controller-runtime/releases)
- [Changelog](https://github.com/kubernetes-sigs/controller-runtime/blob/main/RELEASE.md)
- [Commits](https://github.com/kubernetes-sigs/controller-runtime/compare/v0.17.0...v0.17.1)
---
updated-dependencies:
- dependency-name: sigs.k8s.io/controller-runtime
dependency-type: direct:production
update-type: version-update:semver-patch
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
go.mod | 2 +-
go.sum | 4 ++--
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/go.mod b/go.mod
index 325523c784e..3301daffff2 100644
--- a/go.mod
+++ b/go.mod
@@ -41,7 +41,7 @@ require (
k8s.io/client-go v0.29.1
k8s.io/klog/v2 v2.120.1
k8s.io/utils v0.0.0-20240102154912-e7106e64919e
- sigs.k8s.io/controller-runtime v0.17.0
+ sigs.k8s.io/controller-runtime v0.17.1
sigs.k8s.io/controller-tools v0.14.0
sigs.k8s.io/gateway-api v1.0.0
sigs.k8s.io/kustomize/kyaml v0.16.0
diff --git a/go.sum b/go.sum
index f0b7e04b19a..8945ccc650a 100644
--- a/go.sum
+++ b/go.sum
@@ -811,8 +811,8 @@ rsc.io/pdf v0.1.1 h1:k1MczvYDUvJBe93bYd7wrZLLUEcLZAuF824/I4e5Xr4=
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=
-sigs.k8s.io/controller-runtime v0.17.0 h1:fjJQf8Ukya+VjogLO6/bNX9HE6Y2xpsO5+fyS26ur/s=
-sigs.k8s.io/controller-runtime v0.17.0/go.mod h1:+MngTvIQQQhfXtwfdGw/UOQ/aIaqsYywfCINOtwMO/s=
+sigs.k8s.io/controller-runtime v0.17.1 h1:V1dQELMGVk46YVXXQUbTFujU7u4DQj6YUj9Rb6cuzz8=
+sigs.k8s.io/controller-runtime v0.17.1/go.mod h1:+MngTvIQQQhfXtwfdGw/UOQ/aIaqsYywfCINOtwMO/s=
sigs.k8s.io/controller-tools v0.14.0 h1:rnNoCC5wSXlrNoBKKzL70LNJKIQKEzT6lloG6/LF73A=
sigs.k8s.io/controller-tools v0.14.0/go.mod h1:TV7uOtNNnnR72SpzhStvPkoS/U5ir0nMudrkrC4M9Sc=
sigs.k8s.io/gateway-api v1.0.0 h1:iPTStSv41+d9p0xFydll6d7f7MOBGuqXM6p2/zVYMAs=
From f188984a879b5947ca86012526351cbf874ed0ef Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:12:28 -0500
Subject: [PATCH 12/83] build(deps): bump aquasecurity/trivy-action from 0.16.1
to 0.17.0 (#6170)
Bumps [aquasecurity/trivy-action](https://github.com/aquasecurity/trivy-action) from 0.16.1 to 0.17.0.
- [Release notes](https://github.com/aquasecurity/trivy-action/releases)
- [Commits](https://github.com/aquasecurity/trivy-action/compare/d43c1f16c00cfd3978dde6c07f4bbcf9eb6993ca...84384bd6e777ef152729993b8145ea352e9dd3ef)
---
updated-dependencies:
- dependency-name: aquasecurity/trivy-action
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/trivy-scan.yaml | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/.github/workflows/trivy-scan.yaml b/.github/workflows/trivy-scan.yaml
index 6b7421caea3..29478cbb0ad 100644
--- a/.github/workflows/trivy-scan.yaml
+++ b/.github/workflows/trivy-scan.yaml
@@ -27,7 +27,7 @@ jobs:
with:
persist-credentials: false
ref: ${{ matrix.branch }}
- - uses: aquasecurity/trivy-action@d43c1f16c00cfd3978dde6c07f4bbcf9eb6993ca # 0.16.1
+ - uses: aquasecurity/trivy-action@84384bd6e777ef152729993b8145ea352e9dd3ef # 0.17.0
with:
scanners: vuln
scan-type: 'fs'
From 09aaadf86374e9efab61e6a966ed42fb741f31a2 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:17:55 -0500
Subject: [PATCH 13/83] build(deps): bump the artifact-actions group with 2
updates (#6169)
Bumps the artifact-actions group with 2 updates: [actions/upload-artifact](https://github.com/actions/upload-artifact) and [actions/download-artifact](https://github.com/actions/download-artifact).
Updates `actions/upload-artifact` from 4.3.0 to 4.3.1
- [Release notes](https://github.com/actions/upload-artifact/releases)
- [Commits](https://github.com/actions/upload-artifact/compare/26f96dfa697d77e81fd5907df203aa23a56210a8...5d5d22a31266ced268874388b861e4b58bb5c2f3)
Updates `actions/download-artifact` from 4.1.1 to 4.1.2
- [Release notes](https://github.com/actions/download-artifact/releases)
- [Commits](https://github.com/actions/download-artifact/compare/6b208ae046db98c579e8a3aa621ab581ff575935...eaceaf801fd36c7dee90939fad912460b18a1ffe)
---
updated-dependencies:
- dependency-name: actions/upload-artifact
dependency-type: direct:production
update-type: version-update:semver-patch
dependency-group: artifact-actions
- dependency-name: actions/download-artifact
dependency-type: direct:production
update-type: version-update:semver-patch
dependency-group: artifact-actions
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/build_tag.yaml | 2 +-
.github/workflows/openssf-scorecard.yaml | 2 +-
.github/workflows/prbuild.yaml | 8 ++++----
3 files changed, 6 insertions(+), 6 deletions(-)
diff --git a/.github/workflows/build_tag.yaml b/.github/workflows/build_tag.yaml
index 6c71ce27b37..f4e250ed0f1 100644
--- a/.github/workflows/build_tag.yaml
+++ b/.github/workflows/build_tag.yaml
@@ -84,7 +84,7 @@ jobs:
export CONTOUR_E2E_IMAGE="ghcr.io/projectcontour/contour:$(git describe --tags)"
make setup-kind-cluster run-gateway-conformance cleanup-kind
- name: Upload gateway conformance report
- uses: actions/upload-artifact@26f96dfa697d77e81fd5907df203aa23a56210a8 # v4.3.0
+ uses: actions/upload-artifact@5d5d22a31266ced268874388b861e4b58bb5c2f3 # v4.3.1
with:
name: gateway-conformance-report
path: gateway-conformance-report/projectcontour-contour-*.yaml
diff --git a/.github/workflows/openssf-scorecard.yaml b/.github/workflows/openssf-scorecard.yaml
index eadc61b9dd5..d32a0a5dbc7 100644
--- a/.github/workflows/openssf-scorecard.yaml
+++ b/.github/workflows/openssf-scorecard.yaml
@@ -32,7 +32,7 @@ jobs:
results_format: sarif
publish_results: true
- name: "Upload artifact"
- uses: actions/upload-artifact@26f96dfa697d77e81fd5907df203aa23a56210a8 # v4.3.0
+ uses: actions/upload-artifact@5d5d22a31266ced268874388b861e4b58bb5c2f3 # v4.3.1
with:
name: SARIF file
path: results.sarif
diff --git a/.github/workflows/prbuild.yaml b/.github/workflows/prbuild.yaml
index 05e72a9f3d9..30e633e321c 100644
--- a/.github/workflows/prbuild.yaml
+++ b/.github/workflows/prbuild.yaml
@@ -114,7 +114,7 @@ jobs:
run: |
make multiarch-build
- name: Upload image
- uses: actions/upload-artifact@26f96dfa697d77e81fd5907df203aa23a56210a8 # v4.3.0
+ uses: actions/upload-artifact@5d5d22a31266ced268874388b861e4b58bb5c2f3 # v4.3.1
with:
name: image
path: image/contour-*.tar
@@ -155,7 +155,7 @@ jobs:
with:
persist-credentials: false
- name: Download image
- uses: actions/download-artifact@6b208ae046db98c579e8a3aa621ab581ff575935 # v4.1.1
+ uses: actions/download-artifact@eaceaf801fd36c7dee90939fad912460b18a1ffe # v4.1.2
with:
name: image
path: image
@@ -218,7 +218,7 @@ jobs:
# recent release tag.
fetch-depth: 0
- name: Download image
- uses: actions/download-artifact@6b208ae046db98c579e8a3aa621ab581ff575935 # v4.1.1
+ uses: actions/download-artifact@eaceaf801fd36c7dee90939fad912460b18a1ffe # v4.1.2
with:
name: image
path: image
@@ -345,7 +345,7 @@ jobs:
with:
persist-credentials: false
- name: Download image
- uses: actions/download-artifact@6b208ae046db98c579e8a3aa621ab581ff575935 # v4.1.1
+ uses: actions/download-artifact@eaceaf801fd36c7dee90939fad912460b18a1ffe # v4.1.2
with:
name: image
path: image
From 28180caca07a3fbe78b6c4872928b7197950fa29 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:18:55 -0500
Subject: [PATCH 14/83] build(deps): bump github.com/cert-manager/cert-manager
(#6175)
Bumps [github.com/cert-manager/cert-manager](https://github.com/cert-manager/cert-manager) from 1.14.1 to 1.14.2.
- [Release notes](https://github.com/cert-manager/cert-manager/releases)
- [Commits](https://github.com/cert-manager/cert-manager/compare/v1.14.1...v1.14.2)
---
updated-dependencies:
- dependency-name: github.com/cert-manager/cert-manager
dependency-type: direct:production
update-type: version-update:semver-patch
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
go.mod | 2 +-
go.sum | 4 ++--
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/go.mod b/go.mod
index 3301daffff2..224b257c857 100644
--- a/go.mod
+++ b/go.mod
@@ -8,7 +8,7 @@ require (
github.com/ahmetb/gen-crd-api-reference-docs v0.3.0
github.com/alecthomas/kingpin/v2 v2.4.0
github.com/bombsimon/logrusr/v4 v4.1.0
- github.com/cert-manager/cert-manager v1.14.1
+ github.com/cert-manager/cert-manager v1.14.2
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc
github.com/distribution/reference v0.5.0
github.com/envoyproxy/go-control-plane v0.12.1-0.20240111020705-5401a878d8bb
diff --git a/go.sum b/go.sum
index 8945ccc650a..edcdb2ad95b 100644
--- a/go.sum
+++ b/go.sum
@@ -68,8 +68,8 @@ github.com/campoy/embedmd v1.0.0/go.mod h1:oxyr9RCiSXg0M3VJ3ks0UGfp98BpSSGr0kpiX
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/census-instrumentation/opencensus-proto v0.4.1 h1:iKLQ0xPNFxR/2hzXZMrBo8f1j86j5WHzznCCQxV/b8g=
github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw=
-github.com/cert-manager/cert-manager v1.14.1 h1:i5sJHfEucqpAfVjkCe3n4sO5S+6YBaN2Yu18+l/1ZMw=
-github.com/cert-manager/cert-manager v1.14.1/go.mod h1:pik7K6jXfgh++lfVJ/i1HzEnDluSUtTVLXSHikj8Lho=
+github.com/cert-manager/cert-manager v1.14.2 h1:C/uci6yxiCRO04PWomBbSX+T4JT58FIIpDj5SZ6Ks6I=
+github.com/cert-manager/cert-manager v1.14.2/go.mod h1:pik7K6jXfgh++lfVJ/i1HzEnDluSUtTVLXSHikj8Lho=
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/chigopher/pathlib v0.19.1 h1:RoLlUJc0CqBGwq239cilyhxPNLXTK+HXoASGyGznx5A=
From 1577e0796cff77475b173736d11a457e0ba510e4 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:25:31 -0500
Subject: [PATCH 15/83] build(deps): bump golang.org/x/oauth2 from 0.16.0 to
0.17.0 (#6174)
Bumps [golang.org/x/oauth2](https://github.com/golang/oauth2) from 0.16.0 to 0.17.0.
- [Commits](https://github.com/golang/oauth2/compare/v0.16.0...v0.17.0)
---
updated-dependencies:
- dependency-name: golang.org/x/oauth2
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
go.mod | 10 +++++-----
go.sum | 20 ++++++++++----------
2 files changed, 15 insertions(+), 15 deletions(-)
diff --git a/go.mod b/go.mod
index 224b257c857..9b1c6f780c5 100644
--- a/go.mod
+++ b/go.mod
@@ -29,7 +29,7 @@ require (
github.com/tsaarni/certyaml v0.9.3
github.com/vektra/mockery/v2 v2.40.1
go.uber.org/automaxprocs v1.5.3
- golang.org/x/oauth2 v0.16.0
+ golang.org/x/oauth2 v0.17.0
gonum.org/v1/plot v0.14.0
google.golang.org/genproto/googleapis/rpc v0.0.0-20240102182953-50ed04b92917
google.golang.org/grpc v1.61.0
@@ -119,13 +119,13 @@ require (
github.com/subosito/gotenv v1.4.2 // indirect
github.com/tsaarni/x500dn v1.0.0 // indirect
github.com/xhit/go-str2duration/v2 v2.1.0 // indirect
- golang.org/x/crypto v0.18.0 // indirect
+ golang.org/x/crypto v0.19.0 // indirect
golang.org/x/exp v0.0.0-20231226003508-02704c960a9b // indirect
golang.org/x/image v0.11.0 // indirect
golang.org/x/mod v0.14.0 // indirect
- golang.org/x/net v0.20.0 // indirect
- golang.org/x/sys v0.16.0 // indirect
- golang.org/x/term v0.16.0 // indirect
+ golang.org/x/net v0.21.0 // indirect
+ golang.org/x/sys v0.17.0 // indirect
+ golang.org/x/term v0.17.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/time v0.5.0 // indirect
golang.org/x/tools v0.16.1 // indirect
diff --git a/go.sum b/go.sum
index edcdb2ad95b..4f404a8fbc5 100644
--- a/go.sum
+++ b/go.sum
@@ -416,8 +416,8 @@ golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPh
golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4=
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/crypto v0.0.0-20211108221036-ceb1ce70b4fa/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
-golang.org/x/crypto v0.18.0 h1:PGVlW0xEltQnzFZ55hkuX5+KLyrMYhHld1YHO4AKcdc=
-golang.org/x/crypto v0.18.0/go.mod h1:R0j02AL6hcrfOiy9T4ZYp/rcWeMxM3L6QYxlOuEG1mg=
+golang.org/x/crypto v0.19.0 h1:ENy+Az/9Y1vSrlrvBSyna3PITt4tiZLf7sgCjZBX7Wo=
+golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU=
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
@@ -492,8 +492,8 @@ golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
-golang.org/x/net v0.20.0 h1:aCL9BSgETF1k+blQaYUBx9hJ9LOGP3gAVemcZlf1Kpo=
-golang.org/x/net v0.20.0/go.mod h1:z8BVo6PvndSri0LbOE3hAn0apkU+1YvI6E70E9jsnvY=
+golang.org/x/net v0.21.0 h1:AQyQV4dYCvJ7vGmJyKki9+PBdyvhkSd8EIx/qb0AYv4=
+golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44=
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
@@ -503,8 +503,8 @@ golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ
golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
-golang.org/x/oauth2 v0.16.0 h1:aDkGMBSYxElaoP81NpoUoz2oo2R2wHdZpGToUxfyQrQ=
-golang.org/x/oauth2 v0.16.0/go.mod h1:hqZ+0LWXsiVoZpeld6jVt06P3adbS2Uu911W1SsJv2o=
+golang.org/x/oauth2 v0.17.0 h1:6m3ZPmLEFdVxKKWnKq4VqZ60gutO35zm+zrAHVmHyDQ=
+golang.org/x/oauth2 v0.17.0/go.mod h1:OzPDGQiuQMguemayvdylqddI7qcD9lnSDb+1FiwQ5HA=
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
@@ -563,13 +563,13 @@ golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBc
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.16.0 h1:xWw16ngr6ZMtmxDyKyIgsE93KNKz5HKmMa3b8ALHidU=
-golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
+golang.org/x/sys v0.17.0 h1:25cE3gD+tdBA7lp7QfhuV+rJiE9YXTcS3VG1SqssI/Y=
+golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
-golang.org/x/term v0.16.0 h1:m+B6fahuftsE9qjo0VWp2FW0mB3MTJvR0BaMQrq0pmE=
-golang.org/x/term v0.16.0/go.mod h1:yn7UURbUtPyrVJPGPq404EukNFxcm/foM+bV/bfcDsY=
+golang.org/x/term v0.17.0 h1:mkTF7LCd6WGJNL3K1Ad7kwxNfYAW6a8a8QqtMblp/4U=
+golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk=
golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
From 1e37c8fb907bce5f45bf83c9e86f53cfa47e1c23 Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 09:56:08 -0500
Subject: [PATCH 16/83] build(deps): bump golangci/golangci-lint-action from
3.7.0 to 4.0.0 (#6171)
Bumps [golangci/golangci-lint-action](https://github.com/golangci/golangci-lint-action) from 3.7.0 to 4.0.0.
- [Release notes](https://github.com/golangci/golangci-lint-action/releases)
- [Commits](https://github.com/golangci/golangci-lint-action/compare/3a919529898de77ec3da873e3063ca4b10e7f5cc...3cfe3a4abbb849e10058ce4af15d205b6da42804)
---
updated-dependencies:
- dependency-name: golangci/golangci-lint-action
dependency-type: direct:production
update-type: version-update:semver-major
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/prbuild.yaml | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/.github/workflows/prbuild.yaml b/.github/workflows/prbuild.yaml
index 30e633e321c..2e4d757838f 100644
--- a/.github/workflows/prbuild.yaml
+++ b/.github/workflows/prbuild.yaml
@@ -27,7 +27,7 @@ jobs:
with:
persist-credentials: false
- name: golangci-lint
- uses: golangci/golangci-lint-action@3a919529898de77ec3da873e3063ca4b10e7f5cc # v3.7.0
+ uses: golangci/golangci-lint-action@3cfe3a4abbb849e10058ce4af15d205b6da42804 # v4.0.0
with:
version: v1.55.2
# TODO: re-enable linting tools package once https://github.com/projectcontour/contour/issues/5077
From 6ea6bb90477f0f5e3443e1e1f5351ec75cc3825b Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 14:56:41 +0000
Subject: [PATCH 17/83] build(deps): bump mheap/github-action-required-labels
(#6172)
Bumps [mheap/github-action-required-labels](https://github.com/mheap/github-action-required-labels) from 5.2.0 to 5.3.0.
- [Release notes](https://github.com/mheap/github-action-required-labels/releases)
- [Commits](https://github.com/mheap/github-action-required-labels/compare/cc7a79fadbba6ed1d6f0efd70707e7b8bf7e6910...80a96a4863886addcbc9f681b5b295ba7f5424e1)
---
updated-dependencies:
- dependency-name: mheap/github-action-required-labels
dependency-type: direct:production
update-type: version-update:semver-minor
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
.github/workflows/label_check.yaml | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/.github/workflows/label_check.yaml b/.github/workflows/label_check.yaml
index 6f3c98d6738..43cb9dbb48d 100644
--- a/.github/workflows/label_check.yaml
+++ b/.github/workflows/label_check.yaml
@@ -24,17 +24,17 @@ jobs:
name: Check release-note label set
runs-on: ubuntu-latest
steps:
- - uses: mheap/github-action-required-labels@cc7a79fadbba6ed1d6f0efd70707e7b8bf7e6910 # v5.2.0
+ - uses: mheap/github-action-required-labels@80a96a4863886addcbc9f681b5b295ba7f5424e1 # v5.3.0
with:
mode: minimum
count: 1
labels: "release-note/major, release-note/minor, release-note/small, release-note/docs, release-note/infra, release-note/deprecation, release-note/none-required"
- - uses: mheap/github-action-required-labels@cc7a79fadbba6ed1d6f0efd70707e7b8bf7e6910 # v5.2.0
+ - uses: mheap/github-action-required-labels@80a96a4863886addcbc9f681b5b295ba7f5424e1 # v5.3.0
with:
mode: maximum
count: 1
labels: "release-note/major, release-note/minor, release-note/small, release-note/docs, release-note/infra, release-note/none-required"
- - uses: mheap/github-action-required-labels@cc7a79fadbba6ed1d6f0efd70707e7b8bf7e6910 # v5.2.0
+ - uses: mheap/github-action-required-labels@80a96a4863886addcbc9f681b5b295ba7f5424e1 # v5.3.0
with:
mode: maximum
count: 1
From 84b111751d1ee5ae20dfb88b2c630c93588e98fe Mon Sep 17 00:00:00 2001
From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com>
Date: Mon, 12 Feb 2024 15:21:52 +0000
Subject: [PATCH 18/83] build(deps): bump github.com/vektra/mockery/v2 from
2.40.1 to 2.40.3 (#6176)
Bumps [github.com/vektra/mockery/v2](https://github.com/vektra/mockery) from 2.40.1 to 2.40.3.
- [Release notes](https://github.com/vektra/mockery/releases)
- [Changelog](https://github.com/vektra/mockery/blob/master/docs/changelog.md)
- [Commits](https://github.com/vektra/mockery/compare/v2.40.1...v2.40.3)
---
updated-dependencies:
- dependency-name: github.com/vektra/mockery/v2
dependency-type: direct:production
update-type: version-update:semver-patch
...
Signed-off-by: dependabot[bot]
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
---
go.mod | 4 ++--
go.sum | 12 ++++++------
2 files changed, 8 insertions(+), 8 deletions(-)
diff --git a/go.mod b/go.mod
index 9b1c6f780c5..22be05bb11c 100644
--- a/go.mod
+++ b/go.mod
@@ -27,7 +27,7 @@ require (
github.com/sirupsen/logrus v1.9.3
github.com/stretchr/testify v1.8.4
github.com/tsaarni/certyaml v0.9.3
- github.com/vektra/mockery/v2 v2.40.1
+ github.com/vektra/mockery/v2 v2.40.3
go.uber.org/automaxprocs v1.5.3
golang.org/x/oauth2 v0.17.0
gonum.org/v1/plot v0.14.0
@@ -128,7 +128,7 @@ require (
golang.org/x/term v0.17.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/time v0.5.0 // indirect
- golang.org/x/tools v0.16.1 // indirect
+ golang.org/x/tools v0.17.0 // indirect
gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect
google.golang.org/appengine v1.6.8 // indirect
google.golang.org/genproto v0.0.0-20240102182953-50ed04b92917 // indirect
diff --git a/go.sum b/go.sum
index 4f404a8fbc5..479bad3f806 100644
--- a/go.sum
+++ b/go.sum
@@ -381,8 +381,8 @@ github.com/tsaarni/certyaml v0.9.3 h1:m8HHbuUzWVUOmv8IQU9HgVZZ8r5ICExKm++54DJKCs
github.com/tsaarni/certyaml v0.9.3/go.mod h1:hhuU1qYr5re488geArUP4gZWqMUMqGlj4HA2qUyGYLk=
github.com/tsaarni/x500dn v1.0.0 h1:LvaWTkqRpse4VHBhB5uwf3wytokK4vF9IOyNAEyiA+U=
github.com/tsaarni/x500dn v1.0.0/go.mod h1:QaHa3EcUKC4dfCAZmj8+ZRGLKukWgpGv9H3oOCsAbcE=
-github.com/vektra/mockery/v2 v2.40.1 h1:8D01rBqloDLDHKZGXkyUD9Yj5Z+oDXBqDZ+tRXYM/oA=
-github.com/vektra/mockery/v2 v2.40.1/go.mod h1:dPzGtjT0/Uu4hqpF6QNHwz+GLago7lq1bxdj9wHbGKo=
+github.com/vektra/mockery/v2 v2.40.3 h1:IZ2lydSDFsY0khnEsbSu13VLcqSsa6UYSS/8F+uOJmo=
+github.com/vektra/mockery/v2 v2.40.3/go.mod h1:KYBZF/7sqOa86BaOZPYsoCZWEWLS90a5oBLg2pVudxY=
github.com/xhit/go-str2duration/v2 v2.1.0 h1:lxklc02Drh6ynqX+DdPyp5pCKLUQpRT8bp8Ydu2Bstc=
github.com/xhit/go-str2duration/v2 v2.1.0/go.mod h1:ohY8p+0f07DiV6Em5LKB0s2YpLtXVyJfNt1+BlmyAsU=
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
@@ -517,8 +517,8 @@ golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJ
golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
-golang.org/x/sync v0.5.0 h1:60k92dhOjHxJkrqnwsfl8KuaHbn/5dl0lUPUklKo3qE=
-golang.org/x/sync v0.5.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
+golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ=
+golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -640,8 +640,8 @@ golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4f
golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
-golang.org/x/tools v0.16.1 h1:TLyB3WofjdOEepBHAU20JdNC1Zbg87elYofWYAY5oZA=
-golang.org/x/tools v0.16.1/go.mod h1:kYVVN6I1mBNoB1OX+noeBjbRk4IUEPa7JJ+TJMEooJ0=
+golang.org/x/tools v0.17.0 h1:FvmRgNOcs3kOa+T20R1uhfP9F6HgG2mfxDv1vrx1Htc=
+golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
From 92d53429e7df957ca721fa4ef5a3b926149733b8 Mon Sep 17 00:00:00 2001
From: izturn <44051386+izturn@users.noreply.github.com>
Date: Mon, 12 Feb 2024 23:44:34 +0800
Subject: [PATCH 19/83] update docs & examples to match gateway-api/v1 (#5925)
Signed-off-by: gang.liu
---
README.md | 2 +-
.../gatewayapi/blue-green/blue-green.yaml | 2 +-
.../example-workload/gatewayapi/kuard/kuard.yaml | 2 +-
examples/gateway/03-gatewayclass.yaml | 2 +-
examples/gateway/04-gateway.yaml | 2 +-
examples/render/contour-gateway.yaml | 4 ++--
internal/sorter/sorter_test.go | 2 +-
site/content/docs/main/config/gateway-api.md | 4 ++--
site/content/docs/main/guides/gateway-api.md | 8 ++++----
site/content/getting-started/_index.md | 6 +++---
site/content/guides/gateway-api.md | 12 ++++++------
test/scripts/run-gateway-conformance.sh | 2 +-
12 files changed, 24 insertions(+), 24 deletions(-)
diff --git a/README.md b/README.md
index ac72100c733..3c72d2ad4da 100644
--- a/README.md
+++ b/README.md
@@ -16,7 +16,7 @@ Contour supports multiple configuration APIs in order to meet the needs of as ma
- **[Ingress](https://kubernetes.io/docs/concepts/services-networking/ingress/)** - A stable upstream API that enables basic ingress use cases.
- **[HTTPProxy](https://projectcontour.io/docs/main/config/fundamentals/)** - Contour's Custom Resource Definition (CRD) which expands upon the functionality of the Ingress API to allow for a richer user experience as well as solve shortcomings in the original design.
-- **[Gateway API](https://gateway-api.sigs.k8s.io/)** (beta) - A new CRD-based API managed by the [Kubernetes SIG-Network community](https://github.com/kubernetes/community/tree/master/sig-network) that aims to evolve Kubernetes service networking APIs in a vendor-neutral way.
+- **[Gateway API](https://gateway-api.sigs.k8s.io/)** - A new CRD-based API managed by the [Kubernetes SIG-Network community](https://github.com/kubernetes/community/tree/master/sig-network) that aims to evolve Kubernetes service networking APIs in a vendor-neutral way.
## Prerequisites
diff --git a/examples/example-workload/gatewayapi/blue-green/blue-green.yaml b/examples/example-workload/gatewayapi/blue-green/blue-green.yaml
index 3a5cbec2bf8..7aceaa46fca 100644
--- a/examples/example-workload/gatewayapi/blue-green/blue-green.yaml
+++ b/examples/example-workload/gatewayapi/blue-green/blue-green.yaml
@@ -96,7 +96,7 @@ spec:
---
kind: HTTPRoute
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: deploy
namespace: projectcontour
diff --git a/examples/example-workload/gatewayapi/kuard/kuard.yaml b/examples/example-workload/gatewayapi/kuard/kuard.yaml
index 6463a9a16b8..5b589e6cfd7 100644
--- a/examples/example-workload/gatewayapi/kuard/kuard.yaml
+++ b/examples/example-workload/gatewayapi/kuard/kuard.yaml
@@ -37,7 +37,7 @@ spec:
type: ClusterIP
---
kind: HTTPRoute
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: kuard
namespace: default
diff --git a/examples/gateway/03-gatewayclass.yaml b/examples/gateway/03-gatewayclass.yaml
index 91f7896a737..3e1f52d5a94 100644
--- a/examples/gateway/03-gatewayclass.yaml
+++ b/examples/gateway/03-gatewayclass.yaml
@@ -1,6 +1,6 @@
---
kind: GatewayClass
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: example
spec:
diff --git a/examples/gateway/04-gateway.yaml b/examples/gateway/04-gateway.yaml
index e130cefc0fc..4a1f76cddfc 100644
--- a/examples/gateway/04-gateway.yaml
+++ b/examples/gateway/04-gateway.yaml
@@ -1,6 +1,6 @@
---
kind: Gateway
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: contour
namespace: projectcontour
diff --git a/examples/render/contour-gateway.yaml b/examples/render/contour-gateway.yaml
index a4e9d4efd53..48672da8e17 100644
--- a/examples/render/contour-gateway.yaml
+++ b/examples/render/contour-gateway.yaml
@@ -21043,7 +21043,7 @@ status:
---
kind: GatewayClass
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: example
spec:
@@ -21051,7 +21051,7 @@ spec:
---
kind: Gateway
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: contour
namespace: projectcontour
diff --git a/internal/sorter/sorter_test.go b/internal/sorter/sorter_test.go
index a362a824c55..9e20a3f8941 100644
--- a/internal/sorter/sorter_test.go
+++ b/internal/sorter/sorter_test.go
@@ -203,7 +203,7 @@ func presentQueryParam(name string) dag.QueryParamMatchCondition {
// others that have identical path matches, number of header matches, and
// number of query matches.
// This is mainly to support Gateway API route match preference.
-// See: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.HTTPRouteRule
+// See: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1.HTTPRouteRule
func TestSortRoutesPriority(t *testing.T) {
want := []*dag.Route{
{
diff --git a/site/content/docs/main/config/gateway-api.md b/site/content/docs/main/config/gateway-api.md
index e6cba341b6a..af67fea4339 100644
--- a/site/content/docs/main/config/gateway-api.md
+++ b/site/content/docs/main/config/gateway-api.md
@@ -143,7 +143,7 @@ A simple example of a parameterized Contour GatewayClass that provisions Envoy a
```yaml
kind: GatewayClass
-apiVersion: gateway.networking.k8s.io/v1beta1
+apiVersion: gateway.networking.k8s.io/v1
metadata:
name: contour-with-envoy-deployment
spec:
@@ -214,7 +214,7 @@ containers:
[7]: https://projectcontour.io/docs/main/config/api/#projectcontour.io/v1alpha1.GatewayConfig
[8]: https://gateway-api.sigs.k8s.io/api-types/gatewayclass/#gatewayclass-controller-selection
[9]: https://projectcontour.io/quickstart/contour-gateway-provisioner.yaml
-[10]: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1beta1.GatewayClass
+[10]: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1.GatewayClass
[11]: https://gateway-api.sigs.k8s.io/concepts/api-overview/#route-resources
[12]: /docs/{{< param version >}}/guides/gateway-api
[13]: https://github.com/projectcontour/contour/issues/5970
diff --git a/site/content/docs/main/guides/gateway-api.md b/site/content/docs/main/guides/gateway-api.md
index 62599fc78cd..0759696d3f6 100644
--- a/site/content/docs/main/guides/gateway-api.md
+++ b/site/content/docs/main/guides/gateway-api.md
@@ -27,7 +27,7 @@ Create a GatewayClass:
```shell
kubectl apply -f - <
Date: Mon, 12 Feb 2024 11:49:40 -0500
Subject: [PATCH 20/83] Site/changelog/workflow updates for v1.28.0 release
(#6180)
Signed-off-by: Sunjay Bhatia
---
.github/dependabot.yml | 12 +-
.github/workflows/trivy-scan.yaml | 2 +-
changelogs/CHANGELOG-v1.28.0.md | 198 +
changelogs/unreleased/5828-KauzClay-major.md | 9 -
changelogs/unreleased/5840-izturn-minor.md | 3 -
.../unreleased/5849-KauzClay-deprecation.md | 6 -
changelogs/unreleased/5849-KauzClay-minor.md | 5 -
changelogs/unreleased/5898-skriss-major.md | 8 -
changelogs/unreleased/5934-izturn-minor.md | 3 -
changelogs/unreleased/5961-izturn-small.md | 1 -
.../unreleased/5968-skriss-deprecation.md | 1 -
changelogs/unreleased/5968-skriss-small.md | 1 -
changelogs/unreleased/5969-skriss-small.md | 1 -
changelogs/unreleased/5972-skriss-small.md | 1 -
changelogs/unreleased/5997-izturn-minor.md | 3 -
changelogs/unreleased/6013-davinci26-minor.md | 3 -
changelogs/unreleased/6016-relu-small.md | 1 -
changelogs/unreleased/6031-skriss-small.md | 1 -
.../unreleased/6058-flawedmatrix-minor.md | 5 -
.../unreleased/6063-lubronzhan-small.md | 1 -
.../unreleased/6070-sunjayBhatia-small.md | 1 -
.../unreleased/6073-lubronzhan-small.md | 1 -
.../unreleased/6077-abbas-gheydi-small.md | 1 -
changelogs/unreleased/6079-tsaarni-minor.md | 4 -
.../unreleased/6099-sunjayBhatia-minor.md | 5 -
.../unreleased/6119-flawedmatrix-major.md | 9 -
changelogs/unreleased/6124-skriss-small.md | 1 -
changelogs/unreleased/6143-skriss-docs.md | 1 -
.../unreleased/6144-skriss-deprecation.md | 5 -
.../unreleased/6147-sunjayBhatia-small.md | 1 -
.../unreleased/6148-lubronzhan-minor.md | 3 -
.../unreleased/6151-christianang-small.md | 1 -
.../unreleased/6152-lubronzhan-minor.md | 7 -
.../unreleased/6161-sunjayBhatia-docs.md | 1 -
.../unreleased/6164-sunjayBhatia-small.md | 1 -
netlify.toml | 6 +-
site/config.yaml | 3 +-
site/content/docs/1.28/_index.md | 48 +
site/content/docs/1.28/architecture.md | 74 +
.../docs/1.28/config/access-logging.md | 148 +
site/content/docs/1.28/config/annotations.md | 98 +
.../docs/1.28/config/api-reference.html | 9131 +++++++++++++++++
site/content/docs/1.28/config/api.md | 3 +
.../docs/1.28/config/client-authorization.md | 123 +
.../docs/1.28/config/cookie-rewriting.md | 109 +
site/content/docs/1.28/config/cors.md | 82 +
.../1.28/config/external-service-routing.md | 47 +
site/content/docs/1.28/config/fundamentals.md | 197 +
site/content/docs/1.28/config/gateway-api.md | 221 +
.../content/docs/1.28/config/health-checks.md | 160 +
.../docs/1.28/config/inclusion-delegation.md | 139 +
site/content/docs/1.28/config/ingress.md | 94 +
site/content/docs/1.28/config/ip-filtering.md | 80 +
.../docs/1.28/config/jwt-verification.md | 182 +
.../docs/1.28/config/overload-manager.md | 30 +
.../content/docs/1.28/config/rate-limiting.md | 366 +
.../docs/1.28/config/request-rewriting.md | 337 +
.../docs/1.28/config/request-routing.md | 535 +
site/content/docs/1.28/config/slow-start.md | 39 +
.../docs/1.28/config/tls-delegation.md | 79 +
.../docs/1.28/config/tls-termination.md | 353 +
site/content/docs/1.28/config/tracing.md | 117 +
site/content/docs/1.28/config/upstream-tls.md | 108 +
.../content/docs/1.28/config/virtual-hosts.md | 138 +
site/content/docs/1.28/config/websockets.md | 27 +
site/content/docs/1.28/configuration.md | 541 +
site/content/docs/1.28/deploy-options.md | 383 +
site/content/docs/1.28/github.md | 80 +
site/content/docs/1.28/grpc-tls-howto.md | 169 +
site/content/docs/1.28/guides/_index.md | 9 +
site/content/docs/1.28/guides/cert-manager.md | 670 ++
.../docs/1.28/guides/deploy-aws-nlb.md | 47 +
.../docs/1.28/guides/deploy-aws-tls-nlb.md | 135 +
.../1.28/guides/external-authorization.md | 538 +
site/content/docs/1.28/guides/fips.md | 169 +
site/content/docs/1.28/guides/gatekeeper.md | 456 +
site/content/docs/1.28/guides/gateway-api.md | 210 +
.../docs/1.28/guides/global-rate-limiting.md | 503 +
site/content/docs/1.28/guides/grpc.md | 225 +
.../docs/1.28/guides/health-checking.md | 11 +
site/content/docs/1.28/guides/kind.md | 63 +
.../content/docs/1.28/guides/metrics/table.md | 20 +
site/content/docs/1.28/guides/prometheus.md | 94 +
site/content/docs/1.28/guides/proxy-proto.md | 53 +
.../docs/1.28/guides/resource-limits.md | 161 +
site/content/docs/1.28/img/archoverview.png | Bin 0 -> 78807 bytes
.../1.28/img/contour_deployment_in_k8s.png | Bin 0 -> 134492 bytes
.../content/docs/1.28/img/shutdownmanager.png | Bin 0 -> 51051 bytes
.../1.28/img/source/shutdownmanager.drawio | 1 +
site/content/docs/1.28/redeploy-envoy.md | 72 +
site/content/docs/1.28/start-contributing.md | 130 +
site/content/docs/1.28/troubleshooting.md | 41 +
.../troubleshooting/common-proxy-errors.md | 96 +
.../1.28/troubleshooting/contour-debug-log.md | 6 +
.../1.28/troubleshooting/contour-graph.md | 25 +
.../troubleshooting/contour-xds-resources.md | 19 +
.../troubleshooting/envoy-admin-interface.md | 32 +
.../envoy-container-draining.md | 29 +
.../1.28/troubleshooting/envoy-debug-log.md | 8 +
.../1.28/troubleshooting/profiling-contour.md | 14 +
.../content/resources/compatibility-matrix.md | 1 +
site/data/docs/1-28-toc.yml | 151 +
site/data/docs/toc-mapping.yml | 1 +
versions.yaml | 12 +-
104 files changed, 18449 insertions(+), 107 deletions(-)
create mode 100644 changelogs/CHANGELOG-v1.28.0.md
delete mode 100644 changelogs/unreleased/5828-KauzClay-major.md
delete mode 100644 changelogs/unreleased/5840-izturn-minor.md
delete mode 100644 changelogs/unreleased/5849-KauzClay-deprecation.md
delete mode 100644 changelogs/unreleased/5849-KauzClay-minor.md
delete mode 100644 changelogs/unreleased/5898-skriss-major.md
delete mode 100644 changelogs/unreleased/5934-izturn-minor.md
delete mode 100644 changelogs/unreleased/5961-izturn-small.md
delete mode 100644 changelogs/unreleased/5968-skriss-deprecation.md
delete mode 100644 changelogs/unreleased/5968-skriss-small.md
delete mode 100644 changelogs/unreleased/5969-skriss-small.md
delete mode 100644 changelogs/unreleased/5972-skriss-small.md
delete mode 100644 changelogs/unreleased/5997-izturn-minor.md
delete mode 100644 changelogs/unreleased/6013-davinci26-minor.md
delete mode 100644 changelogs/unreleased/6016-relu-small.md
delete mode 100644 changelogs/unreleased/6031-skriss-small.md
delete mode 100644 changelogs/unreleased/6058-flawedmatrix-minor.md
delete mode 100644 changelogs/unreleased/6063-lubronzhan-small.md
delete mode 100644 changelogs/unreleased/6070-sunjayBhatia-small.md
delete mode 100644 changelogs/unreleased/6073-lubronzhan-small.md
delete mode 100644 changelogs/unreleased/6077-abbas-gheydi-small.md
delete mode 100644 changelogs/unreleased/6079-tsaarni-minor.md
delete mode 100644 changelogs/unreleased/6099-sunjayBhatia-minor.md
delete mode 100644 changelogs/unreleased/6119-flawedmatrix-major.md
delete mode 100644 changelogs/unreleased/6124-skriss-small.md
delete mode 100644 changelogs/unreleased/6143-skriss-docs.md
delete mode 100644 changelogs/unreleased/6144-skriss-deprecation.md
delete mode 100644 changelogs/unreleased/6147-sunjayBhatia-small.md
delete mode 100644 changelogs/unreleased/6148-lubronzhan-minor.md
delete mode 100644 changelogs/unreleased/6151-christianang-small.md
delete mode 100644 changelogs/unreleased/6152-lubronzhan-minor.md
delete mode 100644 changelogs/unreleased/6161-sunjayBhatia-docs.md
delete mode 100644 changelogs/unreleased/6164-sunjayBhatia-small.md
create mode 100644 site/content/docs/1.28/_index.md
create mode 100644 site/content/docs/1.28/architecture.md
create mode 100644 site/content/docs/1.28/config/access-logging.md
create mode 100644 site/content/docs/1.28/config/annotations.md
create mode 100644 site/content/docs/1.28/config/api-reference.html
create mode 100644 site/content/docs/1.28/config/api.md
create mode 100644 site/content/docs/1.28/config/client-authorization.md
create mode 100644 site/content/docs/1.28/config/cookie-rewriting.md
create mode 100644 site/content/docs/1.28/config/cors.md
create mode 100644 site/content/docs/1.28/config/external-service-routing.md
create mode 100644 site/content/docs/1.28/config/fundamentals.md
create mode 100644 site/content/docs/1.28/config/gateway-api.md
create mode 100644 site/content/docs/1.28/config/health-checks.md
create mode 100644 site/content/docs/1.28/config/inclusion-delegation.md
create mode 100644 site/content/docs/1.28/config/ingress.md
create mode 100644 site/content/docs/1.28/config/ip-filtering.md
create mode 100644 site/content/docs/1.28/config/jwt-verification.md
create mode 100644 site/content/docs/1.28/config/overload-manager.md
create mode 100644 site/content/docs/1.28/config/rate-limiting.md
create mode 100644 site/content/docs/1.28/config/request-rewriting.md
create mode 100644 site/content/docs/1.28/config/request-routing.md
create mode 100644 site/content/docs/1.28/config/slow-start.md
create mode 100644 site/content/docs/1.28/config/tls-delegation.md
create mode 100644 site/content/docs/1.28/config/tls-termination.md
create mode 100644 site/content/docs/1.28/config/tracing.md
create mode 100644 site/content/docs/1.28/config/upstream-tls.md
create mode 100644 site/content/docs/1.28/config/virtual-hosts.md
create mode 100644 site/content/docs/1.28/config/websockets.md
create mode 100644 site/content/docs/1.28/configuration.md
create mode 100644 site/content/docs/1.28/deploy-options.md
create mode 100644 site/content/docs/1.28/github.md
create mode 100644 site/content/docs/1.28/grpc-tls-howto.md
create mode 100644 site/content/docs/1.28/guides/_index.md
create mode 100644 site/content/docs/1.28/guides/cert-manager.md
create mode 100644 site/content/docs/1.28/guides/deploy-aws-nlb.md
create mode 100644 site/content/docs/1.28/guides/deploy-aws-tls-nlb.md
create mode 100644 site/content/docs/1.28/guides/external-authorization.md
create mode 100644 site/content/docs/1.28/guides/fips.md
create mode 100644 site/content/docs/1.28/guides/gatekeeper.md
create mode 100644 site/content/docs/1.28/guides/gateway-api.md
create mode 100644 site/content/docs/1.28/guides/global-rate-limiting.md
create mode 100644 site/content/docs/1.28/guides/grpc.md
create mode 100644 site/content/docs/1.28/guides/health-checking.md
create mode 100644 site/content/docs/1.28/guides/kind.md
create mode 100644 site/content/docs/1.28/guides/metrics/table.md
create mode 100644 site/content/docs/1.28/guides/prometheus.md
create mode 100644 site/content/docs/1.28/guides/proxy-proto.md
create mode 100644 site/content/docs/1.28/guides/resource-limits.md
create mode 100644 site/content/docs/1.28/img/archoverview.png
create mode 100644 site/content/docs/1.28/img/contour_deployment_in_k8s.png
create mode 100644 site/content/docs/1.28/img/shutdownmanager.png
create mode 100644 site/content/docs/1.28/img/source/shutdownmanager.drawio
create mode 100644 site/content/docs/1.28/redeploy-envoy.md
create mode 100644 site/content/docs/1.28/start-contributing.md
create mode 100644 site/content/docs/1.28/troubleshooting.md
create mode 100644 site/content/docs/1.28/troubleshooting/common-proxy-errors.md
create mode 100644 site/content/docs/1.28/troubleshooting/contour-debug-log.md
create mode 100644 site/content/docs/1.28/troubleshooting/contour-graph.md
create mode 100644 site/content/docs/1.28/troubleshooting/contour-xds-resources.md
create mode 100644 site/content/docs/1.28/troubleshooting/envoy-admin-interface.md
create mode 100644 site/content/docs/1.28/troubleshooting/envoy-container-draining.md
create mode 100644 site/content/docs/1.28/troubleshooting/envoy-debug-log.md
create mode 100644 site/content/docs/1.28/troubleshooting/profiling-contour.md
create mode 100644 site/data/docs/1-28-toc.yml
diff --git a/.github/dependabot.yml b/.github/dependabot.yml
index c4fd727abe7..0886364fb0b 100644
--- a/.github/dependabot.yml
+++ b/.github/dependabot.yml
@@ -37,7 +37,7 @@ updates:
- "actions/download-artifact"
# release branch N targets
-- target-branch: release-1.27
+- target-branch: release-1.28
package-ecosystem: "gomod"
directory: "/"
schedule:
@@ -57,7 +57,7 @@ updates:
k8s-dependencies:
patterns:
- "k8s.io/*"
-- target-branch: release-1.27
+- target-branch: release-1.28
package-ecosystem: "github-actions"
directory: "/"
schedule:
@@ -80,7 +80,7 @@ updates:
- "actions/download-artifact"
# release branch N-1 targets
-- target-branch: release-1.26
+- target-branch: release-1.27
package-ecosystem: "gomod"
directory: "/"
schedule:
@@ -100,7 +100,7 @@ updates:
k8s-dependencies:
patterns:
- "k8s.io/*"
-- target-branch: release-1.26
+- target-branch: release-1.27
package-ecosystem: "github-actions"
directory: "/"
schedule:
@@ -123,7 +123,7 @@ updates:
- "actions/download-artifact"
# release branch N-2 targets
-- target-branch: release-1.25
+- target-branch: release-1.26
package-ecosystem: "gomod"
directory: "/"
schedule:
@@ -143,7 +143,7 @@ updates:
k8s-dependencies:
patterns:
- "k8s.io/*"
-- target-branch: release-1.25
+- target-branch: release-1.26
package-ecosystem: "github-actions"
directory: "/"
schedule:
diff --git a/.github/workflows/trivy-scan.yaml b/.github/workflows/trivy-scan.yaml
index 29478cbb0ad..6f907cb33b8 100644
--- a/.github/workflows/trivy-scan.yaml
+++ b/.github/workflows/trivy-scan.yaml
@@ -16,9 +16,9 @@ jobs:
matrix:
branch:
- main
+ - release-1.28
- release-1.27
- release-1.26
- - release-1.25
runs-on: ubuntu-latest
permissions:
security-events: write
diff --git a/changelogs/CHANGELOG-v1.28.0.md b/changelogs/CHANGELOG-v1.28.0.md
new file mode 100644
index 00000000000..12517e6daf0
--- /dev/null
+++ b/changelogs/CHANGELOG-v1.28.0.md
@@ -0,0 +1,198 @@
+We are delighted to present version v1.28.0 of Contour, our layer 7 HTTP reverse proxy for Kubernetes clusters.
+
+A big thank you to everyone who contributed to the release.
+
+
+- [Major Changes](#major-changes)
+- [Minor Changes](#minor-changes)
+- [Other Changes](#other-changes)
+- [Docs Changes](#docs-changes)
+- [Deprecations/Removals](#deprecation-and-removal-notices)
+- [Installing/Upgrading](#installing-and-upgrading)
+- [Compatible Kubernetes Versions](#compatible-kubernetes-versions)
+- [Community Thanks!](#community-thanks)
+
+# Major Changes
+
+## Upstream TLS now supports TLS 1.3 and TLS parameters can be configured
+
+The default maximum TLS version for upstream connections is now 1.3, instead of the Envoy default of 1.2.
+
+In a similar way to how Contour users can configure Min/Max TLS version and
+Cipher Suites for Envoy's listeners, users can now specify the
+same information for upstream connections. In the ContourConfiguration, this is
+available under `spec.envoy.cluster.upstreamTLS`. The equivalent config file
+parameter is `cluster.upstream-tls`.
+
+(#5828, @KauzClay)
+
+## Update to Gateway API 1.0
+
+Contour now uses [Gateway API 1.0](https://github.com/kubernetes-sigs/gateway-api/releases/tag/v1.0.0), which graduates the core resources GatewayClass, Gateway and HTTPRoute to the `v1` API version.
+
+For backwards compatibility, this version of Contour continues to watch for `v1beta1` versions of these resources, to ease the migration process for users.
+However, future versions of Contour will move to watching for `v1` versions of these resources.
+Note that if you are using Gateway API 1.0 and the `v1` API group, the resources you create will also be available from the API server as `v1beta1` resources so Contour will correctly reconcile them as well.
+
+(#5898, @skriss)
+
+## Support for Gateway API BackendTLSPolicy
+
+The BackendTLSPolicy CRD can now be used with HTTPRoute to configure a Contour gateway to connect to a backend Service with TLS. This will give users the ability to use Gateway API to configure their routes to securely connect to backends that use TLS with Contour.
+
+The BackendTLSPolicy spec requires you to specify a `targetRef`, which can currently only be a Kubernetes Service within the same namespace as the BackendTLSPolicy. The targetRef is what Service should be watched to apply the BackendTLSPolicy to. A `SectionName` can also be configured to the port name of a Service to reference a specific section of the Service.
+
+The spec also requires you to specify `caCertRefs`, which can either be a ConfigMap or Secret with a `ca.crt` key in the data map containing a PEM-encoded TLS certificate. The CA certificates referenced will be configured to be used by the gateway to perform TLS to the backend Service. You will also need to specify a `Hostname`, which will be used to configure the SNI the gateway will use for the connection.
+
+See Gateway API's [GEP-1897](https://gateway-api.sigs.k8s.io/geps/gep-1897) for the proposal for BackendTLSPolicy.
+
+(#6119, @flawedmatrix, @christianang)
+
+
+# Minor Changes
+
+## JWT Authentication happens before External Authorization
+
+Fixes a bug where when the external authorization filter and JWT authentication filter were both configured, the external authorization filter was executed _before_ the JWT authentication filter. Now, JWT authentication happens before external authorization when they are both configured.
+
+(#5840, @izturn)
+
+## Allow Multiple SANs in Upstream Validation section of HTTPProxy
+
+This change introduces a max length of 250 characters to the field `subjectName` in the UpstreamValidation block.
+
+Allow multiple SANs in Upstream Validation by adding a new field `subjectNames` to the UpstreamValidtion block. This will exist side by side with the previous `subjectName` field. Using CEL validation, we can enforce that when both are present, the first entry in `subjectNames` must match the value of `subjectName`.
+
+(#5849, @KauzClay)
+
+## Gateway API Backend Protocol Selection
+
+For Gateway API, Contour now enables end-users to specify backend protocols by setting the backend Service's [ServicePort.AppProtocol](https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol) parameter. The accepted values are `kubernetes.io/h2c` and `kubernetes.io/ws`. Note that websocket upgrades are already enabled by default for Gateway API. If `AppProtocol` is set, any other configurations, such as the annotation: `projectcontour.io/upstream-protocol.{protocol}` will be disregarded.
+
+(#5934, @izturn)
+
+## Gateway API: support HTTPRoute request timeouts
+
+Contour now enables end-users to specify request timeouts by setting the [HTTPRouteRule.Timeouts.Request](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.HTTPRouteTimeouts) parameter. Note that `BackendRequest` is not yet implemented because without Gateway API support for retries, it's functionally equivalent to `Request`.
+
+(#5997, @izturn)
+
+## Support for Global Circuit Breaker Policy
+
+The way [circuit-breaker-annotations](https://projectcontour.io/docs/1.27/config/annotations/) work currently is that when not present they are being defaulted to Envoy defaults. The Envoy defaults can be quite low for larger clusters with more traffic so if a user accidentally deletes them or unset them this cause an issue. With this change we are providing contour administrators the ability to provide global defaults that are good. In that case even if the user forgets to set them or deletes them they can have the safety net of good defaults. They can be configured via [cluster.circuit-breakers](https://projectcontour.io/docs/1.28/configuration/#circuit-breakers) or via `ContourConfiguration`` CRD in [spec.envoy.cluster.circuitBreakers](https://projectcontour.io/docs/1.28/config/api/#projectcontour.io/v1alpha1.GlobalCircuitBreakerDefaults)
+
+(#6013, @davinci26)
+
+## Allow setting connection limit per listener
+
+Adds a `listeners.max-connections-per-listener` config option to Contour config file and `spec.envoy.listener.maxConnectionsPerListener` to the ContourConfiguration CRD.
+
+Setting the max connection limit per listener field limits the number of active connections to a listener. The default, if unset, is unlimited.
+
+(#6058, @flawedmatrix, @christianang)
+
+## Upstream TLS validation and client certificate for TCPProxy
+
+TCPProxy now supports validating server certificate and using client certificate for upstream TLS connections.
+Set `httpproxy.spec.tcpproxy.services.validation.caSecret` and `subjectName` to enable optional validation and `tls.envoy-client-certificate` configuration file field or `ContourConfiguration.spec.envoy.clientCertificate` to set the optional client certificate.
+
+(#6079, @tsaarni)
+
+## Remove Contour container readiness probe initial delay
+
+The Contour Deployment Contour server container previously had its readiness probe `initialDelaySeconds` field set to 15.
+This has been removed from the example YAML manifests and Gateway Provisioner generated Contour Deployment since as of [PR #5672](https://github.com/projectcontour/contour/pull/5672) Contour's xDS server will not start or serve any configuration (and the readiness probe will not succeed) until the existing state of the cluster is synced.
+In clusters with few resources this will improve the Contour Deployment's update/rollout time as initial startup time should be low.
+
+(#6099, @sunjayBhatia)
+
+## Add anti-affinity rule for envoy deployed by provisioner
+
+The envoy deployment created by the gateway provisioner now includes a default anti-affinity rule. The anti-affinity rule in the [example envoy deployment manifest](https://github.com/projectcontour/contour/blob/main/examples/deployment/03-envoy-deployment.yaml) is also updated to `preferredDuringSchedulingIgnoredDuringExecution` to be consistent with the contour deployment and the gateway provisioner anti-affinity rule.
+
+(#6148, @lubronzhan)
+
+## Add DisabledFeatures to ContourDeployment for gateway provisioner
+
+A new flag DisabledFeatures is added to ContourDeployment so that user can configure contour which is deployed by the provisioner to skip reconciling CRDs which are specified inside the flag.
+
+Accepted values are `grpcroutes|tlsroutes|extensionservices|backendtlspolicies`.
+
+(#6152, @lubronzhan)
+
+
+# Other Changes
+- For Gateway API v1.0, the successful attachment of a Route to a Listener is based solely on the combination of the AllowedRoutes field on the corresponding Listener and the Route's ParentRefs field. (#5961, @izturn)
+- Gateway API: adds support for [Gateway infrastructure labels and annotations](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.GatewayInfrastructure)``. (#5968, @skriss)
+- Gateway API: add the `gateway.networking.k8s.io/gateway-name` label to generated resources. (#5969, @skriss)
+- Fixes a bug with the `envoy` xDS server where at startup, xDS configuration would not be generated and served until a subsequent configuration change. (#5972, @skriss)
+- Envoy: Adds support for setting [per-host circuit breaker max-connections threshold](https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-per-host-thresholds) using a new service-level annotation: `projectcontour.io/per-host-max-connections`. (#6016, @relu)
+- Updates to Kubernetes 1.29. Supported/tested Kubernetes versions are now 1.27, 1.28 and 1.29. (#6031, @skriss)
+- Remove static base runtime layer from bootstrap (#6063, @lubronzhan)
+- Updates to Go 1.21.6. See the [Go release notes](https://go.dev/doc/devel/release#go1.21.minor) for more information. (#6070, @sunjayBhatia)
+- Allow gatewayProvisioner to create contour that only watch limited namespaces of resources (#6073, @lubronzhan)
+- Access Log: Contour excludes empty fields in Envoy JSON based access logs by default. (#6077, @abbas-gheydi)
+- Updates HTTP filter names to match between the HTTP connection manager and per-filter config on virtual hosts/routes, and to use canonical names. (#6124, @skriss)
+- Gateway API provisioner now checks `gateway.networking.k8s.io/bundle-version` annotation on Gateway CRDs and sets SupportedVersion status condition on GatewayClass if annotation value matches supported Gateway API version. Best-effort support is provided if version does not match. (#6147, @sunjayBhatia)
+- For Gateway API, add "Accepted" condition to BackendTLSPolicy. If the condition is true the BackendTLSPolicy was accepted by the Gateway and if false a reason will be stated on the policy as to why it wasn't accepted. (#6151, @christianang)
+- Updates Envoy to v1.29.1. See the release notes [here](https://www.envoyproxy.io/docs/envoy/v1.29.1/version_history/v1.29/v1.29.1). (#6164, @sunjayBhatia)
+
+
+# Docs Changes
+- Document that Gateway names should be 63 characters or shorter to avoid issues with generating dependent resources when using the Gateway provisioner. (#6143, @skriss)
+- Add troubleshooting guide for general app traffic errors. (#6161, @sunjayBhatia)
+
+
+# Deprecation and Removal Notices
+
+
+## Deprecate `subjectName` field on UpstreamValidation
+
+The `subjectName` field is being deprecated in favor of `subjectNames`, which is
+an list of subjectNames. `subjectName` will continue to behave as it has. If
+using `subjectNames`, the first entry in `subjectNames` must match the value of
+`subjectName`. this will be enforced by CEL validation.
+
+(#5849, @KauzClay)
+
+## ContourDeployment.Spec.ResourceLabels is deprecated
+
+The `ContourDeployment.Spec.ResourceLabels` field is now deprecated. You should use `Gateway.Spec.Infrastructure.Labels` instead. The `ResourceLabels` field will be removed in a future release.
+
+(#5968, @skriss)
+
+## Configuring Contour with a GatewayClass controller name is deprecated
+
+Contour should no longer be configured with a GatewayClass controller name (`gateway.controllerName` in the config file or ContourConfiguration CRD).
+Instead, either use a specific Gateway reference (`gateway.gatewayRef`), or use the Gateway provisioner.
+`gateway.controllerName` will be removed in a future release.
+
+(#6144, @skriss)
+
+
+# Installing and Upgrading
+
+For a fresh install of Contour, consult the [getting started documentation](https://projectcontour.io/getting-started/).
+
+To upgrade an existing Contour installation, please consult the [upgrade documentation](https://projectcontour.io/resources/upgrading/).
+
+
+# Compatible Kubernetes Versions
+
+Contour v1.28.0 is tested against Kubernetes 1.27 through 1.29.
+
+# Community Thanks!
+We’re immensely grateful for all the community contributions that help make Contour even better! For this release, special thanks go out to the following contributors:
+
+- @KauzClay
+- @abbas-gheydi
+- @christianang
+- @davinci26
+- @flawedmatrix
+- @izturn
+- @lubronzhan
+- @relu
+
+
+# Are you a Contour user? We would love to know!
+If you're using Contour and want to add your organization to our adopters list, please visit this [page](https://github.com/projectcontour/contour/blob/master/ADOPTERS.md). If you prefer to keep your organization name anonymous but still give us feedback into your usage and scenarios for Contour, please post on this [GitHub thread](https://github.com/projectcontour/contour/issues/1269).
diff --git a/changelogs/unreleased/5828-KauzClay-major.md b/changelogs/unreleased/5828-KauzClay-major.md
deleted file mode 100644
index 707f24a4cd0..00000000000
--- a/changelogs/unreleased/5828-KauzClay-major.md
+++ /dev/null
@@ -1,9 +0,0 @@
-## Upstream TLS now supports TLS 1.3 and TLS parameters can be configured
-
-The default maximum TLS version for upstream connections is now 1.3, instead of the Envoy default of 1.2.
-
-In a similar way to how Contour users can configure Min/Max TLS version and
-Cipher Suites for Envoy's listeners, users can now specify the
-same information for upstream connections. In the ContourConfiguration, this is
-available under `spec.envoy.cluster.upstreamTLS`. The equivalent config file
-parameter is `cluster.upstream-tls`.
diff --git a/changelogs/unreleased/5840-izturn-minor.md b/changelogs/unreleased/5840-izturn-minor.md
deleted file mode 100644
index e7ef76d5755..00000000000
--- a/changelogs/unreleased/5840-izturn-minor.md
+++ /dev/null
@@ -1,3 +0,0 @@
-## JWT Authentication happens before External Authorization
-
-Fixes a bug where when the external authorization filter and JWT authentication filter were both configured, the external authorization filter was executed _before_ the JWT authentication filter. Now, JWT authentication happens before external authorization when they are both configured.
\ No newline at end of file
diff --git a/changelogs/unreleased/5849-KauzClay-deprecation.md b/changelogs/unreleased/5849-KauzClay-deprecation.md
deleted file mode 100644
index c4266be7d80..00000000000
--- a/changelogs/unreleased/5849-KauzClay-deprecation.md
+++ /dev/null
@@ -1,6 +0,0 @@
-## Deprecate `subjectName` field on UpstreamValidation
-
-The `subjectName` field is being deprecated in favor of `subjectNames`, which is
-an list of subjectNames. `subjectName` will continue to behave as it has. If
-using `subjectNames`, the first entry in `subjectNames` must match the value of
-`subjectName`. this will be enforced by CEL validation.
\ No newline at end of file
diff --git a/changelogs/unreleased/5849-KauzClay-minor.md b/changelogs/unreleased/5849-KauzClay-minor.md
deleted file mode 100644
index afa791c072b..00000000000
--- a/changelogs/unreleased/5849-KauzClay-minor.md
+++ /dev/null
@@ -1,5 +0,0 @@
-## Allow Multiple SANs in Upstream Validation section of HTTPProxy
-
-This change introduces a max length of 250 characters to the field `subjectName` in the UpstreamValidation block.
-
-Allow multiple SANs in Upstream Validation by adding a new field `subjectNames` to the UpstreamValidtion block. This will exist side by side with the previous `subjectName` field. Using CEL validation, we can enforce that when both are present, the first entry in `subjectNames` must match the value of `subjectName`.
\ No newline at end of file
diff --git a/changelogs/unreleased/5898-skriss-major.md b/changelogs/unreleased/5898-skriss-major.md
deleted file mode 100644
index b4c8adc0c04..00000000000
--- a/changelogs/unreleased/5898-skriss-major.md
+++ /dev/null
@@ -1,8 +0,0 @@
-## Update to Gateway API 1.0
-
-Contour now uses [Gateway API 1.0](https://github.com/kubernetes-sigs/gateway-api/releases/tag/v1.0.0), which graduates the core resources GatewayClass, Gateway and HTTPRoute to the `v1` API version.
-
-For backwards compatibility, this version of Contour continues to watch for `v1beta1` versions of these resources, to ease the migration process for users.
-However, future versions of Contour will move to watching for `v1` versions of these resources.
-Note that if you are using Gateway API 1.0 and the `v1` API group, the resources you create will also be available from the API server as `v1beta1` resources so Contour will correctly reconcile them as well.
-
diff --git a/changelogs/unreleased/5934-izturn-minor.md b/changelogs/unreleased/5934-izturn-minor.md
deleted file mode 100644
index 3d8504126cf..00000000000
--- a/changelogs/unreleased/5934-izturn-minor.md
+++ /dev/null
@@ -1,3 +0,0 @@
-## Gateway API Backend Protocol Selection
-
-For Gateway API, Contour now enables end-users to specify backend protocols by setting the backend Service's [ServicePort.AppProtocol](https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol) parameter. The accepted values are `kubernetes.io/h2c` and `kubernetes.io/ws`. Note that websocket upgrades are already enabled by default for Gateway API. If `AppProtocol` is set, any other configurations, such as the annotation: `projectcontour.io/upstream-protocol.{protocol}` will be disregarded.
\ No newline at end of file
diff --git a/changelogs/unreleased/5961-izturn-small.md b/changelogs/unreleased/5961-izturn-small.md
deleted file mode 100644
index 130c664704c..00000000000
--- a/changelogs/unreleased/5961-izturn-small.md
+++ /dev/null
@@ -1 +0,0 @@
-For Gateway API v1.0, the successful attachment of a Route to a Listener is based solely on the combination of the AllowedRoutes field on the corresponding Listener and the Route's ParentRefs field.
diff --git a/changelogs/unreleased/5968-skriss-deprecation.md b/changelogs/unreleased/5968-skriss-deprecation.md
deleted file mode 100644
index 6146bec7461..00000000000
--- a/changelogs/unreleased/5968-skriss-deprecation.md
+++ /dev/null
@@ -1 +0,0 @@
-The `ContourDeployment.Spec.ResourceLabels` field is now deprecated. You should use `Gateway.Spec.Infrastructure.Labels` instead. The `ResourceLabels` field will be removed in a future release.
\ No newline at end of file
diff --git a/changelogs/unreleased/5968-skriss-small.md b/changelogs/unreleased/5968-skriss-small.md
deleted file mode 100644
index a0e80d9f346..00000000000
--- a/changelogs/unreleased/5968-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Gateway API: adds support for [Gateway infrastructure labels and annotations](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.GatewayInfrastructure)``.
\ No newline at end of file
diff --git a/changelogs/unreleased/5969-skriss-small.md b/changelogs/unreleased/5969-skriss-small.md
deleted file mode 100644
index 064a7f3f4df..00000000000
--- a/changelogs/unreleased/5969-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Gateway API: add the `gateway.networking.k8s.io/gateway-name` label to generated resources.
\ No newline at end of file
diff --git a/changelogs/unreleased/5972-skriss-small.md b/changelogs/unreleased/5972-skriss-small.md
deleted file mode 100644
index 84322581ce0..00000000000
--- a/changelogs/unreleased/5972-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Fixes a bug with the `envoy` xDS server where at startup, xDS configuration would not be generated and served until a subsequent configuration change.
\ No newline at end of file
diff --git a/changelogs/unreleased/5997-izturn-minor.md b/changelogs/unreleased/5997-izturn-minor.md
deleted file mode 100644
index efd30f164c4..00000000000
--- a/changelogs/unreleased/5997-izturn-minor.md
+++ /dev/null
@@ -1,3 +0,0 @@
-## Gateway API: support HTTPRoute request timeouts
-
-Contour now enables end-users to specify request timeouts by setting the [HTTPRouteRule.Timeouts.Request](https://gateway-api.sigs.k8s.io/reference/spec/#gateway.networking.k8s.io/v1.HTTPRouteTimeouts) parameter. Note that `BackendRequest` is not yet implemented because without Gateway API support for retries, it's functionally equivalent to `Request`.
\ No newline at end of file
diff --git a/changelogs/unreleased/6013-davinci26-minor.md b/changelogs/unreleased/6013-davinci26-minor.md
deleted file mode 100644
index df19a22dc78..00000000000
--- a/changelogs/unreleased/6013-davinci26-minor.md
+++ /dev/null
@@ -1,3 +0,0 @@
-## Support for Global Circuit Breaker Policy
-
-The way [circuit-breaker-annotations](https://projectcontour.io/docs/1.27/config/annotations/) work currently is that when not present they are being defaulted to Envoy defaults. The Envoy defaults can be quite low for larger clusters with more traffic so if a user accidentally deletes them or unset them this cause an issue. With this change we are providing contour administrators the ability to provide global defaults that are good. In that case even if the user forgets to set them or deletes them they can have the safety net of good defaults. They can be configured via [cluster.circuit-breakers](https://projectcontour.io/docs/1.28/configuration/#circuit-breakers) or via `ContourConfiguration`` CRD in [spec.envoy.cluster.circuitBreakers](https://projectcontour.io/docs/1.28/config/api/#projectcontour.io/v1alpha1.GlobalCircuitBreakerDefaults)
diff --git a/changelogs/unreleased/6016-relu-small.md b/changelogs/unreleased/6016-relu-small.md
deleted file mode 100644
index 54f7a59fc3c..00000000000
--- a/changelogs/unreleased/6016-relu-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Envoy: Adds support for setting [per-host circuit breaker max-connections threshold](https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-per-host-thresholds) using a new service-level annotation: `projectcontour.io/per-host-max-connections`.
diff --git a/changelogs/unreleased/6031-skriss-small.md b/changelogs/unreleased/6031-skriss-small.md
deleted file mode 100644
index 71864af4250..00000000000
--- a/changelogs/unreleased/6031-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Updates to Kubernetes 1.29. Supported/tested Kubernetes versions are now 1.27, 1.28 and 1.29.
\ No newline at end of file
diff --git a/changelogs/unreleased/6058-flawedmatrix-minor.md b/changelogs/unreleased/6058-flawedmatrix-minor.md
deleted file mode 100644
index 5bd27dcdf75..00000000000
--- a/changelogs/unreleased/6058-flawedmatrix-minor.md
+++ /dev/null
@@ -1,5 +0,0 @@
-## Allow setting connection limit per listener
-
-Adds a `listeners.max-connections-per-listener` config option to Contour config file and `spec.envoy.listener.maxConnectionsPerListener` to the ContourConfiguration CRD.
-
-Setting the max connection limit per listener field limits the number of active connections to a listener. The default, if unset, is unlimited.
diff --git a/changelogs/unreleased/6063-lubronzhan-small.md b/changelogs/unreleased/6063-lubronzhan-small.md
deleted file mode 100644
index e9b526895fa..00000000000
--- a/changelogs/unreleased/6063-lubronzhan-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Remove static base runtime layer from bootstrap
diff --git a/changelogs/unreleased/6070-sunjayBhatia-small.md b/changelogs/unreleased/6070-sunjayBhatia-small.md
deleted file mode 100644
index 5ee1d76652f..00000000000
--- a/changelogs/unreleased/6070-sunjayBhatia-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Updates to Go 1.21.6. See the [Go release notes](https://go.dev/doc/devel/release#go1.21.minor) for more information.
diff --git a/changelogs/unreleased/6073-lubronzhan-small.md b/changelogs/unreleased/6073-lubronzhan-small.md
deleted file mode 100644
index f9e86422557..00000000000
--- a/changelogs/unreleased/6073-lubronzhan-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Allow gatewayProvisioner to create contour that only watch limited namespaces of resources
diff --git a/changelogs/unreleased/6077-abbas-gheydi-small.md b/changelogs/unreleased/6077-abbas-gheydi-small.md
deleted file mode 100644
index bfa20f6d79f..00000000000
--- a/changelogs/unreleased/6077-abbas-gheydi-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Access Log: Contour excludes empty fields in Envoy JSON based access logs by default.
diff --git a/changelogs/unreleased/6079-tsaarni-minor.md b/changelogs/unreleased/6079-tsaarni-minor.md
deleted file mode 100644
index a3e372871f8..00000000000
--- a/changelogs/unreleased/6079-tsaarni-minor.md
+++ /dev/null
@@ -1,4 +0,0 @@
-## Upstream TLS validation and client certificate for TCPProxy
-
-TCPProxy now supports validating server certificate and using client certificate for upstream TLS connections.
-Set `httpproxy.spec.tcpproxy.services.validation.caSecret` and `subjectName` to enable optional validation and `tls.envoy-client-certificate` configuration file field or `ContourConfiguration.spec.envoy.clientCertificate` to set the optional client certificate.
diff --git a/changelogs/unreleased/6099-sunjayBhatia-minor.md b/changelogs/unreleased/6099-sunjayBhatia-minor.md
deleted file mode 100644
index a6c5fc0a3b6..00000000000
--- a/changelogs/unreleased/6099-sunjayBhatia-minor.md
+++ /dev/null
@@ -1,5 +0,0 @@
-## Remove Contour container readiness probe initial delay
-
-The Contour Deployment Contour server container previously had its readiness probe `initialDelaySeconds` field set to 15.
-This has been removed from the example YAML manifests and Gateway Provisioner generated Contour Deployment since as of [PR #5672](https://github.com/projectcontour/contour/pull/5672) Contour's xDS server will not start or serve any configuration (and the readiness probe will not succeed) until the existing state of the cluster is synced.
-In clusters with few resources this will improve the Contour Deployment's update/rollout time as initial startup time should be low.
diff --git a/changelogs/unreleased/6119-flawedmatrix-major.md b/changelogs/unreleased/6119-flawedmatrix-major.md
deleted file mode 100644
index 600f9b8f2a2..00000000000
--- a/changelogs/unreleased/6119-flawedmatrix-major.md
+++ /dev/null
@@ -1,9 +0,0 @@
-## Support for Gateway API BackendTLSPolicy
-
-The BackendTLSPolicy CRD can now be used with HTTPRoute to configure a Contour gateway to connect to a backend Service with TLS. This will give users the ability to use Gateway API to configure their routes to securely connect to backends that use TLS with Contour.
-
-The BackendTLSPolicy spec requires you to specify a `targetRef`, which can currently only be a Kubernetes Service within the same namespace as the BackendTLSPolicy. The targetRef is what Service should be watched to apply the BackendTLSPolicy to. A `SectionName` can also be configured to the port name of a Service to reference a specific section of the Service.
-
-The spec also requires you to specify `caCertRefs`, which can either be a ConfigMap or Secret with a `ca.crt` key in the data map containing a PEM-encoded TLS certificate. The CA certificates referenced will be configured to be used by the gateway to perform TLS to the backend Service. You will also need to specify a `Hostname`, which will be used to configure the SNI the gateway will use for the connection.
-
-See Gateway API's [GEP-1897](https://gateway-api.sigs.k8s.io/geps/gep-1897) for the proposal for BackendTLSPolicy.
diff --git a/changelogs/unreleased/6124-skriss-small.md b/changelogs/unreleased/6124-skriss-small.md
deleted file mode 100644
index c070f9cc4d4..00000000000
--- a/changelogs/unreleased/6124-skriss-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Updates HTTP filter names to match between the HTTP connection manager and per-filter config on virtual hosts/routes, and to use canonical names.
\ No newline at end of file
diff --git a/changelogs/unreleased/6143-skriss-docs.md b/changelogs/unreleased/6143-skriss-docs.md
deleted file mode 100644
index 2c702c3f738..00000000000
--- a/changelogs/unreleased/6143-skriss-docs.md
+++ /dev/null
@@ -1 +0,0 @@
-Document that Gateway names should be 63 characters or shorter to avoid issues with generating dependent resources when using the Gateway provisioner.
\ No newline at end of file
diff --git a/changelogs/unreleased/6144-skriss-deprecation.md b/changelogs/unreleased/6144-skriss-deprecation.md
deleted file mode 100644
index d10319fdf6e..00000000000
--- a/changelogs/unreleased/6144-skriss-deprecation.md
+++ /dev/null
@@ -1,5 +0,0 @@
-## Configuring Contour with a GatewayClass controller name is deprecated
-
-Contour should no longer be configured with a GatewayClass controller name (`gateway.controllerName` in the config file or ContourConfiguration CRD).
-Instead, either use a specific Gateway reference (`gateway.gatewayRef`), or use the Gateway provisioner.
-`gateway.controllerName` will be removed in a future release.
\ No newline at end of file
diff --git a/changelogs/unreleased/6147-sunjayBhatia-small.md b/changelogs/unreleased/6147-sunjayBhatia-small.md
deleted file mode 100644
index 3ab0d2cf58a..00000000000
--- a/changelogs/unreleased/6147-sunjayBhatia-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Gateway API provisioner now checks `gateway.networking.k8s.io/bundle-version` annotation on Gateway CRDs and sets SupportedVersion status condition on GatewayClass if annotation value matches supported Gateway API version. Best-effort support is provided if version does not match.
diff --git a/changelogs/unreleased/6148-lubronzhan-minor.md b/changelogs/unreleased/6148-lubronzhan-minor.md
deleted file mode 100644
index 3124fff004c..00000000000
--- a/changelogs/unreleased/6148-lubronzhan-minor.md
+++ /dev/null
@@ -1,3 +0,0 @@
-## Add anti-affinity rule for envoy deployed by provisioner
-
-The envoy deployment created by the gateway provisioner now includes a default anti-affinity rule. The anti-affinity rule in the [example envoy deployment manifest](https://github.com/projectcontour/contour/blob/main/examples/deployment/03-envoy-deployment.yaml) is also updated to `preferredDuringSchedulingIgnoredDuringExecution` to be consistent with the contour deployment and the gateway provisioner anti-affinity rule.
diff --git a/changelogs/unreleased/6151-christianang-small.md b/changelogs/unreleased/6151-christianang-small.md
deleted file mode 100644
index 8a7c85b38bf..00000000000
--- a/changelogs/unreleased/6151-christianang-small.md
+++ /dev/null
@@ -1 +0,0 @@
-For Gateway API, add "Accepted" condition to BackendTLSPolicy. If the condition is true the BackendTLSPolicy was accepted by the Gateway and if false a reason will be stated on the policy as to why it wasn't accepted.
diff --git a/changelogs/unreleased/6152-lubronzhan-minor.md b/changelogs/unreleased/6152-lubronzhan-minor.md
deleted file mode 100644
index 116189f08cc..00000000000
--- a/changelogs/unreleased/6152-lubronzhan-minor.md
+++ /dev/null
@@ -1,7 +0,0 @@
-## Add DisabledFeatures to ContourDeployment for gateway provisioner
-
-A new flag DisabledFeatures is added to ContourDeployment so that user can configure contour which is deployed by the provisioner to skip reconciling CRDs which are specified inside the flag.
-
-Accepted values are `grpcroutes|tlsroutes|extensionservices|backendtlspolicies`.
-
-
diff --git a/changelogs/unreleased/6161-sunjayBhatia-docs.md b/changelogs/unreleased/6161-sunjayBhatia-docs.md
deleted file mode 100644
index f2ce37e8298..00000000000
--- a/changelogs/unreleased/6161-sunjayBhatia-docs.md
+++ /dev/null
@@ -1 +0,0 @@
-Add troubleshooting guide for general app traffic errors.
diff --git a/changelogs/unreleased/6164-sunjayBhatia-small.md b/changelogs/unreleased/6164-sunjayBhatia-small.md
deleted file mode 100644
index 3da383b22f4..00000000000
--- a/changelogs/unreleased/6164-sunjayBhatia-small.md
+++ /dev/null
@@ -1 +0,0 @@
-Updates Envoy to v1.29.1. See the release notes [here](https://www.envoyproxy.io/docs/envoy/v1.29.1/version_history/v1.29/v1.29.1).
diff --git a/netlify.toml b/netlify.toml
index d653471a366..331aadeb3f6 100644
--- a/netlify.toml
+++ b/netlify.toml
@@ -42,7 +42,7 @@
# kubectl apply https://projectcontour.io/quickstart/contour.yaml
[[redirects]]
from = "/quickstart/contour.yaml"
- to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.27/examples/render/contour.yaml"
+ to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.28/examples/render/contour.yaml"
status = 302
# Redirect versioned quickstarts so that they can easily be referenced by
@@ -59,7 +59,7 @@
# kubectl apply https://projectcontour.io/quickstart/contour-gateway.yaml
[[redirects]]
from = "/quickstart/contour-gateway.yaml"
- to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.27/examples/render/contour-gateway.yaml"
+ to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.28/examples/render/contour-gateway.yaml"
status = 302
# Redirect versioned quickstarts so that they can easily be referenced by
@@ -76,7 +76,7 @@
# kubectl apply https://projectcontour.io/quickstart/contour-gateway-provisioner.yaml
[[redirects]]
from = "/quickstart/contour-gateway-provisioner.yaml"
- to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.27/examples/render/contour-gateway-provisioner.yaml"
+ to = "https://raw.githubusercontent.com/projectcontour/contour/release-1.28/examples/render/contour-gateway-provisioner.yaml"
status = 302
# Redirect versioned quickstarts so that they can easily be referenced by
diff --git a/site/config.yaml b/site/config.yaml
index 6ceb7044f00..53a7502e048 100644
--- a/site/config.yaml
+++ b/site/config.yaml
@@ -29,7 +29,7 @@ params:
github_url: "https://github.com/projectcontour/contour"
github_raw_url: "https://raw.githubusercontent.com/projectcontour/contour"
slack_url: "https://kubernetes.slack.com/messages/contour"
- latest_version: "1.27"
+ latest_version: "1.28"
use_advanced_docs: true
docs_right_sidebar: true
docs_search: true
@@ -39,6 +39,7 @@ params:
docs_versioning: true
docs_versions:
- main
+ - "1.28"
- "1.27"
- "1.26"
- "1.25"
diff --git a/site/content/docs/1.28/_index.md b/site/content/docs/1.28/_index.md
new file mode 100644
index 00000000000..c27017b94e1
--- /dev/null
+++ b/site/content/docs/1.28/_index.md
@@ -0,0 +1,48 @@
+---
+cascade:
+ layout: docs
+ version: "1.28"
+ branch: release-1.28
+---
+
+## Overview
+Contour is an Ingress controller for Kubernetes that works by deploying the [Envoy proxy][1] as a reverse proxy and load balancer.
+Contour supports dynamic configuration updates out of the box while maintaining a lightweight profile.
+
+## Philosophy
+- Follow an opinionated approach which allows us to better serve most users
+- Design Contour to serve both the cluster administrator and the application developer
+- Use our experience with ingress to define reasonable defaults for both cluster administrators and application developers.
+- Meet users where they are by understanding and adapting Contour to their use cases
+
+See the full [Contour Philosophy][8] page.
+
+## Why Contour?
+Contour bridges other solution gaps in several ways:
+- Dynamically update the ingress configuration with minimal dropped connections
+- Safely support multiple types of ingress config in multi-team Kubernetes clusters
+ - [Ingress/v1][10]
+ - [HTTPProxy (Contour custom resource)][2]
+ - [Gateway API][9]
+- Cleanly integrate with the Kubernetes object model
+
+## Prerequisites
+Contour is tested with Kubernetes clusters running version [1.21 and later][4].
+
+## Get started
+Getting started with Contour is as simple as one command.
+See the [Getting Started][3] document.
+
+## Troubleshooting
+If you encounter issues review the [troubleshooting][5] page, [file an issue][6], or talk to us on the [#contour channel][7] on Kubernetes slack.
+
+[1]: https://www.envoyproxy.io/
+[2]: config/fundamentals.md
+[3]: /getting-started
+[4]: /resources/compatibility-matrix.md
+[5]: /docs/main/troubleshooting
+[6]: https://github.com/projectcontour/contour/issues
+[7]: https://kubernetes.slack.com/messages/contour
+[8]: /resources/philosophy
+[9]: guides/gateway-api
+[10]: /docs/{{< param version >}}/config/ingress
diff --git a/site/content/docs/1.28/architecture.md b/site/content/docs/1.28/architecture.md
new file mode 100644
index 00000000000..b29cb409d39
--- /dev/null
+++ b/site/content/docs/1.28/architecture.md
@@ -0,0 +1,74 @@
+# Contour Architecture
+
+The Contour Ingress controller is a collaboration between:
+
+* Envoy, which provides the high performance reverse proxy.
+* Contour, which acts as a management server for Envoy and provides it with configuration.
+
+These containers are deployed separately, Contour as a Deployment and Envoy as a Kubernetes Daemonset or Deployment, although other configurations are possible.
+
+In the Envoy Pods, Contour runs as an initcontainer in `bootstrap` mode and writes an Envoy bootstrap configuration to a temporary volume.
+This volume is passed to the Envoy container and directs Envoy to treat Contour as its [management server][1].
+
+After initialization is complete, the Envoy container starts, retrieves the bootstrap configuration written by Contour's `bootstrap` mode, and establishes a GRPC session with Contour to receive configuration.
+
+Envoy will gracefully retry if the management server is unavailable, which removes any container startup ordering issues.
+
+Contour is a client of the Kubernetes API.
+Contour watches Ingress, HTTPProxy, Gateway API, Secret, Service, and Endpoint objects, and acts as the management server for its Envoy sibling by translating its cache of objects into the relevant JSON stanzas: Service objects for CDS, Ingress for RDS, Endpoint objects for EDS, and so on).
+
+The transfer of information from Kubernetes to Contour is by watching the Kubernetes API utilizing [controller-runtime][4] primitives.
+
+Kubernetes readiness probes are configured to check whether Envoy is ready to accept connections.
+The Envoy readiness probe sends GET requests to `/ready` in Envoy's administration endpoint.
+
+For Contour, a liveness probe checks the `/healthz` running on the Pod's metrics port.
+Readiness probe is a check that Contour can access the Kubernetes API.
+
+## Architectural Overview
+Below are a couple of high level architectural diagrams of how Contour works inside a Kubernetes cluster as well as showing the data path of a request to a backend pod.
+
+A request to `projectcontour.io/blog` gets routed via a load balancer to an instance of an Envoy proxy which then sends the request to a pod.
+
+![architectural overview][2]
+
+Following is a diagram of how Contour and Envoy are deployed in a Kubernetes cluster.
+
+### Kubernetes API Server
+
+The following API objects are watched:
+- Services
+- Endpoints
+- Secrets
+- Ingress
+- HTTPProxy
+- Gateway API (Optional)
+
+### Contour Deployment
+
+Contour is deployed in the cluster using a Kubernetes Deployment.
+It has built-in leader election which is responsible for updating httproxy/ingress/gateway api resources via Kube API server.
+All instances are able to serve xDS configuration to any Envoy instance, but only the leader can write status back to the API server.
+
+The data being served from contour instances are eventually consistent in an HA based deployment.
+However HA mode is operationally scalable when you have high request rate from envoy to contour as requests are loadbalanced among contour instances.
+This also helps availability zone /data center degradation events as your service continue to function.
+
+### Envoy Deployment
+
+Envoy can be deployed in two different models, as a Kubernetes Daemonset or as a Kubernetes Deployment.
+
+Daemonset is the standard deployment model where a single instance of Envoy is deployed per Kubernetes Node.
+This allows for simple Envoy pod distribution across the cluster as well as being able to expose Envoy using `hostPorts` to improve network performance.
+One potential downside of this deployment model is when a node is removed from the cluster (e.g. on a cluster scale down, etc) then the configured `preStop` hooks are not available so connections can be dropped.
+This is a limitation that applies to any Daemonset in Kubernetes.
+
+An alternative Envoy deployment model is utilizing a Kubernetes Deployment with a configured `podAntiAffinity` which attempts to mirror the Daemonset deployment model.
+A benefit of this model compared to the Daemonset version is when a node is removed from the cluster, the proper shutdown events are available so connections can be cleanly drained from Envoy before terminating.
+
+![architectural overview 2][3]
+
+[1]: https://www.envoyproxy.io/docs/envoy/v1.13.0/api-docs/xds_protocol
+[2]: ../img/archoverview.png
+[3]: ../img/contour_deployment_in_k8s.png
+[4]: https://github.com/kubernetes-sigs/controller-runtime
diff --git a/site/content/docs/1.28/config/access-logging.md b/site/content/docs/1.28/config/access-logging.md
new file mode 100644
index 00000000000..0c5b6e1583c
--- /dev/null
+++ b/site/content/docs/1.28/config/access-logging.md
@@ -0,0 +1,148 @@
+# Access Logging
+
+## Overview
+
+Contour allows you to control Envoy's access logging.
+By default, HTTP and HTTPS access logs are written to `/dev/stdout` by the Envoy containers and look like following:
+
+```
+[2021-04-14T16:36:00.361Z] "GET /foo HTTP/1.1" 200 - 0 463 6 3 "-" "HTTPie/1.0.3" "837aa8dc-344f-4faa-b7d5-c9cce1028519" "localhost:8080" "127.0.0.1:8081"
+```
+
+The detailed description of each field can be found in [Envoy access logging documentation][7].
+
+
+## Customizing Access Log Destination
+
+You can change the destination file where the access log is written by using Contour [command line parameters][1] `--envoy-http-access-log` and `--envoy-https-access-log`.
+
+## Customizing Access Log Format
+
+The access log can take two different formats, both can be customized
+
+* Text based access logs, like shown in the example above.
+* Structured JSON logging.
+
+### Text Based Access Logging
+
+Ensure that you have selected `envoy` as the access log format.
+Note that this is the default format if the parameters are not given.
+
+- Add `--accesslog-format=envoy` to your Contour startup line, or
+- Add `accesslog-format: envoy` to your configuration file.
+
+Customize the access log format by defining `accesslog-format-string` in your configuration file.
+
+```yaml
+accesslog-format-string: "[%START_TIME%] \"%REQ(:METHOD)% %REQ(X-ENVOY-ORIGINAL-PATH?:PATH)% %PROTOCOL%\" %RESPONSE_CODE% %RESPONSE_FLAGS% %BYTES_RECEIVED% %BYTES_SENT% %DURATION% %RESP(X-ENVOY-UPSTREAM-SERVICE-TIME)% \"%REQ(X-FORWARDED-FOR)%\" \"%REQ(USER-AGENT)%\" \"%REQ(X-REQUEST-ID)%\" \"%REQ(:AUTHORITY)%\" \"%UPSTREAM_HOST%\"\n"
+```
+After restarting Contour and successful validation of the configuration, the new format will take effect in a short while.
+
+Refer to [Envoy access logging documentation][7] for the description of the command operators, and note that the format string needs to end in a linefeed `\n`.
+
+### Structured JSON Logging
+
+Contour allows you to choose from a set of JSON fields that will be expanded into Envoy templates and sent to Envoy.
+There is a default set of fields if you enable JSON logging, and you may customize which fields you log.
+
+The list of available fields are discoverable in the following objects:
+- [jsonFields][2] are fields that have built in mappings to commonly used envoy operators.
+- [envoySimpleOperators][3] are the names of simple envoy operators that don't require arguments, they are case-insensitive when configured.
+- [envoyComplexOperators][4] are the names of complex envoy operators that require arguments.
+
+The default list of fields is available at [DefaultAccessLogJSONFields][5].
+
+#### Enabling the Feature
+
+To enable the feature you have two options:
+
+- Add `--accesslog-format=json` to your Contour startup line.
+- Add `accesslog-format: json` to your configuration file.
+
+Without any further customization, the [default fields][5] will be used.
+
+#### Customizing Logged Fields
+
+To customize the logged fields, add a `json-fields` list of strings to your configuration file.
+If the `json-fields` key is not specified, the [default fields][5] will be configured.
+
+To use a value from [jsonFields][2] or [envoySimpleOperators][3], simply include the name of the value in the list of strings.
+The jsonFields are case-sensitive, but envoySimpleOperators are not.
+
+To use [envoyComplexOperators][4] or to use alternative field names, specify strings as key/value pairs like `"fieldName=%OPERATOR(...)%"`.
+
+Unknown field names in non key/value fields will result in validation errors, as will unknown Envoy operators in key/value fields.
+Note that the `DYNAMIC_METADATA` and `FILTER_STATE` Envoy logging operators are not supported at this time due to the complexity of their validation.
+
+See the [example config file][6] to see this used in context.
+
+#### Omitting Logs with Empty Values
+
+Contour automatically omits empty fields in Envoy JSON access logs, enhancing clarity and delivering more concise and relevant log outputs by default.
+
+#### Sample Configuration File
+
+Here is a sample config:
+
+```yaml
+accesslog-format: json
+json-fields:
+ - "@timestamp"
+ - "authority"
+ - "bytes_received"
+ - "bytes_sent"
+ - "customer_id=%REQ(X-CUSTOMER-ID)%"
+ - "downstream_local_address"
+ - "downstream_remote_address"
+ - "duration"
+ - "method"
+ - "path"
+ - "protocol"
+ - "request_id"
+ - "requested_server_name"
+ - "response_code"
+ - "response_flags"
+ - "uber_trace_id"
+ - "upstream_cluster"
+ - "upstream_host"
+ - "upstream_local_address"
+ - "upstream_service_time"
+ - "user_agent"
+ - "x_forwarded_for"
+```
+
+### Logging the route source
+
+Contour can log the kind, namespace and name of the Kubernetes resource that generated the route for a given access log entry.
+
+For text-based access logging, the following command operators can be used:
+- `%METADATA(ROUTE:envoy.access_loggers.file:io.projectcontour.kind)%`
+- `%METADATA(ROUTE:envoy.access_loggers.file:io.projectcontour.namespace)%`
+- `%METADATA(ROUTE:envoy.access_loggers.file:io.projectcontour.name)%`
+
+For JSON access logging, the following fields can be added (these are Contour-specific aliases to the above command operators):
+- `contour_config_kind`
+- `contour_config_namespace`
+- `contour_config_name`
+
+## Using Access Log Formatter Extensions
+
+Envoy allows implementing custom access log command operators as extensions.
+Following extensions are supported by Contour:
+
+| Command operator | Description |
+|------------------|-------------|
+| [REQ_WITHOUT_QUERY][8] | Works the same way as REQ except that it will remove the query string. It is used to avoid logging any sensitive information into the access log. |
+| [METADATA][9] | Prints all types of metadata. |
+
+
+
+[1]: ../configuration#serve-flags
+[2]: https://github.com/search?q=%22var+jsonFields%22+repo%3Aprojectcontour%2Fcontour+path%3Aapis&type=code
+[3]: https://github.com/search?q=%22var+envoySimpleOperators%22+repo%3Aprojectcontour%2Fcontour+path%3Aapis&type=code
+[4]: https://github.com/search?q=%22var+envoyComplexOperators%22+repo%3Aprojectcontour%2Fcontour+path%3Aapis&type=code
+[5]: https://github.com/search?q=%22var+DefaultAccessLogJSONFields%22+repo%3Aprojectcontour%2Fcontour+path%3Aapis&type=code
+[6]: {{< param github_url >}}/tree/{{< param latest_version >}}/examples/contour/01-contour-config.yaml
+[7]: https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage
+[8]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/formatter/req_without_query/v3/req_without_query.proto
+[9]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/formatter/metadata/v3/metadata.proto
\ No newline at end of file
diff --git a/site/content/docs/1.28/config/annotations.md b/site/content/docs/1.28/config/annotations.md
new file mode 100644
index 00000000000..4d805f8f0f1
--- /dev/null
+++ b/site/content/docs/1.28/config/annotations.md
@@ -0,0 +1,98 @@
+# Annotations Reference
+
+
+
+Annotations are used in Ingress Controllers to configure features that are not covered by the Kubernetes Ingress API.
+
+Some of the features that have been historically configured via annotations are supported as first-class features in Contour's [HTTPProxy API][15], which provides a more robust configuration interface over annotations.
+
+However, Contour still supports a number of annotations on the Ingress resources.
+
+## Standard Kubernetes Ingress annotations
+
+The following Kubernetes annotations are supported on `Ingress` objects:
+
+### Ingress Class
+
+The Ingress class annotation can be used to specify which Ingress controller should serve a particular Ingress object.
+This annotation may be specified as the standard `kubernetes.io/ingress.class` or a Contour-specific `projectcontour.io/ingress.class`.
+In both cases, they will behave as follows, by default:
+
+* If not set, then all Ingress controllers serve the Ingress.
+* If specified as `kubernetes.io/ingress.class: contour`, then Contour serves the Ingress.
+* If any other value, Contour ignores the Ingress definition.
+
+You can override the default class `contour` by providing the `--ingress-class-name` flag to Contour.
+This can be useful while you are migrating from another controller, or if you need multiple instances of Contour.
+If you do this, the behavior is as follows:
+* If the annotation is not set, Contour will ignore the Ingress.
+* If the annotation is set to any value other than the one passed to the `--ingress-class-name` flag, Contour will ignore the Ingress.
+* If the annotation matches the value that you passed to `--ingress-class-name` flag, Contour will serve the Ingress.
+
+This same logic applies for these annotations on HTTPProxy objects.
+
+_Note: Both `Ingress` and `HTTPProxy` now have an `IngressClassName` field in their spec. Going forward this is the preferred way to specify an ingress class, rather than using an annotation. If both the annotation and the spec field are specified on an object, the annotation takes preference for backwards compatibility._
+
+_Note: The `--ingress-class-name` value can be a comma-separated list of class names to match against. Contour will serve the Ingress or HTTPProxy if the annotation or IngressClassName matches any of the specified class name values.
+
+### Other annotations
+
+ - `ingress.kubernetes.io/force-ssl-redirect`: Requires TLS/SSL for the Ingress to Envoy by setting the [Envoy virtual host option require_tls][16].
+ - `kubernetes.io/ingress.allow-http`: Instructs Contour to not create an Envoy HTTP route for the virtual host. The Ingress exists only for HTTPS requests. Specify `"false"` for Envoy to mark the endpoint as HTTPS only. All other values are ignored.
+
+The `ingress.kubernetes.io/force-ssl-redirect` annotation takes precedence over `kubernetes.io/ingress.allow-http`. If they are set to `"true"` and `"false"` respectively, Contour *will* create an Envoy HTTP route for the Virtual host, and set the `require_tls` virtual host option.
+
+## Contour specific Ingress annotations
+
+ - `projectcontour.io/ingress.class`: The Ingress class that should interpret and serve the Ingress. See the [main Ingress class annotation section](#ingress-class) for more details.
+ - `projectcontour.io/num-retries`: [The maximum number of retries][1] Envoy should make before abandoning and returning an error to the client. Applies only if `projectcontour.io/retry-on` is specified. Set to -1 to disable retries.
+ - `projectcontour.io/per-try-timeout`: [The timeout per retry attempt][2], if there should be one. Applies only if `projectcontour.io/retry-on` is specified.
+ - `projectcontour.io/response-timeout`: [The Envoy HTTP route timeout][3], specified as a [golang duration][4]. By default, Envoy has a 15 second timeout for a backend service to respond. Set this to `infinity` to specify that Envoy should never timeout the connection to the backend. Note that the value `0s` / zero has special semantics for Envoy.
+ - `projectcontour.io/retry-on`: [The conditions for Envoy to retry a request][5]. See also [possible values and their meanings for `retry-on`][6].
+ - `projectcontour.io/tls-minimum-protocol-version`: [The minimum TLS protocol version][7] the TLS listener should support. Valid options are `1.3`, `1.2` (default).
+ - `projectcontour.io/tls-maximum-protocol-version`: [The maximum TLS protocol version][7] the TLS listener should support. Valid options are `1.2`, `1.3` (default).
+ - `projectcontour.io/websocket-routes`: [The routes supporting websocket protocol][8], the annotation value contains a list of route paths separated by a comma that must match with the ones defined in the `Ingress` definition. Defaults to Envoy's default behavior which is `use_websocket` to `false`.
+ - `projectcontour.io/tls-cert-namespace`: The namespace where all TLS secrets of this Ingress are searched. This is necessary to use [TLS Certificate Delegation][18] with Ingress v1 because the slash notation (ex: different-ns/app-cert) used by HTTPProxy and Ingress v1beta1 is not accepted. See [this issue][19] for details.
+
+## Contour specific Service annotations
+
+A [Kubernetes Service][9] maps to an [Envoy Cluster][10]. Envoy clusters have many settings to control specific behaviors. These annotations allow access to some of those settings.
+
+- `projectcontour.io/max-connections`: [The maximum number of connections][11] that a single Envoy instance allows to the Kubernetes Service; defaults to 1024.
+- `projectcontour.io/max-pending-requests`: [The maximum number of pending requests][13] that a single Envoy instance allows to the Kubernetes Service; defaults to 1024.
+- `projectcontour.io/max-requests`: [The maximum parallel requests][13] a single Envoy instance allows to the Kubernetes Service; defaults to 1024
+- `projectcontour.io/max-retries`: [The maximum number of parallel retries][14] a single Envoy instance allows to the Kubernetes Service; defaults to 3. This is independent of the per-Kubernetes Ingress number of retries (`projectcontour.io/num-retries`) and retry-on (`projectcontour.io/retry-on`), which control whether retries are attempted and how many times a single request can retry.
+- `projectcontour.io/per-host-max-connections`: [The maximum number of connections][20] that a single Envoy instance allows to an individual Kubernetes Service endpoint; no default (unlimited).
+- `projectcontour.io/upstream-protocol.{protocol}` : The protocol used to proxy requests to the upstream service.
+ The annotation value contains a comma-separated list of port names and/or numbers that must match with the ones defined in the `Service` definition.
+ This value can also be specified in the `spec.routes.services[].protocol` field on the HTTPProxy object, where it takes precedence over the Service annotation.
+ Supported protocol names are: `h2`, `h2c`, and `tls`:
+ - The `tls` protocol allows for requests which terminate at Envoy to proxy via TLS to the upstream.
+ This protocol should be used for HTTP/1.1 services over TLS.
+ _Note that validating the upstream TLS certificate requires additionally setting the [validation][17] field._
+ - The `h2` protocol proxies requests to the upstream using HTTP/2 over TLS.
+ - The `h2c` protocol proxies requests to the upstream using cleartext HTTP/2.
+
+## Contour specific HTTPProxy annotations
+- `projectcontour.io/ingress.class`: The Ingress class that should interpret and serve the HTTPProxy. See the [main Ingress class annotation section](#ingress-class) for more details.
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/router_filter#config-http-filters-router-x-envoy-max-retries
+[2]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-retrypolicy-retry-on
+[3]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-routeaction-timeout
+[4]: https://golang.org/pkg/time/#ParseDuration
+[5]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-retrypolicy-retry-on
+[6]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/router_filter#config-http-filters-router-x-envoy-retry-on
+[7]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/transport_sockets/tls/v3/common.proto.html#extensions-transport-sockets-tls-v3-tlsparameters
+[8]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-routeaction-upgrade-configs
+[9]: https://kubernetes.io/docs/concepts/services-networking/service/
+[10]: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/intro/terminology
+[11]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-thresholds-max-connections
+[12]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-thresholds-max-pending-requests
+[13]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-thresholds-max-requests
+[14]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-thresholds-max-retries
+[15]: fundamentals.md
+[16]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-virtualhost-require-tls
+[17]: api/#projectcontour.io/v1.UpstreamValidation
+[18]: ../config/tls-delegation/
+[19]: https://github.com/projectcontour/contour/issues/3544
+[20]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/circuit_breaker.proto#envoy-v3-api-field-config-cluster-v3-circuitbreakers-per-host-thresholds
\ No newline at end of file
diff --git a/site/content/docs/1.28/config/api-reference.html b/site/content/docs/1.28/config/api-reference.html
new file mode 100644
index 00000000000..522e89d16a8
--- /dev/null
+++ b/site/content/docs/1.28/config/api-reference.html
@@ -0,0 +1,9131 @@
+Packages:
+
+projectcontour.io/v1
+
+
Package v1 holds the specification for the projectcontour.io Custom Resource Definitions (CRDs).
+In building this CRD, we’ve inadvertently overloaded the word “Condition”, so we’ve tried to make
+this spec clear as to which types of condition are which.
+MatchConditions
are used by Routes
and Includes
to specify rules to match requests against for either
+routing or inclusion.
+DetailedConditions
are used in the Status
of these objects to hold information about the relevant
+state of the object and the world around it.
+SubConditions
are used underneath DetailedConditions
to give more detail to errors or warnings.
+
+Resource Types:
+
+HTTPProxy
+
+
+
HTTPProxy is an Ingress CRD specification.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+apiVersion
+string |
+
+
+projectcontour.io/v1
+
+ |
+
+
+
+kind
+string
+ |
+HTTPProxy |
+
+
+
+metadata
+
+
+
+Kubernetes meta/v1.ObjectMeta
+
+
+ |
+
+Refer to the Kubernetes API documentation for the fields of the
+metadata field.
+ |
+
+
+
+spec
+
+
+
+HTTPProxySpec
+
+
+ |
+
+
+
+
+
+
+virtualhost
+
+
+
+VirtualHost
+
+
+ |
+
+(Optional)
+ Virtualhost appears at most once. If it is present, the object is considered
+to be a “root” HTTPProxy.
+ |
+
+
+
+routes
+
+
+
+[]Route
+
+
+ |
+
+(Optional)
+ Routes are the ingress routes. If TCPProxy is present, Routes is ignored.
+ |
+
+
+
+tcpproxy
+
+
+
+TCPProxy
+
+
+ |
+
+(Optional)
+ TCPProxy holds TCP proxy information.
+ |
+
+
+
+includes
+
+
+
+[]Include
+
+
+ |
+
+(Optional)
+ Includes allow for specific routing configuration to be included from another HTTPProxy,
+possibly in another namespace.
+ |
+
+
+
+ingressClassName
+
+
+string
+
+ |
+
+(Optional)
+ IngressClassName optionally specifies the ingress class to use for this
+HTTPProxy. This replaces the deprecated kubernetes.io/ingress.class
+annotation. For backwards compatibility, when that annotation is set, it
+is given precedence over this field.
+ |
+
+
+ |
+
+
+
+status
+
+
+
+HTTPProxyStatus
+
+
+ |
+
+(Optional)
+ Status is a container for computed information about the HTTPProxy.
+ |
+
+
+
+TLSCertificateDelegation
+
+
+
TLSCertificateDelegation is an TLS Certificate Delegation CRD specification.
+See design/tls-certificate-delegation.md for details.
+
+
+AuthorizationPolicy
+
+
+(Appears on:
+AuthorizationServer,
+Route)
+
+
+
AuthorizationPolicy modifies how client requests are authenticated.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+disabled
+
+
+bool
+
+ |
+
+(Optional)
+ When true, this field disables client request authentication
+for the scope of the policy.
+ |
+
+
+
+context
+
+
+map[string]string
+
+ |
+
+(Optional)
+ Context is a set of key/value pairs that are sent to the
+authentication server in the check request. If a context
+is provided at an enclosing scope, the entries are merged
+such that the inner scope overrides matching keys from the
+outer scope.
+ |
+
+
+
+AuthorizationServer
+
+
+(Appears on:
+VirtualHost,
+ContourConfigurationSpec)
+
+
+
AuthorizationServer configures an external server to authenticate
+client requests. The external server must implement the v3 Envoy
+external authorization GRPC protocol (https://www.envoyproxy.io/docs/envoy/latest/api-v3/service/auth/v3/external_auth.proto).
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+extensionRef
+
+
+
+ExtensionServiceReference
+
+
+ |
+
+(Optional)
+ ExtensionServiceRef specifies the extension resource that will authorize client requests.
+ |
+
+
+
+authPolicy
+
+
+
+AuthorizationPolicy
+
+
+ |
+
+(Optional)
+ AuthPolicy sets a default authorization policy for client requests.
+This policy will be used unless overridden by individual routes.
+ |
+
+
+
+responseTimeout
+
+
+string
+
+ |
+
+(Optional)
+ ResponseTimeout configures maximum time to wait for a check response from the authorization server.
+Timeout durations are expressed in the Go Duration format.
+Valid time units are “ns”, “us” (or “µs”), “ms”, “s”, “m”, “h”.
+The string “infinity” is also a valid input and specifies no timeout.
+ |
+
+
+
+failOpen
+
+
+bool
+
+ |
+
+(Optional)
+ If FailOpen is true, the client request is forwarded to the upstream service
+even if the authorization server fails to respond. This field should not be
+set in most cases. It is intended for use only while migrating applications
+from internal authorization to Contour external authorization.
+ |
+
+
+
+withRequestBody
+
+
+
+AuthorizationServerBufferSettings
+
+
+ |
+
+(Optional)
+ WithRequestBody specifies configuration for sending the client request’s body to authorization server.
+ |
+
+
+
+AuthorizationServerBufferSettings
+
+
+(Appears on:
+AuthorizationServer)
+
+
+
AuthorizationServerBufferSettings enables ExtAuthz filter to buffer client request data and send it as part of authorization request
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+maxRequestBytes
+
+
+uint32
+
+ |
+
+(Optional)
+ MaxRequestBytes sets the maximum size of message body ExtAuthz filter will hold in-memory.
+ |
+
+
+
+allowPartialMessage
+
+
+bool
+
+ |
+
+(Optional)
+ If AllowPartialMessage is true, then Envoy will buffer the body until MaxRequestBytes are reached.
+ |
+
+
+
+packAsBytes
+
+
+bool
+
+ |
+
+(Optional)
+ If PackAsBytes is true, the body sent to Authorization Server is in raw bytes.
+ |
+
+
+
+
+
+(Appears on:
+CORSPolicy)
+
+
+
CORSHeaderValue specifies the value of the string headers returned by a cross-domain request.
+
+CORSPolicy
+
+
+(Appears on:
+VirtualHost)
+
+
+
CORSPolicy allows setting the CORS policy
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+allowCredentials
+
+
+bool
+
+ |
+
+(Optional)
+ Specifies whether the resource allows credentials.
+ |
+
+
+
+allowOrigin
+
+
+[]string
+
+ |
+
+ AllowOrigin specifies the origins that will be allowed to do CORS requests.
+Allowed values include “*” which signifies any origin is allowed, an exact
+origin of the form “scheme://host[:port]” (where port is optional), or a valid
+regex pattern.
+Note that regex patterns are validated and a simple “glob” pattern (e.g. *.foo.com)
+will be rejected or produce unexpected matches when applied as a regex.
+ |
+
+
+
+allowMethods
+
+
+
+[]CORSHeaderValue
+
+
+ |
+
+ AllowMethods specifies the content for the access-control-allow-methods header.
+ |
+
+
+
+allowHeaders
+
+
+
+[]CORSHeaderValue
+
+
+ |
+
+(Optional)
+ AllowHeaders specifies the content for the access-control-allow-headers header.
+ |
+
+
+
+exposeHeaders
+
+
+
+[]CORSHeaderValue
+
+
+ |
+
+(Optional)
+ ExposeHeaders Specifies the content for the access-control-expose-headers header.
+ |
+
+
+
+maxAge
+
+
+string
+
+ |
+
+(Optional)
+ MaxAge indicates for how long the results of a preflight request can be cached.
+MaxAge durations are expressed in the Go Duration format.
+Valid time units are “ns”, “us” (or “µs”), “ms”, “s”, “m”, “h”.
+Only positive values are allowed while 0 disables the cache requiring a preflight OPTIONS
+check for all cross-origin requests.
+ |
+
+
+
+allowPrivateNetwork
+
+
+bool
+
+ |
+
+ AllowPrivateNetwork specifies whether to allow private network requests.
+See https://developer.chrome.com/blog/private-network-access-preflight.
+ |
+
+
+
+CertificateDelegation
+
+
+(Appears on:
+TLSCertificateDelegationSpec)
+
+
+
CertificateDelegation maps the authority to reference a secret
+in the current namespace to a set of namespaces.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+secretName
+
+
+string
+
+ |
+
+ required, the name of a secret in the current namespace.
+ |
+
+
+
+targetNamespaces
+
+
+[]string
+
+ |
+
+ required, the namespaces the authority to reference the
+secret will be delegated to.
+If TargetNamespaces is nil or empty, the CertificateDelegation’
+is ignored. If the TargetNamespace list contains the character, “*”
+the secret will be delegated to all namespaces.
+ |
+
+
+
+ClientCertificateDetails
+
+
+(Appears on:
+DownstreamValidation)
+
+
+
ClientCertificateDetails defines which parts of the client certificate will be forwarded.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+subject
+
+
+bool
+
+ |
+
+(Optional)
+ Subject of the client cert.
+ |
+
+
+
+cert
+
+
+bool
+
+ |
+
+(Optional)
+ Client cert in URL encoded PEM format.
+ |
+
+
+
+chain
+
+
+bool
+
+ |
+
+(Optional)
+ Client cert chain (including the leaf cert) in URL encoded PEM format.
+ |
+
+
+
+dns
+
+
+bool
+
+ |
+
+(Optional)
+ DNS type Subject Alternative Names of the client cert.
+ |
+
+
+
+uri
+
+
+bool
+
+ |
+
+(Optional)
+ URI type Subject Alternative Name of the client cert.
+ |
+
+
+
+CookieDomainRewrite
+
+
+(Appears on:
+CookieRewritePolicy)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+value
+
+
+string
+
+ |
+
+ Value is the value to rewrite the Domain attribute to.
+For now this is required.
+ |
+
+
+
+CookiePathRewrite
+
+
+(Appears on:
+CookieRewritePolicy)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+value
+
+
+string
+
+ |
+
+ Value is the value to rewrite the Path attribute to.
+For now this is required.
+ |
+
+
+
+CookieRewritePolicy
+
+
+(Appears on:
+Route,
+Service)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name is the name of the cookie for which attributes will be rewritten.
+ |
+
+
+
+pathRewrite
+
+
+
+CookiePathRewrite
+
+
+ |
+
+(Optional)
+ PathRewrite enables rewriting the Set-Cookie Path element.
+If not set, Path will not be rewritten.
+ |
+
+
+
+domainRewrite
+
+
+
+CookieDomainRewrite
+
+
+ |
+
+(Optional)
+ DomainRewrite enables rewriting the Set-Cookie Domain element.
+If not set, Domain will not be rewritten.
+ |
+
+
+
+secure
+
+
+bool
+
+ |
+
+(Optional)
+ Secure enables rewriting the Set-Cookie Secure element.
+If not set, Secure attribute will not be rewritten.
+ |
+
+
+
+sameSite
+
+
+string
+
+ |
+
+(Optional)
+ SameSite enables rewriting the Set-Cookie SameSite element.
+If not set, SameSite attribute will not be rewritten.
+ |
+
+
+
+DetailedCondition
+
+
+(Appears on:
+HTTPProxyStatus,
+TLSCertificateDelegationStatus,
+ContourConfigurationStatus,
+ExtensionServiceStatus)
+
+
+
DetailedCondition is an extension of the normal Kubernetes conditions, with two extra
+fields to hold sub-conditions, which provide more detailed reasons for the state (True or False)
+of the condition.
+errors
holds information about sub-conditions which are fatal to that condition and render its state False.
+warnings
holds information about sub-conditions which are not fatal to that condition and do not force the state to be False.
+Remember that Conditions have a type, a status, and a reason.
+The type is the type of the condition, the most important one in this CRD set is Valid
.
+Valid
is a positive-polarity condition: when it is status: true
there are no problems.
+In more detail, status: true
means that the object is has been ingested into Contour with no errors.
+warnings
may still be present, and will be indicated in the Reason field. There must be zero entries in the errors
+slice in this case.
+Valid
, status: false
means that the object has had one or more fatal errors during processing into Contour.
+The details of the errors will be present under the errors
field. There must be at least one error in the errors
+slice if status
is false
.
+For DetailedConditions of types other than Valid
, the Condition must be in the negative polarity.
+When they have status
true
, there is an error. There must be at least one entry in the errors
Subcondition slice.
+When they have status
false
, there are no serious errors, and there must be zero entries in the errors
slice.
+In either case, there may be entries in the warnings
slice.
+Regardless of the polarity, the reason
and message
fields must be updated with either the detail of the reason
+(if there is one and only one entry in total across both the errors
and warnings
slices), or
+MultipleReasons
if there is more than one entry.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+Condition
+
+
+
+Kubernetes meta/v1.Condition
+
+
+ |
+
+
+(Members of Condition are embedded into this type.)
+
+ |
+
+
+
+errors
+
+
+
+[]SubCondition
+
+
+ |
+
+(Optional)
+ Errors contains a slice of relevant error subconditions for this object.
+Subconditions are expected to appear when relevant (when there is a error), and disappear when not relevant.
+An empty slice here indicates no errors.
+ |
+
+
+
+warnings
+
+
+
+[]SubCondition
+
+
+ |
+
+(Optional)
+ Warnings contains a slice of relevant warning subconditions for this object.
+Subconditions are expected to appear when relevant (when there is a warning), and disappear when not relevant.
+An empty slice here indicates no warnings.
+ |
+
+
+
+DownstreamValidation
+
+
+(Appears on:
+TLS)
+
+
+
DownstreamValidation defines how to verify the client certificate.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+caSecret
+
+
+string
+
+ |
+
+(Optional)
+ Name of a Kubernetes secret that contains a CA certificate bundle.
+The secret must contain key named ca.crt.
+The client certificate must validate against the certificates in the bundle.
+If specified and SkipClientCertValidation is true, client certificates will
+be required on requests.
+The name can be optionally prefixed with namespace “namespace/name”.
+When cross-namespace reference is used, TLSCertificateDelegation resource must exist in the namespace to grant access to the secret.
+ |
+
+
+
+skipClientCertValidation
+
+
+bool
+
+ |
+
+(Optional)
+ SkipClientCertValidation disables downstream client certificate
+validation. Defaults to false. This field is intended to be used in
+conjunction with external authorization in order to enable the external
+authorization server to validate client certificates. When this field
+is set to true, client certificates are requested but not verified by
+Envoy. If CACertificate is specified, client certificates are required on
+requests, but not verified. If external authorization is in use, they are
+presented to the external authorization server.
+ |
+
+
+
+forwardClientCertificate
+
+
+
+ClientCertificateDetails
+
+
+ |
+
+(Optional)
+ ForwardClientCertificate adds the selected data from the passed client TLS certificate
+to the x-forwarded-client-cert header.
+ |
+
+
+
+crlSecret
+
+
+string
+
+ |
+
+(Optional)
+ Name of a Kubernetes opaque secret that contains a concatenated list of PEM encoded CRLs.
+The secret must contain key named crl.pem.
+This field will be used to verify that a client certificate has not been revoked.
+CRLs must be available from all CAs, unless crlOnlyVerifyLeafCert is true.
+Large CRL lists are not supported since individual secrets are limited to 1MiB in size.
+The name can be optionally prefixed with namespace “namespace/name”.
+When cross-namespace reference is used, TLSCertificateDelegation resource must exist in the namespace to grant access to the secret.
+ |
+
+
+
+crlOnlyVerifyLeafCert
+
+
+bool
+
+ |
+
+(Optional)
+ If this option is set to true, only the certificate at the end of the
+certificate chain will be subject to validation by CRL.
+ |
+
+
+
+optionalClientCertificate
+
+
+bool
+
+ |
+
+(Optional)
+ OptionalClientCertificate when set to true will request a client certificate
+but allow the connection to continue if the client does not provide one.
+If a client certificate is sent, it will be verified according to the
+other properties, which includes disabling validation if
+SkipClientCertValidation is set. Defaults to false.
+ |
+
+
+
+ExtensionServiceReference
+
+
+(Appears on:
+AuthorizationServer)
+
+
+
ExtensionServiceReference names an ExtensionService resource.
+
+
+Feature
+(string
alias)
+
+(Appears on:
+ContourSettings)
+
+
+
+GenericKeyDescriptor
+
+
+(Appears on:
+RateLimitDescriptorEntry)
+
+
+
GenericKeyDescriptor defines a descriptor entry with a static key and
+value.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+key
+
+
+string
+
+ |
+
+(Optional)
+ Key defines the key of the descriptor entry. If not set, the
+key is set to “generic_key”.
+ |
+
+
+
+value
+
+
+string
+
+ |
+
+ Value defines the value of the descriptor entry.
+ |
+
+
+
+GlobalRateLimitPolicy
+
+
+(Appears on:
+RateLimitPolicy,
+RateLimitServiceConfig)
+
+
+
GlobalRateLimitPolicy defines global rate limiting parameters.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+disabled
+
+
+bool
+
+ |
+
+(Optional)
+ Disabled configures the HTTPProxy to not use
+the default global rate limit policy defined by the Contour configuration.
+ |
+
+
+
+descriptors
+
+
+
+[]RateLimitDescriptor
+
+
+ |
+
+(Optional)
+ Descriptors defines the list of descriptors that will
+be generated and sent to the rate limit service. Each
+descriptor contains 1+ key-value pair entries.
+ |
+
+
+
+HTTPDirectResponsePolicy
+
+
+(Appears on:
+Route)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+statusCode
+
+
+int
+
+ |
+
+ StatusCode is the HTTP response status to be returned.
+ |
+
+
+
+body
+
+
+string
+
+ |
+
+(Optional)
+ Body is the content of the response body.
+If this setting is omitted, no body is included in the generated response.
+Note: Body is not recommended to set too long
+otherwise it can have significant resource usage impacts.
+ |
+
+
+
+HTTPHealthCheckPolicy
+
+
+(Appears on:
+Route)
+
+
+
HTTPHealthCheckPolicy defines health checks on the upstream service.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+path
+
+
+string
+
+ |
+
+ HTTP endpoint used to perform health checks on upstream service
+ |
+
+
+
+host
+
+
+string
+
+ |
+
+ The value of the host header in the HTTP health check request.
+If left empty (default value), the name “contour-envoy-healthcheck”
+will be used.
+ |
+
+
+
+intervalSeconds
+
+
+int64
+
+ |
+
+(Optional)
+ The interval (seconds) between health checks
+ |
+
+
+
+timeoutSeconds
+
+
+int64
+
+ |
+
+(Optional)
+ The time to wait (seconds) for a health check response
+ |
+
+
+
+unhealthyThresholdCount
+
+
+int64
+
+ |
+
+(Optional)
+ The number of unhealthy health checks required before a host is marked unhealthy
+ |
+
+
+
+healthyThresholdCount
+
+
+int64
+
+ |
+
+(Optional)
+ The number of healthy health checks required before a host is marked healthy
+ |
+
+
+
+expectedStatuses
+
+
+
+[]HTTPStatusRange
+
+
+ |
+
+(Optional)
+ The ranges of HTTP response statuses considered healthy. Follow half-open
+semantics, i.e. for each range the start is inclusive and the end is exclusive.
+Must be within the range [100,600). If not specified, only a 200 response status
+is considered healthy.
+ |
+
+
+
+HTTPInternalRedirectPolicy
+
+
+(Appears on:
+Route)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+maxInternalRedirects
+
+
+uint32
+
+ |
+
+(Optional)
+ MaxInternalRedirects An internal redirect is not handled, unless the number of previous internal
+redirects that a downstream request has encountered is lower than this value.
+ |
+
+
+
+redirectResponseCodes
+
+
+
+[]RedirectResponseCode
+
+
+ |
+
+(Optional)
+ RedirectResponseCodes If unspecified, only 302 will be treated as internal redirect.
+Only 301, 302, 303, 307 and 308 are valid values.
+ |
+
+
+
+allowCrossSchemeRedirect
+
+
+string
+
+ |
+
+(Optional)
+ AllowCrossSchemeRedirect Allow internal redirect to follow a target URI with a different scheme
+than the value of x-forwarded-proto.
+SafeOnly allows same scheme redirect and safe cross scheme redirect, which means if the downstream
+scheme is HTTPS, both HTTPS and HTTP redirect targets are allowed, but if the downstream scheme
+is HTTP, only HTTP redirect targets are allowed.
+ |
+
+
+
+denyRepeatedRouteRedirect
+
+
+bool
+
+ |
+
+(Optional)
+ If DenyRepeatedRouteRedirect is true, rejects redirect targets that are pointing to a route that has
+been followed by a previous redirect from the current route.
+ |
+
+
+
+HTTPProxySpec
+
+
+(Appears on:
+HTTPProxy)
+
+
+
HTTPProxySpec defines the spec of the CRD.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+virtualhost
+
+
+
+VirtualHost
+
+
+ |
+
+(Optional)
+ Virtualhost appears at most once. If it is present, the object is considered
+to be a “root” HTTPProxy.
+ |
+
+
+
+routes
+
+
+
+[]Route
+
+
+ |
+
+(Optional)
+ Routes are the ingress routes. If TCPProxy is present, Routes is ignored.
+ |
+
+
+
+tcpproxy
+
+
+
+TCPProxy
+
+
+ |
+
+(Optional)
+ TCPProxy holds TCP proxy information.
+ |
+
+
+
+includes
+
+
+
+[]Include
+
+
+ |
+
+(Optional)
+ Includes allow for specific routing configuration to be included from another HTTPProxy,
+possibly in another namespace.
+ |
+
+
+
+ingressClassName
+
+
+string
+
+ |
+
+(Optional)
+ IngressClassName optionally specifies the ingress class to use for this
+HTTPProxy. This replaces the deprecated kubernetes.io/ingress.class
+annotation. For backwards compatibility, when that annotation is set, it
+is given precedence over this field.
+ |
+
+
+
+HTTPProxyStatus
+
+
+(Appears on:
+HTTPProxy)
+
+
+
HTTPProxyStatus reports the current state of the HTTPProxy.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+currentStatus
+
+
+string
+
+ |
+
+(Optional)
+ |
+
+
+
+description
+
+
+string
+
+ |
+
+(Optional)
+ |
+
+
+
+loadBalancer
+
+
+
+Kubernetes core/v1.LoadBalancerStatus
+
+
+ |
+
+(Optional)
+ LoadBalancer contains the current status of the load balancer.
+ |
+
+
+
+conditions
+
+
+
+[]DetailedCondition
+
+
+ |
+
+(Optional)
+ Conditions contains information about the current status of the HTTPProxy,
+in an upstream-friendly container.
+Contour will update a single condition, Valid , that is in normal-true polarity.
+That is, when currentStatus is valid , the Valid condition will be status: true ,
+and vice versa.
+Contour will leave untouched any other Conditions set in this block,
+in case some other controller wants to add a Condition.
+If you are another controller owner and wish to add a condition, you should
+namespace your condition with a label, like controller.domain.com/ConditionName .
+ |
+
+
+
+HTTPRequestRedirectPolicy
+
+
+(Appears on:
+Route)
+
+
+
HTTPRequestRedirectPolicy defines configuration for redirecting a request.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+scheme
+
+
+string
+
+ |
+
+(Optional)
+ Scheme is the scheme to be used in the value of the Location
+header in the response.
+When empty, the scheme of the request is used.
+ |
+
+
+
+hostname
+
+
+string
+
+ |
+
+(Optional)
+ Hostname is the precise hostname to be used in the value of the Location
+header in the response.
+When empty, the hostname of the request is used.
+No wildcards are allowed.
+ |
+
+
+
+port
+
+
+int32
+
+ |
+
+(Optional)
+ Port is the port to be used in the value of the Location
+header in the response.
+When empty, port (if specified) of the request is used.
+ |
+
+
+
+statusCode
+
+
+int
+
+ |
+
+(Optional)
+ StatusCode is the HTTP status code to be used in response.
+ |
+
+
+
+path
+
+
+string
+
+ |
+
+(Optional)
+ Path allows for redirection to a different path from the
+original on the request. The path must start with a
+leading slash.
+Note: Only one of Path or Prefix can be defined.
+ |
+
+
+
+prefix
+
+
+string
+
+ |
+
+(Optional)
+ Prefix defines the value to swap the matched prefix or path with.
+The prefix must start with a leading slash.
+Note: Only one of Path or Prefix can be defined.
+ |
+
+
+
+HTTPStatusRange
+
+
+(Appears on:
+HTTPHealthCheckPolicy)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+start
+
+
+int64
+
+ |
+
+ The start (inclusive) of a range of HTTP status codes.
+ |
+
+
+
+end
+
+
+int64
+
+ |
+
+ The end (exclusive) of a range of HTTP status codes.
+ |
+
+
+
+
+
+(Appears on:
+RequestHashPolicy)
+
+
+
HeaderHashOptions contains options to configure a HTTP request header hash
+policy, used in request attribute hash based load balancing.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+headerName
+
+
+string
+
+ |
+
+ HeaderName is the name of the HTTP request header that will be used to
+calculate the hash key. If the header specified is not present on a
+request, no hash will be produced.
+ |
+
+
+
+
+
+(Appears on:
+MatchCondition,
+RequestHeaderValueMatchDescriptor)
+
+
+
HeaderMatchCondition specifies how to conditionally match against HTTP
+headers. The Name field is required, only one of Present, NotPresent,
+Contains, NotContains, Exact, NotExact and Regex can be set.
+For negative matching rules only (e.g. NotContains or NotExact) you can set
+TreatMissingAsEmpty.
+IgnoreCase has no effect for Regex.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name is the name of the header to match against. Name is required.
+Header names are case insensitive.
+ |
+
+
+
+present
+
+
+bool
+
+ |
+
+(Optional)
+ Present specifies that condition is true when the named header
+is present, regardless of its value. Note that setting Present
+to false does not make the condition true if the named header
+is absent.
+ |
+
+
+
+notpresent
+
+
+bool
+
+ |
+
+(Optional)
+ NotPresent specifies that condition is true when the named header
+is not present. Note that setting NotPresent to false does not
+make the condition true if the named header is present.
+ |
+
+
+
+contains
+
+
+string
+
+ |
+
+(Optional)
+ Contains specifies a substring that must be present in
+the header value.
+ |
+
+
+
+notcontains
+
+
+string
+
+ |
+
+(Optional)
+ NotContains specifies a substring that must not be present
+in the header value.
+ |
+
+
+
+ignoreCase
+
+
+bool
+
+ |
+
+(Optional)
+ IgnoreCase specifies that string matching should be case insensitive.
+Note that this has no effect on the Regex parameter.
+ |
+
+
+
+exact
+
+
+string
+
+ |
+
+(Optional)
+ Exact specifies a string that the header value must be equal to.
+ |
+
+
+
+notexact
+
+
+string
+
+ |
+
+(Optional)
+ NoExact specifies a string that the header value must not be
+equal to. The condition is true if the header has any other value.
+ |
+
+
+
+regex
+
+
+string
+
+ |
+
+(Optional)
+ Regex specifies a regular expression pattern that must match the header
+value.
+ |
+
+
+
+treatMissingAsEmpty
+
+
+bool
+
+ |
+
+(Optional)
+ TreatMissingAsEmpty specifies if the header match rule specified header
+does not exist, this header value will be treated as empty. Defaults to false.
+Unlike the underlying Envoy implementation this is only supported for
+negative matches (e.g. NotContains, NotExact).
+ |
+
+
+
+
+
+(Appears on:
+HeadersPolicy,
+LocalRateLimitPolicy)
+
+
+
HeaderValue represents a header name/value pair
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name represents a key of a header
+ |
+
+
+
+value
+
+
+string
+
+ |
+
+ Value represents the value of a header specified by a key
+ |
+
+
+
+
+
+(Appears on:
+Route,
+Service)
+
+
+
HeadersPolicy defines how headers are managed during forwarding.
+The Host
header is treated specially and if set in a HTTP request
+will be used as the SNI server name when forwarding over TLS. It is an
+error to attempt to set the Host
header in a HTTP response.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+set
+
+
+
+[]HeaderValue
+
+
+ |
+
+(Optional)
+ Set specifies a list of HTTP header values that will be set in the HTTP header.
+If the header does not exist it will be added, otherwise it will be overwritten with the new value.
+ |
+
+
+
+remove
+
+
+[]string
+
+ |
+
+(Optional)
+ Remove specifies a list of HTTP header names to remove.
+ |
+
+
+
+IPFilterPolicy
+
+
+(Appears on:
+Route,
+VirtualHost)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+source
+
+
+
+IPFilterSource
+
+
+ |
+
+ Source indicates how to determine the ip address to filter on, and can be
+one of two values:
+- Remote filters on the ip address of the client, accounting for PROXY and
+X-Forwarded-For as needed.
+- Peer filters on the ip of the network request, ignoring PROXY and
+X-Forwarded-For.
+ |
+
+
+
+cidr
+
+
+string
+
+ |
+
+ CIDR is a CIDR block of ipv4 or ipv6 addresses to filter on. This can also be
+a bare IP address (without a mask) to filter on exactly one address.
+ |
+
+
+
+IPFilterSource
+(string
alias)
+
+(Appears on:
+IPFilterPolicy)
+
+
+
IPFilterSource indicates which IP should be considered for filtering
+
+
+
+
+Value |
+Description |
+
+
+"Peer" |
+ |
+
"Remote" |
+ |
+
+
+Include
+
+
+(Appears on:
+HTTPProxySpec)
+
+
+
Include describes a set of policies that can be applied to an HTTPProxy in a namespace.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name of the HTTPProxy
+ |
+
+
+
+namespace
+
+
+string
+
+ |
+
+(Optional)
+ Namespace of the HTTPProxy to include. Defaults to the current namespace if not supplied.
+ |
+
+
+
+conditions
+
+
+
+[]MatchCondition
+
+
+ |
+
+(Optional)
+ Conditions are a set of rules that are applied to included HTTPProxies.
+In effect, they are added onto the Conditions of included HTTPProxy Route
+structs.
+When applied, they are merged using AND, with one exception:
+There can be only one Prefix MatchCondition per Conditions slice.
+More than one Prefix, or contradictory Conditions, will make the
+include invalid. Exact and Regex match conditions are not allowed
+on includes.
+ |
+
+
+
+JWTProvider
+
+
+(Appears on:
+VirtualHost)
+
+
+
JWTProvider defines how to verify JWTs on requests.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Unique name for the provider.
+ |
+
+
+
+default
+
+
+bool
+
+ |
+
+(Optional)
+ Whether the provider should apply to all
+routes in the HTTPProxy/its includes by
+default. At most one provider can be marked
+as the default. If no provider is marked
+as the default, individual routes must explicitly
+identify the provider they require.
+ |
+
+
+
+issuer
+
+
+string
+
+ |
+
+(Optional)
+ Issuer that JWTs are required to have in the “iss” field.
+If not provided, JWT issuers are not checked.
+ |
+
+
+
+audiences
+
+
+[]string
+
+ |
+
+(Optional)
+ Audiences that JWTs are allowed to have in the “aud” field.
+If not provided, JWT audiences are not checked.
+ |
+
+
+
+remoteJWKS
+
+
+
+RemoteJWKS
+
+
+ |
+
+ Remote JWKS to use for verifying JWT signatures.
+ |
+
+
+
+forwardJWT
+
+
+bool
+
+ |
+
+(Optional)
+ Whether the JWT should be forwarded to the backend
+service after successful verification. By default,
+the JWT is not forwarded.
+ |
+
+
+
+JWTVerificationPolicy
+
+
+(Appears on:
+Route)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+require
+
+
+string
+
+ |
+
+(Optional)
+ Require names a specific JWT provider (defined in the virtual host)
+to require for the route. If specified, this field overrides the
+default provider if one exists. If this field is not specified,
+the default provider will be required if one exists. At most one of
+this field or the “disabled” field can be specified.
+ |
+
+
+
+disabled
+
+
+bool
+
+ |
+
+(Optional)
+ Disabled defines whether to disable all JWT verification for this
+route. This can be used to opt specific routes out of the default
+JWT provider for the HTTPProxy. At most one of this field or the
+“require” field can be specified.
+ |
+
+
+
+LoadBalancerPolicy
+
+
+(Appears on:
+Route,
+TCPProxy,
+ExtensionServiceSpec)
+
+
+
LoadBalancerPolicy defines the load balancing policy.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+strategy
+
+
+string
+
+ |
+
+ Strategy specifies the policy used to balance requests
+across the pool of backend pods. Valid policy names are
+Random , RoundRobin , WeightedLeastRequest , Cookie ,
+and RequestHash . If an unknown strategy name is specified
+or no policy is supplied, the default RoundRobin policy
+is used.
+ |
+
+
+
+requestHashPolicies
+
+
+
+[]RequestHashPolicy
+
+
+ |
+
+ RequestHashPolicies contains a list of hash policies to apply when the
+RequestHash load balancing strategy is chosen. If an element of the
+supplied list of hash policies is invalid, it will be ignored. If the
+list of hash policies is empty after validation, the load balancing
+strategy will fall back to the default RoundRobin .
+ |
+
+
+
+LocalRateLimitPolicy
+
+
+(Appears on:
+RateLimitPolicy)
+
+
+
LocalRateLimitPolicy defines local rate limiting parameters.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+requests
+
+
+uint32
+
+ |
+
+ Requests defines how many requests per unit of time should
+be allowed before rate limiting occurs.
+ |
+
+
+
+unit
+
+
+string
+
+ |
+
+ Unit defines the period of time within which requests
+over the limit will be rate limited. Valid values are
+“second”, “minute” and “hour”.
+ |
+
+
+
+burst
+
+
+uint32
+
+ |
+
+(Optional)
+ Burst defines the number of requests above the requests per
+unit that should be allowed within a short period of time.
+ |
+
+
+
+responseStatusCode
+
+
+uint32
+
+ |
+
+(Optional)
+ ResponseStatusCode is the HTTP status code to use for responses
+to rate-limited requests. Codes must be in the 400-599 range
+(inclusive). If not specified, the Envoy default of 429 (Too
+Many Requests) is used.
+ |
+
+
+
+responseHeadersToAdd
+
+
+
+[]HeaderValue
+
+
+ |
+
+(Optional)
+ ResponseHeadersToAdd is an optional list of response headers to
+set when a request is rate-limited.
+ |
+
+
+
+MatchCondition
+
+
+(Appears on:
+Include,
+Route)
+
+
+
MatchCondition are a general holder for matching rules for HTTPProxies.
+One of Prefix, Exact, Regex, Header or QueryParameter must be provided.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+prefix
+
+
+string
+
+ |
+
+(Optional)
+ Prefix defines a prefix match for a request.
+ |
+
+
+
+exact
+
+
+string
+
+ |
+
+(Optional)
+ Exact defines a exact match for a request.
+This field is not allowed in include match conditions.
+ |
+
+
+
+regex
+
+
+string
+
+ |
+
+(Optional)
+ Regex defines a regex match for a request.
+This field is not allowed in include match conditions.
+ |
+
+
+
+header
+
+
+
+HeaderMatchCondition
+
+
+ |
+
+(Optional)
+ Header specifies the header condition to match.
+ |
+
+
+
+queryParameter
+
+
+
+QueryParameterMatchCondition
+
+
+ |
+
+(Optional)
+ QueryParameter specifies the query parameter condition to match.
+ |
+
+
+
+Namespace
+(string
alias)
+
+(Appears on:
+ContourSettings)
+
+
+
Namespace refers to a Kubernetes namespace. It must be a RFC 1123 label.
+This validation is based off of the corresponding Kubernetes validation:
+https://github.com/kubernetes/apimachinery/blob/02cfb53916346d085a6c6c7c66f882e3c6b0eca6/pkg/util/validation/validation.go#L187
+This is used for Namespace name validation here:
+https://github.com/kubernetes/apimachinery/blob/02cfb53916346d085a6c6c7c66f882e3c6b0eca6/pkg/api/validation/generic.go#L63
+Valid values include:
+
+Invalid values include:
+
+- “example.com” - “.” is an invalid character
+
+
+PathRewritePolicy
+
+
+(Appears on:
+Route)
+
+
+
PathRewritePolicy specifies how a request URL path should be
+rewritten. This rewriting takes place after a request is routed
+and has no subsequent effects on the proxy’s routing decision.
+No HTTP headers or body content is rewritten.
+Exactly one field in this struct may be specified.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+replacePrefix
+
+
+
+[]ReplacePrefix
+
+
+ |
+
+(Optional)
+ ReplacePrefix describes how the path prefix should be replaced.
+ |
+
+
+
+QueryParameterHashOptions
+
+
+(Appears on:
+RequestHashPolicy)
+
+
+
QueryParameterHashOptions contains options to configure a query parameter based hash
+policy, used in request attribute hash based load balancing.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+parameterName
+
+
+string
+
+ |
+
+ ParameterName is the name of the HTTP request query parameter that will be used to
+calculate the hash key. If the query parameter specified is not present on a
+request, no hash will be produced.
+ |
+
+
+
+QueryParameterMatchCondition
+
+
+(Appears on:
+MatchCondition)
+
+
+
QueryParameterMatchCondition specifies how to conditionally match against HTTP
+query parameters. The Name field is required, only one of Exact, Prefix,
+Suffix, Regex, Contains and Present can be set. IgnoreCase has no effect
+for Regex.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name is the name of the query parameter to match against. Name is required.
+Query parameter names are case insensitive.
+ |
+
+
+
+exact
+
+
+string
+
+ |
+
+(Optional)
+ Exact specifies a string that the query parameter value must be equal to.
+ |
+
+
+
+prefix
+
+
+string
+
+ |
+
+(Optional)
+ Prefix defines a prefix match for the query parameter value.
+ |
+
+
+
+suffix
+
+
+string
+
+ |
+
+(Optional)
+ Suffix defines a suffix match for a query parameter value.
+ |
+
+
+
+regex
+
+
+string
+
+ |
+
+(Optional)
+ Regex specifies a regular expression pattern that must match the query
+parameter value.
+ |
+
+
+
+contains
+
+
+string
+
+ |
+
+(Optional)
+ Contains specifies a substring that must be present in
+the query parameter value.
+ |
+
+
+
+ignoreCase
+
+
+bool
+
+ |
+
+(Optional)
+ IgnoreCase specifies that string matching should be case insensitive.
+Note that this has no effect on the Regex parameter.
+ |
+
+
+
+present
+
+
+bool
+
+ |
+
+(Optional)
+ Present specifies that condition is true when the named query parameter
+is present, regardless of its value. Note that setting Present
+to false does not make the condition true if the named query parameter
+is absent.
+ |
+
+
+
+RateLimitDescriptor
+
+
+(Appears on:
+GlobalRateLimitPolicy)
+
+
+
RateLimitDescriptor defines a list of key-value pair generators.
+
+
+RateLimitDescriptorEntry
+
+
+(Appears on:
+RateLimitDescriptor)
+
+
+
RateLimitDescriptorEntry is a key-value pair generator. Exactly
+one field on this struct must be non-nil.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+genericKey
+
+
+
+GenericKeyDescriptor
+
+
+ |
+
+(Optional)
+ GenericKey defines a descriptor entry with a static key and value.
+ |
+
+
+
+requestHeader
+
+
+
+RequestHeaderDescriptor
+
+
+ |
+
+(Optional)
+ RequestHeader defines a descriptor entry that’s populated only if
+a given header is present on the request. The descriptor key is static,
+and the descriptor value is equal to the value of the header.
+ |
+
+
+
+requestHeaderValueMatch
+
+
+
+RequestHeaderValueMatchDescriptor
+
+
+ |
+
+(Optional)
+ RequestHeaderValueMatch defines a descriptor entry that’s populated
+if the request’s headers match a set of 1+ match criteria. The
+descriptor key is “header_match”, and the descriptor value is static.
+ |
+
+
+
+remoteAddress
+
+
+
+RemoteAddressDescriptor
+
+
+ |
+
+(Optional)
+ RemoteAddress defines a descriptor entry with a key of “remote_address”
+and a value equal to the client’s IP address (from x-forwarded-for).
+ |
+
+
+
+RateLimitPolicy
+
+
+(Appears on:
+Route,
+VirtualHost)
+
+
+
RateLimitPolicy defines rate limiting parameters.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+local
+
+
+
+LocalRateLimitPolicy
+
+
+ |
+
+(Optional)
+ Local defines local rate limiting parameters, i.e. parameters
+for rate limiting that occurs within each Envoy pod as requests
+are handled.
+ |
+
+
+
+global
+
+
+
+GlobalRateLimitPolicy
+
+
+ |
+
+(Optional)
+ Global defines global rate limiting parameters, i.e. parameters
+defining descriptors that are sent to an external rate limit
+service (RLS) for a rate limit decision on each request.
+ |
+
+
+
+RedirectResponseCode
+(uint32
alias)
+
+(Appears on:
+HTTPInternalRedirectPolicy)
+
+
+
RedirectResponseCode is a uint32 type alias with validation to ensure that the value is valid.
+
+RemoteAddressDescriptor
+
+
+(Appears on:
+RateLimitDescriptorEntry)
+
+
+
RemoteAddressDescriptor defines a descriptor entry with a key of
+“remote_address” and a value equal to the client’s IP address
+(from x-forwarded-for).
+
+RemoteJWKS
+
+
+(Appears on:
+JWTProvider)
+
+
+
RemoteJWKS defines how to fetch a JWKS from an HTTP endpoint.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+uri
+
+
+string
+
+ |
+
+ The URI for the JWKS.
+ |
+
+
+
+validation
+
+
+
+UpstreamValidation
+
+
+ |
+
+(Optional)
+ UpstreamValidation defines how to verify the JWKS’s TLS certificate.
+ |
+
+
+
+timeout
+
+
+string
+
+ |
+
+(Optional)
+ How long to wait for a response from the URI.
+If not specified, a default of 1s applies.
+ |
+
+
+
+cacheDuration
+
+
+string
+
+ |
+
+(Optional)
+ How long to cache the JWKS locally. If not specified,
+Envoy’s default of 5m applies.
+ |
+
+
+
+dnsLookupFamily
+
+
+string
+
+ |
+
+(Optional)
+ The DNS IP address resolution policy for the JWKS URI.
+When configured as “v4”, the DNS resolver will only perform a lookup
+for addresses in the IPv4 family. If “v6” is configured, the DNS resolver
+will only perform a lookup for addresses in the IPv6 family.
+If “all” is configured, the DNS resolver
+will perform a lookup for addresses in both the IPv4 and IPv6 family.
+If “auto” is configured, the DNS resolver will first perform a lookup
+for addresses in the IPv6 family and fallback to a lookup for addresses
+in the IPv4 family. If not specified, the Contour-wide setting defined
+in the config file or ContourConfiguration applies (defaults to “auto”).
+See https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/cluster.proto.html#envoy-v3-api-enum-config-cluster-v3-cluster-dnslookupfamily
+for more information.
+ |
+
+
+
+ReplacePrefix
+
+
+(Appears on:
+PathRewritePolicy)
+
+
+
ReplacePrefix describes a path prefix replacement.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+prefix
+
+
+string
+
+ |
+
+(Optional)
+ Prefix specifies the URL path prefix to be replaced.
+If Prefix is specified, it must exactly match the MatchCondition
+prefix that is rendered by the chain of including HTTPProxies
+and only that path prefix will be replaced by Replacement.
+This allows HTTPProxies that are included through multiple
+roots to only replace specific path prefixes, leaving others
+unmodified.
+If Prefix is not specified, all routing prefixes rendered
+by the include chain will be replaced.
+ |
+
+
+
+replacement
+
+
+string
+
+ |
+
+ Replacement is the string that the routing path prefix
+will be replaced with. This must not be empty.
+ |
+
+
+
+RequestHashPolicy
+
+
+(Appears on:
+LoadBalancerPolicy)
+
+
+
RequestHashPolicy contains configuration for an individual hash policy
+on a request attribute.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+terminal
+
+
+bool
+
+ |
+
+ Terminal is a flag that allows for short-circuiting computing of a hash
+for a given request. If set to true, and the request attribute specified
+in the attribute hash options is present, no further hash policies will
+be used to calculate a hash for the request.
+ |
+
+
+
+headerHashOptions
+
+
+
+HeaderHashOptions
+
+
+ |
+
+(Optional)
+ HeaderHashOptions should be set when request header hash based load
+balancing is desired. It must be the only hash option field set,
+otherwise this request hash policy object will be ignored.
+ |
+
+
+
+queryParameterHashOptions
+
+
+
+QueryParameterHashOptions
+
+
+ |
+
+(Optional)
+ QueryParameterHashOptions should be set when request query parameter hash based load
+balancing is desired. It must be the only hash option field set,
+otherwise this request hash policy object will be ignored.
+ |
+
+
+
+hashSourceIP
+
+
+bool
+
+ |
+
+(Optional)
+ HashSourceIP should be set to true when request source IP hash based
+load balancing is desired. It must be the only hash option field set,
+otherwise this request hash policy object will be ignored.
+ |
+
+
+
+
+
+(Appears on:
+RateLimitDescriptorEntry)
+
+
+
RequestHeaderDescriptor defines a descriptor entry that’s populated only
+if a given header is present on the request. The value of the descriptor
+entry is equal to the value of the header (if present).
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+headerName
+
+
+string
+
+ |
+
+ HeaderName defines the name of the header to look for on the request.
+ |
+
+
+
+descriptorKey
+
+
+string
+
+ |
+
+ DescriptorKey defines the key to use on the descriptor entry.
+ |
+
+
+
+
+
+(Appears on:
+RateLimitDescriptorEntry)
+
+
+
RequestHeaderValueMatchDescriptor defines a descriptor entry that’s populated
+if the request’s headers match a set of 1+ match criteria. The descriptor key
+is “header_match”, and the descriptor value is statically defined.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+headers
+
+
+
+[]HeaderMatchCondition
+
+
+ |
+
+ Headers is a list of 1+ match criteria to apply against the request
+to determine whether to populate the descriptor entry or not.
+ |
+
+
+
+expectMatch
+
+
+bool
+
+ |
+
+ ExpectMatch defines whether the request must positively match the match
+criteria in order to generate a descriptor entry (i.e. true), or not
+match the match criteria in order to generate a descriptor entry (i.e. false).
+The default is true.
+ |
+
+
+
+value
+
+
+string
+
+ |
+
+ Value defines the value of the descriptor entry.
+ |
+
+
+
+RetryOn
+(string
alias)
+
+(Appears on:
+RetryPolicy)
+
+
+
RetryOn is a string type alias with validation to ensure that the value is valid.
+
+RetryPolicy
+
+
+(Appears on:
+Route)
+
+
+
RetryPolicy defines the attributes associated with retrying policy.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+count
+
+
+int64
+
+ |
+
+(Optional)
+ NumRetries is maximum allowed number of retries.
+If set to -1, then retries are disabled.
+If set to 0 or not supplied, the value is set
+to the Envoy default of 1.
+ |
+
+
+
+perTryTimeout
+
+
+string
+
+ |
+
+(Optional)
+ PerTryTimeout specifies the timeout per retry attempt.
+Ignored if NumRetries is not supplied.
+ |
+
+
+
+retryOn
+
+
+
+[]RetryOn
+
+
+ |
+
+(Optional)
+ RetryOn specifies the conditions on which to retry a request.
+Supported HTTP conditions:
+
+5xx
+gateway-error
+reset
+connect-failure
+retriable-4xx
+refused-stream
+retriable-status-codes
+retriable-headers
+
+Supported gRPC conditions:
+
+cancelled
+deadline-exceeded
+internal
+resource-exhausted
+unavailable
+
+ |
+
+
+
+retriableStatusCodes
+
+
+[]uint32
+
+ |
+
+(Optional)
+ RetriableStatusCodes specifies the HTTP status codes that should be retried.
+This field is only respected when you include retriable-status-codes in the RetryOn field.
+ |
+
+
+
+Route
+
+
+(Appears on:
+HTTPProxySpec)
+
+
+
Route contains the set of routes for a virtual host.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+conditions
+
+
+
+[]MatchCondition
+
+
+ |
+
+(Optional)
+ Conditions are a set of rules that are applied to a Route.
+When applied, they are merged using AND, with one exception:
+There can be only one Prefix, Exact or Regex MatchCondition
+per Conditions slice. More than one of these condition types,
+or contradictory Conditions, will make the route invalid.
+ |
+
+
+
+services
+
+
+
+[]Service
+
+
+ |
+
+(Optional)
+ Services are the services to proxy traffic.
+ |
+
+
+
+enableWebsockets
+
+
+bool
+
+ |
+
+(Optional)
+ Enables websocket support for the route.
+ |
+
+
+
+permitInsecure
+
+
+bool
+
+ |
+
+(Optional)
+ Allow this path to respond to insecure requests over HTTP which are normally
+not permitted when a virtualhost.tls block is present.
+ |
+
+
+
+authPolicy
+
+
+
+AuthorizationPolicy
+
+
+ |
+
+(Optional)
+ AuthPolicy updates the authorization policy that was set
+on the root HTTPProxy object for client requests that
+match this route.
+ |
+
+
+
+timeoutPolicy
+
+
+
+TimeoutPolicy
+
+
+ |
+
+(Optional)
+ The timeout policy for this route.
+ |
+
+
+
+retryPolicy
+
+
+
+RetryPolicy
+
+
+ |
+
+(Optional)
+ The retry policy for this route.
+ |
+
+
+
+healthCheckPolicy
+
+
+
+HTTPHealthCheckPolicy
+
+
+ |
+
+(Optional)
+ The health check policy for this route.
+ |
+
+
+
+loadBalancerPolicy
+
+
+
+LoadBalancerPolicy
+
+
+ |
+
+(Optional)
+ The load balancing policy for this route.
+ |
+
+
+
+pathRewritePolicy
+
+
+
+PathRewritePolicy
+
+
+ |
+
+(Optional)
+ The policy for rewriting the path of the request URL
+after the request has been routed to a Service.
+ |
+
+
+
+requestHeadersPolicy
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ The policy for managing request headers during proxying.
+You may dynamically rewrite the Host header to be forwarded
+upstream to the content of a request header using
+the below format “%REQ(X-Header-Name)%”. If the value of the header
+is empty, it is ignored.
+*NOTE: Pay attention to the potential security implications of using this option.
+Provided header must come from trusted source.
+**NOTE: The header rewrite is only done while forwarding and has no bearing
+on the routing decision.
+ |
+
+
+
+responseHeadersPolicy
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ The policy for managing response headers during proxying.
+Rewriting the ‘Host’ header is not supported.
+ |
+
+
+
+cookieRewritePolicies
+
+
+
+[]CookieRewritePolicy
+
+
+ |
+
+(Optional)
+ The policies for rewriting Set-Cookie header attributes. Note that
+rewritten cookie names must be unique in this list. Order rewrite
+policies are specified in does not matter.
+ |
+
+
+
+rateLimitPolicy
+
+
+
+RateLimitPolicy
+
+
+ |
+
+(Optional)
+ The policy for rate limiting on the route.
+ |
+
+
+
+requestRedirectPolicy
+
+
+
+HTTPRequestRedirectPolicy
+
+
+ |
+
+(Optional)
+ RequestRedirectPolicy defines an HTTP redirection.
+ |
+
+
+
+directResponsePolicy
+
+
+
+HTTPDirectResponsePolicy
+
+
+ |
+
+(Optional)
+ DirectResponsePolicy returns an arbitrary HTTP response directly.
+ |
+
+
+
+internalRedirectPolicy
+
+
+
+HTTPInternalRedirectPolicy
+
+
+ |
+
+(Optional)
+ The policy to define when to handle redirects responses internally.
+ |
+
+
+
+jwtVerificationPolicy
+
+
+
+JWTVerificationPolicy
+
+
+ |
+
+(Optional)
+ The policy for verifying JWTs for requests to this route.
+ |
+
+
+
+ipAllowPolicy
+
+
+
+[]IPFilterPolicy
+
+
+ |
+
+ IPAllowFilterPolicy is a list of ipv4/6 filter rules for which matching
+requests should be allowed. All other requests will be denied.
+Only one of IPAllowFilterPolicy and IPDenyFilterPolicy can be defined.
+The rules defined here override any rules set on the root HTTPProxy.
+ |
+
+
+
+ipDenyPolicy
+
+
+
+[]IPFilterPolicy
+
+
+ |
+
+ IPDenyFilterPolicy is a list of ipv4/6 filter rules for which matching
+requests should be denied. All other requests will be allowed.
+Only one of IPAllowFilterPolicy and IPDenyFilterPolicy can be defined.
+The rules defined here override any rules set on the root HTTPProxy.
+ |
+
+
+
+Service
+
+
+(Appears on:
+Route,
+TCPProxy)
+
+
+
Service defines an Kubernetes Service to proxy traffic.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name is the name of Kubernetes service to proxy traffic.
+Names defined here will be used to look up corresponding endpoints which contain the ips to route.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+ Port (defined as Integer) to proxy traffic to since a service can have multiple defined.
+ |
+
+
+
+healthPort
+
+
+int
+
+ |
+
+(Optional)
+ HealthPort is the port for this service healthcheck.
+If not specified, Port is used for service healthchecks.
+ |
+
+
+
+protocol
+
+
+string
+
+ |
+
+(Optional)
+ Protocol may be used to specify (or override) the protocol used to reach this Service.
+Values may be tls, h2, h2c. If omitted, protocol-selection falls back on Service annotations.
+ |
+
+
+
+weight
+
+
+int64
+
+ |
+
+(Optional)
+ Weight defines percentage of traffic to balance traffic
+ |
+
+
+
+validation
+
+
+
+UpstreamValidation
+
+
+ |
+
+(Optional)
+ UpstreamValidation defines how to verify the backend service’s certificate
+ |
+
+
+
+mirror
+
+
+bool
+
+ |
+
+ If Mirror is true the Service will receive a read only mirror of the traffic for this route.
+If Mirror is true, then fractional mirroring can be enabled by optionally setting the Weight
+field. Legal values for Weight are 1-100. Omitting the Weight field will result in 100% mirroring.
+NOTE: Setting Weight explicitly to 0 will unexpectedly result in 100% traffic mirroring. This
+occurs since we cannot distinguish omitted fields from those explicitly set to their default
+values
+ |
+
+
+
+requestHeadersPolicy
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ The policy for managing request headers during proxying.
+ |
+
+
+
+responseHeadersPolicy
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ The policy for managing response headers during proxying.
+Rewriting the ‘Host’ header is not supported.
+ |
+
+
+
+cookieRewritePolicies
+
+
+
+[]CookieRewritePolicy
+
+
+ |
+
+(Optional)
+ The policies for rewriting Set-Cookie header attributes.
+ |
+
+
+
+slowStartPolicy
+
+
+
+SlowStartPolicy
+
+
+ |
+
+(Optional)
+ Slow start will gradually increase amount of traffic to a newly added endpoint.
+ |
+
+
+
+SlowStartPolicy
+
+
+(Appears on:
+Service)
+
+
+
SlowStartPolicy will gradually increase amount of traffic to a newly added endpoint.
+It can be used only with RoundRobin and WeightedLeastRequest load balancing strategies.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+window
+
+
+string
+
+ |
+
+ The duration of slow start window.
+Duration is expressed in the Go Duration format.
+Valid time units are “ns”, “us” (or “µs”), “ms”, “s”, “m”, “h”.
+ |
+
+
+
+aggression
+
+
+string
+
+ |
+
+(Optional)
+ The speed of traffic increase over the slow start window.
+Defaults to 1.0, so that endpoint would get linearly increasing amount of traffic.
+When increasing the value for this parameter, the speed of traffic ramp-up increases non-linearly.
+The value of aggression parameter should be greater than 0.0.
+More info: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/upstream/load_balancing/slow_start
+ |
+
+
+
+minWeightPercent
+
+
+uint32
+
+ |
+
+(Optional)
+ The minimum or starting percentage of traffic to send to new endpoints.
+A non-zero value helps avoid a too small initial weight, which may cause endpoints in slow start mode to receive no traffic in the beginning of the slow start window.
+If not specified, the default is 10%.
+ |
+
+
+
+SubCondition
+
+
+(Appears on:
+DetailedCondition)
+
+
+
SubCondition is a Condition-like type intended for use as a subcondition inside a DetailedCondition.
+It contains a subset of the Condition fields.
+It is intended for warnings and errors, so type
names should use abnormal-true polarity,
+that is, they should be of the form “ErrorPresent: true”.
+The expected lifecycle for these errors is that they should only be present when the error or warning is,
+and should be removed when they are not relevant.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+type
+
+
+string
+
+ |
+
+ Type of condition in CamelCase or in foo.example.com/CamelCase .
+This must be in abnormal-true polarity, that is, ErrorFound or controller.io/ErrorFound .
+The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt)
+ |
+
+
+
+status
+
+
+
+Kubernetes meta/v1.ConditionStatus
+
+
+ |
+
+ Status of the condition, one of True, False, Unknown.
+ |
+
+
+
+reason
+
+
+string
+
+ |
+
+ Reason contains a programmatic identifier indicating the reason for the condition’s last transition.
+Producers of specific condition types may define expected values and meanings for this field,
+and whether the values are considered a guaranteed API.
+The value should be a CamelCase string.
+This field may not be empty.
+ |
+
+
+
+message
+
+
+string
+
+ |
+
+ Message is a human readable message indicating details about the transition.
+This may be an empty string.
+ |
+
+
+
+TCPHealthCheckPolicy
+
+
+(Appears on:
+TCPProxy)
+
+
+
TCPHealthCheckPolicy defines health checks on the upstream service.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+intervalSeconds
+
+
+int64
+
+ |
+
+(Optional)
+ The interval (seconds) between health checks
+ |
+
+
+
+timeoutSeconds
+
+
+int64
+
+ |
+
+(Optional)
+ The time to wait (seconds) for a health check response
+ |
+
+
+
+unhealthyThresholdCount
+
+
+uint32
+
+ |
+
+(Optional)
+ The number of unhealthy health checks required before a host is marked unhealthy
+ |
+
+
+
+healthyThresholdCount
+
+
+uint32
+
+ |
+
+(Optional)
+ The number of healthy health checks required before a host is marked healthy
+ |
+
+
+
+TCPProxy
+
+
+(Appears on:
+HTTPProxySpec)
+
+
+
TCPProxy contains the set of services to proxy TCP connections.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+loadBalancerPolicy
+
+
+
+LoadBalancerPolicy
+
+
+ |
+
+(Optional)
+ The load balancing policy for the backend services. Note that the
+Cookie and RequestHash load balancing strategies cannot be used
+here.
+ |
+
+
+
+services
+
+
+
+[]Service
+
+
+ |
+
+(Optional)
+ Services are the services to proxy traffic
+ |
+
+
+
+include
+
+
+
+TCPProxyInclude
+
+
+ |
+
+(Optional)
+ Include specifies that this tcpproxy should be delegated to another HTTPProxy.
+ |
+
+
+
+includes
+
+
+
+TCPProxyInclude
+
+
+ |
+
+(Optional)
+ IncludesDeprecated allow for specific routing configuration to be appended to another HTTPProxy in another namespace.
+Exists due to a mistake when developing HTTPProxy and the field was marked plural
+when it should have been singular. This field should stay to not break backwards compatibility to v1 users.
+ |
+
+
+
+healthCheckPolicy
+
+
+
+TCPHealthCheckPolicy
+
+
+ |
+
+(Optional)
+ The health check policy for this tcp proxy
+ |
+
+
+
+TCPProxyInclude
+
+
+(Appears on:
+TCPProxy)
+
+
+
TCPProxyInclude describes a target HTTPProxy document which contains the TCPProxy details.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name of the child HTTPProxy
+ |
+
+
+
+namespace
+
+
+string
+
+ |
+
+(Optional)
+ Namespace of the HTTPProxy to include. Defaults to the current namespace if not supplied.
+ |
+
+
+
+TLS
+
+
+(Appears on:
+VirtualHost)
+
+
+
TLS describes tls properties. The SNI names that will be matched on
+are described in the HTTPProxy’s Spec.VirtualHost.Fqdn field.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+secretName
+
+
+string
+
+ |
+
+ SecretName is the name of a TLS secret.
+Either SecretName or Passthrough must be specified, but not both.
+If specified, the named secret must contain a matching certificate
+for the virtual host’s FQDN.
+The name can be optionally prefixed with namespace “namespace/name”.
+When cross-namespace reference is used, TLSCertificateDelegation resource must exist in the namespace to grant access to the secret.
+ |
+
+
+
+minimumProtocolVersion
+
+
+string
+
+ |
+
+(Optional)
+ MinimumProtocolVersion is the minimum TLS version this vhost should
+negotiate. Valid options are 1.2 (default) and 1.3 . Any other value
+defaults to TLS 1.2.
+ |
+
+
+
+maximumProtocolVersion
+
+
+string
+
+ |
+
+(Optional)
+ MaximumProtocolVersion is the maximum TLS version this vhost should
+negotiate. Valid options are 1.2 and 1.3 (default). Any other value
+defaults to TLS 1.3.
+ |
+
+
+
+passthrough
+
+
+bool
+
+ |
+
+(Optional)
+ Passthrough defines whether the encrypted TLS handshake will be
+passed through to the backing cluster. Either Passthrough or
+SecretName must be specified, but not both.
+ |
+
+
+
+clientValidation
+
+
+
+DownstreamValidation
+
+
+ |
+
+(Optional)
+ ClientValidation defines how to verify the client certificate
+when an external client establishes a TLS connection to Envoy.
+This setting:
+
+- Enables TLS client certificate validation.
+- Specifies how the client certificate will be validated (i.e.
+validation required or skipped).
+
+Note: Setting client certificate validation to be skipped should
+be only used in conjunction with an external authorization server that
+performs client validation as Contour will ensure client certificates
+are passed along.
+ |
+
+
+
+enableFallbackCertificate
+
+
+bool
+
+ |
+
+ EnableFallbackCertificate defines if the vhost should allow a default certificate to
+be applied which handles all requests which don’t match the SNI defined in this vhost.
+ |
+
+
+
+TLSCertificateDelegationSpec
+
+
+(Appears on:
+TLSCertificateDelegation)
+
+
+
TLSCertificateDelegationSpec defines the spec of the CRD
+
+
+TLSCertificateDelegationStatus
+
+
+(Appears on:
+TLSCertificateDelegation)
+
+
+
TLSCertificateDelegationStatus allows for the status of the delegation
+to be presented to the user.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+conditions
+
+
+
+[]DetailedCondition
+
+
+ |
+
+(Optional)
+ Conditions contains information about the current status of the HTTPProxy,
+in an upstream-friendly container.
+Contour will update a single condition, Valid , that is in normal-true polarity.
+That is, when currentStatus is valid , the Valid condition will be status: true ,
+and vice versa.
+Contour will leave untouched any other Conditions set in this block,
+in case some other controller wants to add a Condition.
+If you are another controller owner and wish to add a condition, you should
+namespace your condition with a label, like controller.domain.com\ConditionName .
+ |
+
+
+
+TimeoutPolicy
+
+
+(Appears on:
+Route,
+ExtensionServiceSpec)
+
+
+
TimeoutPolicy configures timeouts that are used for handling network requests.
+TimeoutPolicy durations are expressed in the Go Duration format.
+Valid time units are “ns”, “us” (or “µs”), “ms”, “s”, “m”, “h”.
+The string “infinity” is also a valid input and specifies no timeout.
+A value of “0s” will be treated as if the field were not set, i.e. by using Envoy’s default behavior.
+Example input values: “300ms”, “5s”, “1m”.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+response
+
+
+string
+
+ |
+
+(Optional)
+ Timeout for receiving a response from the server after processing a request from client.
+If not supplied, Envoy’s default value of 15s applies.
+ |
+
+
+
+idle
+
+
+string
+
+ |
+
+(Optional)
+ Timeout for how long the proxy should wait while there is no activity during single request/response (for HTTP/1.1) or stream (for HTTP/2).
+Timeout will not trigger while HTTP/1.1 connection is idle between two consecutive requests.
+If not specified, there is no per-route idle timeout, though a connection manager-wide
+stream_idle_timeout default of 5m still applies.
+ |
+
+
+
+idleConnection
+
+
+string
+
+ |
+
+(Optional)
+ Timeout for how long connection from the proxy to the upstream service is kept when there are no active requests.
+If not supplied, Envoy’s default value of 1h applies.
+ |
+
+
+
+UpstreamValidation
+
+
+(Appears on:
+RemoteJWKS,
+Service,
+ExtensionServiceSpec)
+
+
+
UpstreamValidation defines how to verify the backend service’s certificate
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+caSecret
+
+
+string
+
+ |
+
+ Name or namespaced name of the Kubernetes secret used to validate the certificate presented by the backend.
+The secret must contain key named ca.crt.
+The name can be optionally prefixed with namespace “namespace/name”.
+When cross-namespace reference is used, TLSCertificateDelegation resource must exist in the namespace to grant access to the secret.
+Max length should be the actual max possible length of a namespaced name (63 + 253 + 1 = 317)
+ |
+
+
+
+subjectName
+
+
+string
+
+ |
+
+ Key which is expected to be present in the ‘subjectAltName’ of the presented certificate.
+Deprecated: migrate to using the plural field subjectNames.
+ |
+
+
+
+subjectNames
+
+
+[]string
+
+ |
+
+(Optional)
+ List of keys, of which at least one is expected to be present in the ‘subjectAltName of the
+presented certificate.
+ |
+
+
+
+VirtualHost
+
+
+(Appears on:
+HTTPProxySpec)
+
+
+
VirtualHost appears at most once. If it is present, the object is considered
+to be a “root”.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+fqdn
+
+
+string
+
+ |
+
+ The fully qualified domain name of the root of the ingress tree
+all leaves of the DAG rooted at this object relate to the fqdn.
+ |
+
+
+
+tls
+
+
+
+TLS
+
+
+ |
+
+(Optional)
+ If present the fields describes TLS properties of the virtual
+host. The SNI names that will be matched on are described in fqdn,
+the tls.secretName secret must contain a certificate that itself
+contains a name that matches the FQDN.
+ |
+
+
+
+authorization
+
+
+
+AuthorizationServer
+
+
+ |
+
+(Optional)
+ This field configures an extension service to perform
+authorization for this virtual host. Authorization can
+only be configured on virtual hosts that have TLS enabled.
+If the TLS configuration requires client certificate
+validation, the client certificate is always included in the
+authentication check request.
+ |
+
+
+
+corsPolicy
+
+
+
+CORSPolicy
+
+
+ |
+
+(Optional)
+ Specifies the cross-origin policy to apply to the VirtualHost.
+ |
+
+
+
+rateLimitPolicy
+
+
+
+RateLimitPolicy
+
+
+ |
+
+(Optional)
+ The policy for rate limiting on the virtual host.
+ |
+
+
+
+jwtProviders
+
+
+
+[]JWTProvider
+
+
+ |
+
+(Optional)
+ Providers to use for verifying JSON Web Tokens (JWTs) on the virtual host.
+ |
+
+
+
+ipAllowPolicy
+
+
+
+[]IPFilterPolicy
+
+
+ |
+
+ IPAllowFilterPolicy is a list of ipv4/6 filter rules for which matching
+requests should be allowed. All other requests will be denied.
+Only one of IPAllowFilterPolicy and IPDenyFilterPolicy can be defined.
+The rules defined here may be overridden in a Route.
+ |
+
+
+
+ipDenyPolicy
+
+
+
+[]IPFilterPolicy
+
+
+ |
+
+ IPDenyFilterPolicy is a list of ipv4/6 filter rules for which matching
+requests should be denied. All other requests will be allowed.
+Only one of IPAllowFilterPolicy and IPDenyFilterPolicy can be defined.
+The rules defined here may be overridden in a Route.
+ |
+
+
+
+
+projectcontour.io/v1alpha1
+
+
Package v1alpha1 contains API Schema definitions for the projectcontour.io v1alpha1 API group
+
+Resource Types:
+
+ContourConfiguration
+
+
+
ContourConfiguration is the schema for a Contour instance.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+apiVersion
+string |
+
+
+projectcontour.io/v1alpha1
+
+ |
+
+
+
+kind
+string
+ |
+ContourConfiguration |
+
+
+
+metadata
+
+
+
+Kubernetes meta/v1.ObjectMeta
+
+
+ |
+
+Refer to the Kubernetes API documentation for the fields of the
+metadata field.
+ |
+
+
+
+spec
+
+
+
+ContourConfigurationSpec
+
+
+ |
+
+
+
+
+
+
+xdsServer
+
+
+
+XDSServerConfig
+
+
+ |
+
+(Optional)
+ XDSServer contains parameters for the xDS server.
+ |
+
+
+
+ingress
+
+
+
+IngressConfig
+
+
+ |
+
+(Optional)
+ Ingress contains parameters for ingress options.
+ |
+
+
+
+debug
+
+
+
+DebugConfig
+
+
+ |
+
+(Optional)
+ Debug contains parameters to enable debug logging
+and debug interfaces inside Contour.
+ |
+
+
+
+health
+
+
+
+HealthConfig
+
+
+ |
+
+(Optional)
+ Health defines the endpoints Contour uses to serve health checks.
+Contour’s default is { address: “0.0.0.0”, port: 8000 }.
+ |
+
+
+
+envoy
+
+
+
+EnvoyConfig
+
+
+ |
+
+(Optional)
+ Envoy contains parameters for Envoy as well
+as how to optionally configure a managed Envoy fleet.
+ |
+
+
+
+gateway
+
+
+
+GatewayConfig
+
+
+ |
+
+(Optional)
+ Gateway contains parameters for the gateway-api Gateway that Contour
+is configured to serve traffic.
+ |
+
+
+
+httpproxy
+
+
+
+HTTPProxyConfig
+
+
+ |
+
+(Optional)
+ HTTPProxy defines parameters on HTTPProxy.
+ |
+
+
+
+enableExternalNameService
+
+
+bool
+
+ |
+
+(Optional)
+ EnableExternalNameService allows processing of ExternalNameServices
+Contour’s default is false for security reasons.
+ |
+
+
+
+globalExtAuth
+
+
+
+AuthorizationServer
+
+
+ |
+
+(Optional)
+ GlobalExternalAuthorization allows envoys external authorization filter
+to be enabled for all virtual hosts.
+ |
+
+
+
+rateLimitService
+
+
+
+RateLimitServiceConfig
+
+
+ |
+
+(Optional)
+ RateLimitService optionally holds properties of the Rate Limit Service
+to be used for global rate limiting.
+ |
+
+
+
+policy
+
+
+
+PolicyConfig
+
+
+ |
+
+(Optional)
+ Policy specifies default policy applied if not overridden by the user
+ |
+
+
+
+metrics
+
+
+
+MetricsConfig
+
+
+ |
+
+(Optional)
+ Metrics defines the endpoint Contour uses to serve metrics.
+Contour’s default is { address: “0.0.0.0”, port: 8000 }.
+ |
+
+
+
+tracing
+
+
+
+TracingConfig
+
+
+ |
+
+ Tracing defines properties for exporting trace data to OpenTelemetry.
+ |
+
+
+
+featureFlags
+
+
+
+FeatureFlags
+
+
+ |
+
+ FeatureFlags defines toggle to enable new contour features.
+Available toggles are:
+useEndpointSlices - configures contour to fetch endpoint data
+from k8s endpoint slices. defaults to false and reading endpoint
+data from the k8s endpoints.
+ |
+
+
+ |
+
+
+
+status
+
+
+
+ContourConfigurationStatus
+
+
+ |
+
+(Optional)
+ |
+
+
+
+ContourDeployment
+
+
+
ContourDeployment is the schema for a Contour Deployment.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+apiVersion
+string |
+
+
+projectcontour.io/v1alpha1
+
+ |
+
+
+
+kind
+string
+ |
+ContourDeployment |
+
+
+
+metadata
+
+
+
+Kubernetes meta/v1.ObjectMeta
+
+
+ |
+
+Refer to the Kubernetes API documentation for the fields of the
+metadata field.
+ |
+
+
+
+spec
+
+
+
+ContourDeploymentSpec
+
+
+ |
+
+
+
+
+
+
+contour
+
+
+
+ContourSettings
+
+
+ |
+
+(Optional)
+ Contour specifies deployment-time settings for the Contour
+part of the installation, i.e. the xDS server/control plane
+and associated resources, including things like replica count
+for the Deployment, and node placement constraints for the pods.
+ |
+
+
+
+envoy
+
+
+
+EnvoySettings
+
+
+ |
+
+(Optional)
+ Envoy specifies deployment-time settings for the Envoy
+part of the installation, i.e. the xDS client/data plane
+and associated resources, including things like the workload
+type to use (DaemonSet or Deployment), node placement constraints
+for the pods, and various options for the Envoy service.
+ |
+
+
+
+runtimeSettings
+
+
+
+ContourConfigurationSpec
+
+
+ |
+
+(Optional)
+ RuntimeSettings is a ContourConfiguration spec to be used when
+provisioning a Contour instance that will influence aspects of
+the Contour instance’s runtime behavior.
+ |
+
+
+
+resourceLabels
+
+
+map[string]string
+
+ |
+
+(Optional)
+ ResourceLabels is a set of labels to add to the provisioned Contour resources.
+Deprecated: use Gateway.Spec.Infrastructure.Labels instead. This field will be
+removed in a future release.
+ |
+
+
+ |
+
+
+
+status
+
+
+
+ContourDeploymentStatus
+
+
+ |
+
+ |
+
+
+
+ExtensionService
+
+
+
ExtensionService is the schema for the Contour extension services API.
+An ExtensionService resource binds a network service to the Contour
+API so that Contour API features can be implemented by collaborating
+components.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+apiVersion
+string |
+
+
+projectcontour.io/v1alpha1
+
+ |
+
+
+
+kind
+string
+ |
+ExtensionService |
+
+
+
+metadata
+
+
+
+Kubernetes meta/v1.ObjectMeta
+
+
+ |
+
+Refer to the Kubernetes API documentation for the fields of the
+metadata field.
+ |
+
+
+
+spec
+
+
+
+ExtensionServiceSpec
+
+
+ |
+
+
+
+
+
+
+services
+
+
+
+[]ExtensionServiceTarget
+
+
+ |
+
+ Services specifies the set of Kubernetes Service resources that
+receive GRPC extension API requests.
+If no weights are specified for any of the entries in
+this array, traffic will be spread evenly across all the
+services.
+Otherwise, traffic is balanced proportionally to the
+Weight field in each entry.
+ |
+
+
+
+validation
+
+
+
+UpstreamValidation
+
+
+ |
+
+(Optional)
+ UpstreamValidation defines how to verify the backend service’s certificate
+ |
+
+
+
+protocol
+
+
+string
+
+ |
+
+(Optional)
+ Protocol may be used to specify (or override) the protocol used to reach this Service.
+Values may be h2 or h2c. If omitted, protocol-selection falls back on Service annotations.
+ |
+
+
+
+loadBalancerPolicy
+
+
+
+LoadBalancerPolicy
+
+
+ |
+
+(Optional)
+ The policy for load balancing GRPC service requests. Note that the
+Cookie and RequestHash load balancing strategies cannot be used
+here.
+ |
+
+
+
+timeoutPolicy
+
+
+
+TimeoutPolicy
+
+
+ |
+
+(Optional)
+ The timeout policy for requests to the services.
+ |
+
+
+
+protocolVersion
+
+
+
+ExtensionProtocolVersion
+
+
+ |
+
+(Optional)
+ This field sets the version of the GRPC protocol that Envoy uses to
+send requests to the extension service. Since Contour always uses the
+v3 Envoy API, this is currently fixed at “v3”. However, other
+protocol options will be available in future.
+ |
+
+
+ |
+
+
+
+status
+
+
+
+ExtensionServiceStatus
+
+
+ |
+
+ |
+
+
+
+
+
+
+AccessLogJSONFields
+([]string
alias)
+
+(Appears on:
+EnvoyLogging)
+
+
+
+AccessLogLevel
+(string
alias)
+
+(Appears on:
+EnvoyLogging)
+
+
+
+
+
+
+Value |
+Description |
+
+
+"critical" |
+Log only requests that result in an server error (i.e. 500+) response code.
+ |
+
"disabled" |
+Disable the access log.
+ |
+
"error" |
+Log only requests that result in a non-success (i.e. 300+) response code
+ |
+
"info" |
+Log all requests. This is the default.
+ |
+
+
+AccessLogType
+(string
alias)
+
+(Appears on:
+EnvoyLogging)
+
+
+
AccessLogType is the name of a supported access logging mechanism.
+
+
+
+
+Value |
+Description |
+
+
+"envoy" |
+DefaultAccessLogType is the default access log format.
+ |
+
"envoy" |
+Set the Envoy access logging to Envoy’s standard format.
+Can be customized using accessLogFormatString .
+ |
+
"json" |
+Set the Envoy access logging to a JSON format.
+Can be customized using jsonFields .
+ |
+
+
+ClusterDNSFamilyType
+(string
alias)
+
+(Appears on:
+ClusterParameters)
+
+
+
ClusterDNSFamilyType is the Ip family to use for resolving DNS
+names in an Envoy cluster config.
+
+
+
+
+Value |
+Description |
+
+
+"all" |
+DNS lookups will attempt both v4 and v6 queries.
+ |
+
"auto" |
+DNS lookups will do a v6 lookup first, followed by a v4 if that fails.
+ |
+
"v4" |
+DNS lookups will only attempt v4 queries.
+ |
+
"v6" |
+DNS lookups will only attempt v6 queries.
+ |
+
+
+ClusterParameters
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
ClusterParameters holds various configurable cluster values.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+dnsLookupFamily
+
+
+
+ClusterDNSFamilyType
+
+
+ |
+
+(Optional)
+ DNSLookupFamily defines how external names are looked up
+When configured as V4, the DNS resolver will only perform a lookup
+for addresses in the IPv4 family. If V6 is configured, the DNS resolver
+will only perform a lookup for addresses in the IPv6 family.
+If AUTO is configured, the DNS resolver will first perform a lookup
+for addresses in the IPv6 family and fallback to a lookup for addresses
+in the IPv4 family. If ALL is specified, the DNS resolver will perform a lookup for
+both IPv4 and IPv6 families, and return all resolved addresses.
+When this is used, Happy Eyeballs will be enabled for upstream connections.
+Refer to Happy Eyeballs Support for more information.
+Note: This only applies to externalName clusters.
+See https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/cluster.proto.html#envoy-v3-api-enum-config-cluster-v3-cluster-dnslookupfamily
+for more information.
+Values: auto (default), v4 , v6 , all .
+Other values will produce an error.
+ |
+
+
+
+maxRequestsPerConnection
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the maximum requests for upstream connections. If not specified, there is no limit.
+see https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/core/v3/protocol.proto#envoy-v3-api-msg-config-core-v3-httpprotocoloptions
+for more information.
+ |
+
+
+
+per-connection-buffer-limit-bytes
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the soft limit on size of the cluster’s new connection read and write buffers in bytes.
+If unspecified, an implementation defined default is applied (1MiB).
+see https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/cluster/v3/cluster.proto#envoy-v3-api-field-config-cluster-v3-cluster-per-connection-buffer-limit-bytes
+for more information.
+ |
+
+
+
+circuitBreakers
+
+
+
+GlobalCircuitBreakerDefaults
+
+
+ |
+
+(Optional)
+ GlobalCircuitBreakerDefaults specifies default circuit breaker budget across all services.
+If defined, this will be used as the default for all services.
+ |
+
+
+
+upstreamTLS
+
+
+
+EnvoyTLS
+
+
+ |
+
+(Optional)
+ UpstreamTLS contains the TLS policy parameters for upstream connections
+ |
+
+
+
+ContourConfigurationSpec
+
+
+(Appears on:
+ContourConfiguration,
+ContourDeploymentSpec)
+
+
+
ContourConfigurationSpec represents a configuration of a Contour controller.
+It contains most of all the options that can be customized, the
+other remaining options being command line flags.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+xdsServer
+
+
+
+XDSServerConfig
+
+
+ |
+
+(Optional)
+ XDSServer contains parameters for the xDS server.
+ |
+
+
+
+ingress
+
+
+
+IngressConfig
+
+
+ |
+
+(Optional)
+ Ingress contains parameters for ingress options.
+ |
+
+
+
+debug
+
+
+
+DebugConfig
+
+
+ |
+
+(Optional)
+ Debug contains parameters to enable debug logging
+and debug interfaces inside Contour.
+ |
+
+
+
+health
+
+
+
+HealthConfig
+
+
+ |
+
+(Optional)
+ Health defines the endpoints Contour uses to serve health checks.
+Contour’s default is { address: “0.0.0.0”, port: 8000 }.
+ |
+
+
+
+envoy
+
+
+
+EnvoyConfig
+
+
+ |
+
+(Optional)
+ Envoy contains parameters for Envoy as well
+as how to optionally configure a managed Envoy fleet.
+ |
+
+
+
+gateway
+
+
+
+GatewayConfig
+
+
+ |
+
+(Optional)
+ Gateway contains parameters for the gateway-api Gateway that Contour
+is configured to serve traffic.
+ |
+
+
+
+httpproxy
+
+
+
+HTTPProxyConfig
+
+
+ |
+
+(Optional)
+ HTTPProxy defines parameters on HTTPProxy.
+ |
+
+
+
+enableExternalNameService
+
+
+bool
+
+ |
+
+(Optional)
+ EnableExternalNameService allows processing of ExternalNameServices
+Contour’s default is false for security reasons.
+ |
+
+
+
+globalExtAuth
+
+
+
+AuthorizationServer
+
+
+ |
+
+(Optional)
+ GlobalExternalAuthorization allows envoys external authorization filter
+to be enabled for all virtual hosts.
+ |
+
+
+
+rateLimitService
+
+
+
+RateLimitServiceConfig
+
+
+ |
+
+(Optional)
+ RateLimitService optionally holds properties of the Rate Limit Service
+to be used for global rate limiting.
+ |
+
+
+
+policy
+
+
+
+PolicyConfig
+
+
+ |
+
+(Optional)
+ Policy specifies default policy applied if not overridden by the user
+ |
+
+
+
+metrics
+
+
+
+MetricsConfig
+
+
+ |
+
+(Optional)
+ Metrics defines the endpoint Contour uses to serve metrics.
+Contour’s default is { address: “0.0.0.0”, port: 8000 }.
+ |
+
+
+
+tracing
+
+
+
+TracingConfig
+
+
+ |
+
+ Tracing defines properties for exporting trace data to OpenTelemetry.
+ |
+
+
+
+featureFlags
+
+
+
+FeatureFlags
+
+
+ |
+
+ FeatureFlags defines toggle to enable new contour features.
+Available toggles are:
+useEndpointSlices - configures contour to fetch endpoint data
+from k8s endpoint slices. defaults to false and reading endpoint
+data from the k8s endpoints.
+ |
+
+
+
+ContourConfigurationStatus
+
+
+(Appears on:
+ContourConfiguration)
+
+
+
ContourConfigurationStatus defines the observed state of a ContourConfiguration resource.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+conditions
+
+
+
+[]DetailedCondition
+
+
+ |
+
+(Optional)
+ Conditions contains the current status of the Contour resource.
+Contour will update a single condition, Valid , that is in normal-true polarity.
+Contour will not modify any other Conditions set in this block,
+in case some other controller wants to add a Condition.
+ |
+
+
+
+ContourDeploymentSpec
+
+
+(Appears on:
+ContourDeployment)
+
+
+
ContourDeploymentSpec specifies options for how a Contour
+instance should be provisioned.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+contour
+
+
+
+ContourSettings
+
+
+ |
+
+(Optional)
+ Contour specifies deployment-time settings for the Contour
+part of the installation, i.e. the xDS server/control plane
+and associated resources, including things like replica count
+for the Deployment, and node placement constraints for the pods.
+ |
+
+
+
+envoy
+
+
+
+EnvoySettings
+
+
+ |
+
+(Optional)
+ Envoy specifies deployment-time settings for the Envoy
+part of the installation, i.e. the xDS client/data plane
+and associated resources, including things like the workload
+type to use (DaemonSet or Deployment), node placement constraints
+for the pods, and various options for the Envoy service.
+ |
+
+
+
+runtimeSettings
+
+
+
+ContourConfigurationSpec
+
+
+ |
+
+(Optional)
+ RuntimeSettings is a ContourConfiguration spec to be used when
+provisioning a Contour instance that will influence aspects of
+the Contour instance’s runtime behavior.
+ |
+
+
+
+resourceLabels
+
+
+map[string]string
+
+ |
+
+(Optional)
+ ResourceLabels is a set of labels to add to the provisioned Contour resources.
+Deprecated: use Gateway.Spec.Infrastructure.Labels instead. This field will be
+removed in a future release.
+ |
+
+
+
+ContourDeploymentStatus
+
+
+(Appears on:
+ContourDeployment)
+
+
+
ContourDeploymentStatus defines the observed state of a ContourDeployment resource.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+conditions
+
+
+
+[]Kubernetes meta/v1.Condition
+
+
+ |
+
+(Optional)
+ Conditions describe the current conditions of the ContourDeployment resource.
+ |
+
+
+
+ContourSettings
+
+
+(Appears on:
+ContourDeploymentSpec)
+
+
+
ContourSettings contains settings for the Contour part of the installation,
+i.e. the xDS server/control plane and associated resources.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+replicas
+
+
+int32
+
+ |
+
+(Optional)
+ Deprecated: Use DeploymentSettings.Replicas instead.
+Replicas is the desired number of Contour replicas. If if unset,
+defaults to 2.
+if both DeploymentSettings.Replicas and this one is set, use DeploymentSettings.Replicas .
+ |
+
+
+
+nodePlacement
+
+
+
+NodePlacement
+
+
+ |
+
+(Optional)
+ NodePlacement describes node scheduling configuration of Contour pods.
+ |
+
+
+
+kubernetesLogLevel
+
+
+byte
+
+ |
+
+(Optional)
+ KubernetesLogLevel Enable Kubernetes client debug logging with log level. If unset,
+defaults to 0.
+ |
+
+
+
+logLevel
+
+
+
+LogLevel
+
+
+ |
+
+(Optional)
+ LogLevel sets the log level for Contour
+Allowed values are “info”, “debug”.
+ |
+
+
+
+resources
+
+
+
+Kubernetes core/v1.ResourceRequirements
+
+
+ |
+
+(Optional)
+ Compute Resources required by contour container.
+Cannot be updated.
+More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ |
+
+
+
+deployment
+
+
+
+DeploymentSettings
+
+
+ |
+
+(Optional)
+ Deployment describes the settings for running contour as a Deployment .
+ |
+
+
+
+podAnnotations
+
+
+map[string]string
+
+ |
+
+(Optional)
+ PodAnnotations defines annotations to add to the Contour pods.
+the annotations for Prometheus will be appended or overwritten with predefined value.
+ |
+
+
+
+watchNamespaces
+
+
+
+[]Namespace
+
+
+ |
+
+(Optional)
+ WatchNamespaces is an array of namespaces. Setting it will instruct the contour instance
+to only watch this subset of namespaces.
+ |
+
+
+
+disabledFeatures
+
+
+
+[]Feature
+
+
+ |
+
+(Optional)
+ DisabledFeatures defines an array of resources that will be ignored by
+contour reconciler.
+ |
+
+
+
+CustomTag
+
+
+
CustomTag defines custom tags with unique tag name
+to create tags for the active span.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+tagName
+
+
+string
+
+ |
+
+ TagName is the unique name of the custom tag.
+ |
+
+
+
+literal
+
+
+string
+
+ |
+
+(Optional)
+ Literal is a static custom tag value.
+Precisely one of Literal, RequestHeaderName must be set.
+ |
+
+
+
+requestHeaderName
+
+
+string
+
+ |
+
+(Optional)
+ RequestHeaderName indicates which request header
+the label value is obtained from.
+Precisely one of Literal, RequestHeaderName must be set.
+ |
+
+
+
+DaemonSetSettings
+
+
+(Appears on:
+EnvoySettings)
+
+
+
DaemonSetSettings contains settings for DaemonSet resources.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+updateStrategy
+
+
+
+Kubernetes apps/v1.DaemonSetUpdateStrategy
+
+
+ |
+
+(Optional)
+ Strategy describes the deployment strategy to use to replace existing DaemonSet pods with new pods.
+ |
+
+
+
+DebugConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
DebugConfig contains Contour specific troubleshooting options.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+address
+
+
+string
+
+ |
+
+(Optional)
+ Defines the Contour debug address interface.
+Contour’s default is “127.0.0.1”.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+(Optional)
+ Defines the Contour debug address port.
+Contour’s default is 6060.
+ |
+
+
+
+DeploymentSettings
+
+
+(Appears on:
+ContourSettings,
+EnvoySettings)
+
+
+
DeploymentSettings contains settings for Deployment resources.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+replicas
+
+
+int32
+
+ |
+
+ Replicas is the desired number of replicas.
+ |
+
+
+
+strategy
+
+
+
+Kubernetes apps/v1.DeploymentStrategy
+
+
+ |
+
+(Optional)
+ Strategy describes the deployment strategy to use to replace existing pods with new pods.
+ |
+
+
+
+EnvoyConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
EnvoyConfig defines how Envoy is to be Configured from Contour.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+listener
+
+
+
+EnvoyListenerConfig
+
+
+ |
+
+(Optional)
+ Listener hold various configurable Envoy listener values.
+ |
+
+
+
+service
+
+
+
+NamespacedName
+
+
+ |
+
+(Optional)
+ Service holds Envoy service parameters for setting Ingress status.
+Contour’s default is { namespace: “projectcontour”, name: “envoy” }.
+ |
+
+
+
+http
+
+
+
+EnvoyListener
+
+
+ |
+
+(Optional)
+ Defines the HTTP Listener for Envoy.
+Contour’s default is { address: “0.0.0.0”, port: 8080, accessLog: “/dev/stdout” }.
+ |
+
+
+
+https
+
+
+
+EnvoyListener
+
+
+ |
+
+(Optional)
+ Defines the HTTPS Listener for Envoy.
+Contour’s default is { address: “0.0.0.0”, port: 8443, accessLog: “/dev/stdout” }.
+ |
+
+
+
+health
+
+
+
+HealthConfig
+
+
+ |
+
+(Optional)
+ Health defines the endpoint Envoy uses to serve health checks.
+Contour’s default is { address: “0.0.0.0”, port: 8002 }.
+ |
+
+
+
+metrics
+
+
+
+MetricsConfig
+
+
+ |
+
+(Optional)
+ Metrics defines the endpoint Envoy uses to serve metrics.
+Contour’s default is { address: “0.0.0.0”, port: 8002 }.
+ |
+
+
+
+clientCertificate
+
+
+
+NamespacedName
+
+
+ |
+
+(Optional)
+ ClientCertificate defines the namespace/name of the Kubernetes
+secret containing the client certificate and private key
+to be used when establishing TLS connection to upstream
+cluster.
+ |
+
+
+
+logging
+
+
+
+EnvoyLogging
+
+
+ |
+
+(Optional)
+ Logging defines how Envoy’s logs can be configured.
+ |
+
+
+
+defaultHTTPVersions
+
+
+
+[]HTTPVersionType
+
+
+ |
+
+(Optional)
+ DefaultHTTPVersions defines the default set of HTTPS
+versions the proxy should accept. HTTP versions are
+strings of the form “HTTP/xx”. Supported versions are
+“HTTP/1.1” and “HTTP/2”.
+Values: HTTP/1.1 , HTTP/2 (default: both).
+Other values will produce an error.
+ |
+
+
+
+timeouts
+
+
+
+TimeoutParameters
+
+
+ |
+
+(Optional)
+ Timeouts holds various configurable timeouts that can
+be set in the config file.
+ |
+
+
+
+cluster
+
+
+
+ClusterParameters
+
+
+ |
+
+(Optional)
+ Cluster holds various configurable Envoy cluster values that can
+be set in the config file.
+ |
+
+
+
+network
+
+
+
+NetworkParameters
+
+
+ |
+
+(Optional)
+ Network holds various configurable Envoy network values.
+ |
+
+
+
+EnvoyListener
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
EnvoyListener defines parameters for an Envoy Listener.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+address
+
+
+string
+
+ |
+
+(Optional)
+ Defines an Envoy Listener Address.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+(Optional)
+ Defines an Envoy listener Port.
+ |
+
+
+
+accessLog
+
+
+string
+
+ |
+
+(Optional)
+ AccessLog defines where Envoy logs are outputted for this listener.
+ |
+
+
+
+EnvoyListenerConfig
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
EnvoyListenerConfig hold various configurable Envoy listener values.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+useProxyProtocol
+
+
+bool
+
+ |
+
+(Optional)
+ Use PROXY protocol for all listeners.
+Contour’s default is false.
+ |
+
+
+
+disableAllowChunkedLength
+
+
+bool
+
+ |
+
+(Optional)
+ DisableAllowChunkedLength disables the RFC-compliant Envoy behavior to
+strip the “Content-Length” header if “Transfer-Encoding: chunked” is
+also set. This is an emergency off-switch to revert back to Envoy’s
+default behavior in case of failures. Please file an issue if failures
+are encountered.
+See: https://github.com/projectcontour/contour/issues/3221
+Contour’s default is false.
+ |
+
+
+
+disableMergeSlashes
+
+
+bool
+
+ |
+
+(Optional)
+ DisableMergeSlashes disables Envoy’s non-standard merge_slashes path transformation option
+which strips duplicate slashes from request URL paths.
+Contour’s default is false.
+ |
+
+
+
+serverHeaderTransformation
+
+
+
+ServerHeaderTransformationType
+
+
+ |
+
+(Optional)
+ Defines the action to be applied to the Server header on the response path.
+When configured as overwrite, overwrites any Server header with “envoy”.
+When configured as append_if_absent, if a Server header is present, pass it through, otherwise set it to “envoy”.
+When configured as pass_through, pass through the value of the Server header, and do not append a header if none is present.
+Values: overwrite (default), append_if_absent , pass_through
+Other values will produce an error.
+Contour’s default is overwrite.
+ |
+
+
+
+connectionBalancer
+
+
+string
+
+ |
+
+(Optional)
+ ConnectionBalancer. If the value is exact, the listener will use the exact connection balancer
+See https://www.envoyproxy.io/docs/envoy/latest/api-v2/api/v2/listener.proto#envoy-api-msg-listener-connectionbalanceconfig
+for more information.
+Values: (empty string): use the default ConnectionBalancer, exact : use the Exact ConnectionBalancer.
+Other values will produce an error.
+ |
+
+
+
+maxRequestsPerConnection
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the maximum requests for downstream connections. If not specified, there is no limit.
+see https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/core/v3/protocol.proto#envoy-v3-api-msg-config-core-v3-httpprotocoloptions
+for more information.
+ |
+
+
+
+per-connection-buffer-limit-bytes
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the soft limit on size of the listener’s new connection read and write buffers in bytes.
+If unspecified, an implementation defined default is applied (1MiB).
+see https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/listener/v3/listener.proto#envoy-v3-api-field-config-listener-v3-listener-per-connection-buffer-limit-bytes
+for more information.
+ |
+
+
+
+tls
+
+
+
+EnvoyTLS
+
+
+ |
+
+(Optional)
+ TLS holds various configurable Envoy TLS listener values.
+ |
+
+
+
+socketOptions
+
+
+
+SocketOptions
+
+
+ |
+
+(Optional)
+ SocketOptions defines configurable socket options for the listeners.
+Single set of options are applied to all listeners.
+ |
+
+
+
+maxRequestsPerIOCycle
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the limit on number of HTTP requests that Envoy will process from a single
+connection in a single I/O cycle. Requests over this limit are processed in subsequent
+I/O cycles. Can be used as a mitigation for CVE-2023-44487 when abusive traffic is
+detected. Configures the http.max_requests_per_io_cycle Envoy runtime setting. The default
+value when this is not set is no limit.
+ |
+
+
+
+httpMaxConcurrentStreams
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the value for SETTINGS_MAX_CONCURRENT_STREAMS Envoy will advertise in the
+SETTINGS frame in HTTP/2 connections and the limit for concurrent streams allowed
+for a peer on a single HTTP/2 connection. It is recommended to not set this lower
+than 100 but this field can be used to bound resource usage by HTTP/2 connections
+and mitigate attacks like CVE-2023-44487. The default value when this is not set is
+unlimited.
+ |
+
+
+
+maxConnectionsPerListener
+
+
+uint32
+
+ |
+
+(Optional)
+ Defines the limit on number of active connections to a listener. The limit is applied
+per listener. The default value when this is not set is unlimited.
+ |
+
+
+
+EnvoyLogging
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
EnvoyLogging defines how Envoy’s logs can be configured.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+accessLogFormat
+
+
+
+AccessLogType
+
+
+ |
+
+(Optional)
+ AccessLogFormat sets the global access log format.
+Values: envoy (default), json .
+Other values will produce an error.
+ |
+
+
+
+accessLogFormatString
+
+
+string
+
+ |
+
+(Optional)
+ AccessLogFormatString sets the access log format when format is set to envoy .
+When empty, Envoy’s default format is used.
+ |
+
+
+
+accessLogJSONFields
+
+
+
+AccessLogJSONFields
+
+
+ |
+
+(Optional)
+ AccessLogJSONFields sets the fields that JSON logging will
+output when AccessLogFormat is json.
+ |
+
+
+
+accessLogLevel
+
+
+
+AccessLogLevel
+
+
+ |
+
+(Optional)
+ AccessLogLevel sets the verbosity level of the access log.
+Values: info (default, all requests are logged), error (all non-success requests, i.e. 300+ response code, are logged), critical (all 5xx requests are logged) and disabled .
+Other values will produce an error.
+ |
+
+
+
+EnvoySettings
+
+
+(Appears on:
+ContourDeploymentSpec)
+
+
+
EnvoySettings contains settings for the Envoy part of the installation,
+i.e. the xDS client/data plane and associated resources.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+workloadType
+
+
+
+WorkloadType
+
+
+ |
+
+(Optional)
+ WorkloadType is the type of workload to install Envoy
+as. Choices are DaemonSet and Deployment. If unset, defaults
+to DaemonSet.
+ |
+
+
+
+replicas
+
+
+int32
+
+ |
+
+(Optional)
+ Deprecated: Use DeploymentSettings.Replicas instead.
+Replicas is the desired number of Envoy replicas. If WorkloadType
+is not “Deployment”, this field is ignored. Otherwise, if unset,
+defaults to 2.
+if both DeploymentSettings.Replicas and this one is set, use DeploymentSettings.Replicas .
+ |
+
+
+
+networkPublishing
+
+
+
+NetworkPublishing
+
+
+ |
+
+ NetworkPublishing defines how to expose Envoy to a network.
+ |
+
+
+
+nodePlacement
+
+
+
+NodePlacement
+
+
+ |
+
+(Optional)
+ NodePlacement describes node scheduling configuration of Envoy pods.
+ |
+
+
+
+extraVolumes
+
+
+
+[]Kubernetes core/v1.Volume
+
+
+ |
+
+(Optional)
+ ExtraVolumes holds the extra volumes to add.
+ |
+
+
+
+extraVolumeMounts
+
+
+
+[]Kubernetes core/v1.VolumeMount
+
+
+ |
+
+(Optional)
+ ExtraVolumeMounts holds the extra volume mounts to add (normally used with extraVolumes).
+ |
+
+
+
+podAnnotations
+
+
+map[string]string
+
+ |
+
+(Optional)
+ PodAnnotations defines annotations to add to the Envoy pods.
+the annotations for Prometheus will be appended or overwritten with predefined value.
+ |
+
+
+
+resources
+
+
+
+Kubernetes core/v1.ResourceRequirements
+
+
+ |
+
+(Optional)
+ Compute Resources required by envoy container.
+Cannot be updated.
+More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ |
+
+
+
+logLevel
+
+
+
+LogLevel
+
+
+ |
+
+(Optional)
+ LogLevel sets the log level for Envoy.
+Allowed values are “trace”, “debug”, “info”, “warn”, “error”, “critical”, “off”.
+ |
+
+
+
+daemonSet
+
+
+
+DaemonSetSettings
+
+
+ |
+
+(Optional)
+ DaemonSet describes the settings for running envoy as a DaemonSet .
+if WorkloadType is Deployment ,it’s must be nil
+ |
+
+
+
+deployment
+
+
+
+DeploymentSettings
+
+
+ |
+
+(Optional)
+ Deployment describes the settings for running envoy as a Deployment .
+if WorkloadType is DaemonSet ,it’s must be nil
+ |
+
+
+
+baseID
+
+
+int32
+
+ |
+
+(Optional)
+ The base ID to use when allocating shared memory regions.
+if Envoy needs to be run multiple times on the same machine, each running Envoy will need a unique base ID
+so that the shared memory regions do not conflict.
+defaults to 0.
+ |
+
+
+
+overloadMaxHeapSize
+
+
+uint64
+
+ |
+
+(Optional)
+ OverloadMaxHeapSize defines the maximum heap memory of the envoy controlled by the overload manager.
+When the value is greater than 0, the overload manager is enabled,
+and when envoy reaches 95% of the maximum heap size, it performs a shrink heap operation,
+When it reaches 98% of the maximum heap size, Envoy Will stop accepting requests.
+More info: https://projectcontour.io/docs/main/config/overload-manager/
+ |
+
+
+
+EnvoyTLS
+
+
+(Appears on:
+ClusterParameters,
+EnvoyListenerConfig)
+
+
+
EnvoyTLS describes tls parameters for Envoy listneners.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+minimumProtocolVersion
+
+
+string
+
+ |
+
+(Optional)
+ MinimumProtocolVersion is the minimum TLS version this vhost should
+negotiate.
+Values: 1.2 (default), 1.3 .
+Other values will produce an error.
+ |
+
+
+
+maximumProtocolVersion
+
+
+string
+
+ |
+
+(Optional)
+ MaximumProtocolVersion is the maximum TLS version this vhost should
+negotiate.
+Values: 1.2 , 1.3 (default).
+Other values will produce an error.
+ |
+
+
+
+cipherSuites
+
+
+[]string
+
+ |
+
+(Optional)
+ CipherSuites defines the TLS ciphers to be supported by Envoy TLS
+listeners when negotiating TLS 1.2. Ciphers are validated against the
+set that Envoy supports by default. This parameter should only be used
+by advanced users. Note that these will be ignored when TLS 1.3 is in
+use.
+This field is optional; when it is undefined, a Contour-managed ciphersuite list
+will be used, which may be updated to keep it secure.
+Contour’s default list is:
+- “[ECDHE-ECDSA-AES128-GCM-SHA256|ECDHE-ECDSA-CHACHA20-POLY1305]”
+- “[ECDHE-RSA-AES128-GCM-SHA256|ECDHE-RSA-CHACHA20-POLY1305]”
+- “ECDHE-ECDSA-AES256-GCM-SHA384”
+- “ECDHE-RSA-AES256-GCM-SHA384”
+Ciphers provided are validated against the following list:
+- “[ECDHE-ECDSA-AES128-GCM-SHA256|ECDHE-ECDSA-CHACHA20-POLY1305]”
+- “[ECDHE-RSA-AES128-GCM-SHA256|ECDHE-RSA-CHACHA20-POLY1305]”
+- “ECDHE-ECDSA-AES128-GCM-SHA256”
+- “ECDHE-RSA-AES128-GCM-SHA256”
+- “ECDHE-ECDSA-AES128-SHA”
+- “ECDHE-RSA-AES128-SHA”
+- “AES128-GCM-SHA256”
+- “AES128-SHA”
+- “ECDHE-ECDSA-AES256-GCM-SHA384”
+- “ECDHE-RSA-AES256-GCM-SHA384”
+- “ECDHE-ECDSA-AES256-SHA”
+- “ECDHE-RSA-AES256-SHA”
+- “AES256-GCM-SHA384”
+- “AES256-SHA”
+Contour recommends leaving this undefined unless you are sure you must.
+See: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/transport_sockets/tls/v3/common.proto#extensions-transport-sockets-tls-v3-tlsparameters
+Note: This list is a superset of what is valid for stock Envoy builds and those using BoringSSL FIPS.
+ |
+
+
+
+ExtensionProtocolVersion
+(string
alias)
+
+(Appears on:
+ExtensionServiceSpec)
+
+
+
ExtensionProtocolVersion is the version of the GRPC protocol used
+to access extension services. The only version currently supported
+is “v3”.
+
+
+
+
+Value |
+Description |
+
+
+"v2" |
+SupportProtocolVersion2 requests the “v2” support protocol version.
+Deprecated: this protocol version is no longer supported and the
+constant is retained for backwards compatibility only.
+ |
+
"v3" |
+SupportProtocolVersion3 requests the “v3” support protocol version.
+ |
+
+
+ExtensionServiceSpec
+
+
+(Appears on:
+ExtensionService)
+
+
+
ExtensionServiceSpec defines the desired state of an ExtensionService resource.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+services
+
+
+
+[]ExtensionServiceTarget
+
+
+ |
+
+ Services specifies the set of Kubernetes Service resources that
+receive GRPC extension API requests.
+If no weights are specified for any of the entries in
+this array, traffic will be spread evenly across all the
+services.
+Otherwise, traffic is balanced proportionally to the
+Weight field in each entry.
+ |
+
+
+
+validation
+
+
+
+UpstreamValidation
+
+
+ |
+
+(Optional)
+ UpstreamValidation defines how to verify the backend service’s certificate
+ |
+
+
+
+protocol
+
+
+string
+
+ |
+
+(Optional)
+ Protocol may be used to specify (or override) the protocol used to reach this Service.
+Values may be h2 or h2c. If omitted, protocol-selection falls back on Service annotations.
+ |
+
+
+
+loadBalancerPolicy
+
+
+
+LoadBalancerPolicy
+
+
+ |
+
+(Optional)
+ The policy for load balancing GRPC service requests. Note that the
+Cookie and RequestHash load balancing strategies cannot be used
+here.
+ |
+
+
+
+timeoutPolicy
+
+
+
+TimeoutPolicy
+
+
+ |
+
+(Optional)
+ The timeout policy for requests to the services.
+ |
+
+
+
+protocolVersion
+
+
+
+ExtensionProtocolVersion
+
+
+ |
+
+(Optional)
+ This field sets the version of the GRPC protocol that Envoy uses to
+send requests to the extension service. Since Contour always uses the
+v3 Envoy API, this is currently fixed at “v3”. However, other
+protocol options will be available in future.
+ |
+
+
+
+ExtensionServiceStatus
+
+
+(Appears on:
+ExtensionService)
+
+
+
ExtensionServiceStatus defines the observed state of an
+ExtensionService resource.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+conditions
+
+
+
+[]DetailedCondition
+
+
+ |
+
+(Optional)
+ Conditions contains the current status of the ExtensionService resource.
+Contour will update a single condition, Valid , that is in normal-true polarity.
+Contour will not modify any other Conditions set in this block,
+in case some other controller wants to add a Condition.
+ |
+
+
+
+ExtensionServiceTarget
+
+
+(Appears on:
+ExtensionServiceSpec)
+
+
+
ExtensionServiceTarget defines an Kubernetes Service to target with
+extension service traffic.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ Name is the name of Kubernetes service that will accept service
+traffic.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+ Port (defined as Integer) to proxy traffic to since a service can have multiple defined.
+ |
+
+
+
+weight
+
+
+uint32
+
+ |
+
+(Optional)
+ Weight defines proportion of traffic to balance to the Kubernetes Service.
+ |
+
+
+
+FeatureFlags
+([]string
alias)
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
FeatureFlags defines the set of feature flags
+to toggle new contour features.
+
+GatewayConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
GatewayConfig holds the config for Gateway API controllers.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+controllerName
+
+
+string
+
+ |
+
+(Optional)
+ ControllerName is used to determine whether Contour should reconcile a
+GatewayClass. The string takes the form of “projectcontour.io//contour”.
+If unset, the gatewayclass controller will not be started.
+Exactly one of ControllerName or GatewayRef must be set.
+Deprecated: users should use GatewayRef, or the Gateway provisioner,
+in place of this field. This field will be removed in a future release.
+ |
+
+
+
+gatewayRef
+
+
+
+NamespacedName
+
+
+ |
+
+(Optional)
+ GatewayRef defines a specific Gateway that this Contour
+instance corresponds to. If set, Contour will reconcile
+only this gateway, and will not reconcile any gateway
+classes.
+Exactly one of ControllerName or GatewayRef must be set.
+ |
+
+
+
+GlobalCircuitBreakerDefaults
+
+
+(Appears on:
+ClusterParameters)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+maxConnections
+
+
+uint32
+
+ |
+
+(Optional)
+ The maximum number of connections that a single Envoy instance allows to the Kubernetes Service; defaults to 1024.
+ |
+
+
+
+maxPendingRequests
+
+
+uint32
+
+ |
+
+(Optional)
+ The maximum number of pending requests that a single Envoy instance allows to the Kubernetes Service; defaults to 1024.
+ |
+
+
+
+maxRequests
+
+
+uint32
+
+ |
+
+(Optional)
+ The maximum parallel requests a single Envoy instance allows to the Kubernetes Service; defaults to 1024
+ |
+
+
+
+maxRetries
+
+
+uint32
+
+ |
+
+(Optional)
+ The maximum number of parallel retries a single Envoy instance allows to the Kubernetes Service; defaults to 3.
+ |
+
+
+
+HTTPProxyConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
HTTPProxyConfig defines parameters on HTTPProxy.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+disablePermitInsecure
+
+
+bool
+
+ |
+
+(Optional)
+ DisablePermitInsecure disables the use of the
+permitInsecure field in HTTPProxy.
+Contour’s default is false.
+ |
+
+
+
+rootNamespaces
+
+
+[]string
+
+ |
+
+(Optional)
+ Restrict Contour to searching these namespaces for root ingress routes.
+ |
+
+
+
+fallbackCertificate
+
+
+
+NamespacedName
+
+
+ |
+
+(Optional)
+ FallbackCertificate defines the namespace/name of the Kubernetes secret to
+use as fallback when a non-SNI request is received.
+ |
+
+
+
+HTTPVersionType
+(string
alias)
+
+(Appears on:
+EnvoyConfig)
+
+
+
HTTPVersionType is the name of a supported HTTP version.
+
+
+
+
+Value |
+Description |
+
+
+"HTTP/1.1" |
+HTTPVersion1 is the name of the HTTP/1.1 version.
+ |
+
"HTTP/2" |
+HTTPVersion2 is the name of the HTTP/2 version.
+ |
+
+
+
+
+(Appears on:
+PolicyConfig)
+
+
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+set
+
+
+map[string]string
+
+ |
+
+(Optional)
+ |
+
+
+
+remove
+
+
+[]string
+
+ |
+
+(Optional)
+ |
+
+
+
+HealthConfig
+
+
+(Appears on:
+ContourConfigurationSpec,
+EnvoyConfig)
+
+
+
HealthConfig defines the endpoints to enable health checks.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+address
+
+
+string
+
+ |
+
+(Optional)
+ Defines the health address interface.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+(Optional)
+ Defines the health port.
+ |
+
+
+
+IngressConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
IngressConfig defines ingress specific config items.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+classNames
+
+
+[]string
+
+ |
+
+(Optional)
+ Ingress Class Names Contour should use.
+ |
+
+
+
+statusAddress
+
+
+string
+
+ |
+
+(Optional)
+ Address to set in Ingress object status.
+ |
+
+
+
+LogLevel
+(string
alias)
+
+(Appears on:
+ContourSettings,
+EnvoySettings)
+
+
+
LogLevel is the logging levels available.
+
+
+
+
+Value |
+Description |
+
+
+"critical" |
+CriticalLog sets the log level for Envoy to critical .
+ |
+
"debug" |
+DebugLog sets the log level for Contour/Envoy to debug .
+ |
+
"error" |
+ErrorLog sets the log level for Envoy to error .
+ |
+
"info" |
+InfoLog sets the log level for Contour/Envoy to info .
+ |
+
"off" |
+OffLog disable logging for Envoy.
+ |
+
"trace" |
+TraceLog sets the log level for Envoy to trace .
+ |
+
"warn" |
+WarnLog sets the log level for Envoy to warn .
+ |
+
+
+MetricsConfig
+
+
+(Appears on:
+ContourConfigurationSpec,
+EnvoyConfig)
+
+
+
MetricsConfig defines the metrics endpoint.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+address
+
+
+string
+
+ |
+
+(Optional)
+ Defines the metrics address interface.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+(Optional)
+ Defines the metrics port.
+ |
+
+
+
+tls
+
+
+
+MetricsTLS
+
+
+ |
+
+(Optional)
+ TLS holds TLS file config details.
+Metrics and health endpoints cannot have same port number when metrics is served over HTTPS.
+ |
+
+
+
+MetricsTLS
+
+
+(Appears on:
+MetricsConfig)
+
+
+
TLS holds TLS file config details.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+caFile
+
+
+string
+
+ |
+
+(Optional)
+ CA filename.
+ |
+
+
+
+certFile
+
+
+string
+
+ |
+
+(Optional)
+ Client certificate filename.
+ |
+
+
+
+keyFile
+
+
+string
+
+ |
+
+(Optional)
+ Client key filename.
+ |
+
+
+
+NamespacedName
+
+
+(Appears on:
+EnvoyConfig,
+GatewayConfig,
+HTTPProxyConfig,
+RateLimitServiceConfig,
+TracingConfig)
+
+
+
NamespacedName defines the namespace/name of the Kubernetes resource referred from the config file.
+Used for Contour config YAML file parsing, otherwise we could use K8s types.NamespacedName.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+name
+
+
+string
+
+ |
+
+ |
+
+
+
+namespace
+
+
+string
+
+ |
+
+ |
+
+
+
+NetworkParameters
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
NetworkParameters hold various configurable network values.
+
+
+NetworkPublishing
+
+
+(Appears on:
+EnvoySettings)
+
+
+
NetworkPublishing defines the schema for publishing to a network.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+type
+
+
+
+NetworkPublishingType
+
+
+ |
+
+(Optional)
+ NetworkPublishingType is the type of publishing strategy to use. Valid values are:
+
+In this configuration, network endpoints for Envoy use container networking.
+A Kubernetes LoadBalancer Service is created to publish Envoy network
+endpoints.
+See: https://kubernetes.io/docs/concepts/services-networking/service/#loadbalancer
+
+Publishes Envoy network endpoints using a Kubernetes NodePort Service.
+In this configuration, Envoy network endpoints use container networking. A Kubernetes
+NodePort Service is created to publish the network endpoints.
+See: https://kubernetes.io/docs/concepts/services-networking/service/#nodeport
+NOTE:
+When provisioning an Envoy NodePortService , use Gateway Listeners’ port numbers to populate
+the Service’s node port values, there’s no way to auto-allocate them.
+See: https://github.com/projectcontour/contour/issues/4499
+
+Publishes Envoy network endpoints using a Kubernetes ClusterIP Service.
+In this configuration, Envoy network endpoints use container networking. A Kubernetes
+ClusterIP Service is created to publish the network endpoints.
+See: https://kubernetes.io/docs/concepts/services-networking/service/#publishing-services-service-types
+If unset, defaults to LoadBalancerService.
+ |
+
+
+
+externalTrafficPolicy
+
+
+
+Kubernetes core/v1.ServiceExternalTrafficPolicy
+
+
+ |
+
+(Optional)
+ ExternalTrafficPolicy describes how nodes distribute service traffic they
+receive on one of the Service’s “externally-facing” addresses (NodePorts, ExternalIPs,
+and LoadBalancer IPs).
+If unset, defaults to “Local”.
+ |
+
+
+
+ipFamilyPolicy
+
+
+
+Kubernetes core/v1.IPFamilyPolicy
+
+
+ |
+
+(Optional)
+ IPFamilyPolicy represents the dual-stack-ness requested or required by
+this Service. If there is no value provided, then this field will be set
+to SingleStack. Services can be “SingleStack” (a single IP family),
+“PreferDualStack” (two IP families on dual-stack configured clusters or
+a single IP family on single-stack clusters), or “RequireDualStack”
+(two IP families on dual-stack configured clusters, otherwise fail).
+ |
+
+
+
+serviceAnnotations
+
+
+map[string]string
+
+ |
+
+(Optional)
+ ServiceAnnotations is the annotations to add to
+the provisioned Envoy service.
+ |
+
+
+
+NetworkPublishingType
+(string
alias)
+
+(Appears on:
+NetworkPublishing)
+
+
+
NetworkPublishingType is a way to publish network endpoints.
+
+
+
+
+Value |
+Description |
+
+
+"ClusterIPService" |
+ClusterIPServicePublishingType publishes a network endpoint using a Kubernetes
+ClusterIP Service.
+ |
+
"LoadBalancerService" |
+LoadBalancerServicePublishingType publishes a network endpoint using a Kubernetes
+LoadBalancer Service.
+ |
+
"NodePortService" |
+NodePortServicePublishingType publishes a network endpoint using a Kubernetes
+NodePort Service.
+ |
+
+
+NodePlacement
+
+
+(Appears on:
+ContourSettings,
+EnvoySettings)
+
+
+
NodePlacement describes node scheduling configuration for pods.
+If nodeSelector and tolerations are specified, the scheduler will use both to
+determine where to place the pod(s).
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+nodeSelector
+
+
+map[string]string
+
+ |
+
+(Optional)
+ NodeSelector is the simplest recommended form of node selection constraint
+and specifies a map of key-value pairs. For the pod to be eligible
+to run on a node, the node must have each of the indicated key-value pairs
+as labels (it can have additional labels as well).
+If unset, the pod(s) will be scheduled to any available node.
+ |
+
+
+
+tolerations
+
+
+
+[]Kubernetes core/v1.Toleration
+
+
+ |
+
+(Optional)
+ Tolerations work with taints to ensure that pods are not scheduled
+onto inappropriate nodes. One or more taints are applied to a node; this
+marks that the node should not accept any pods that do not tolerate the
+taints.
+The default is an empty list.
+See https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/
+for additional details.
+ |
+
+
+
+PolicyConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
PolicyConfig holds default policy used if not explicitly set by the user
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+requestHeaders
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ RequestHeadersPolicy defines the request headers set/removed on all routes
+ |
+
+
+
+responseHeaders
+
+
+
+HeadersPolicy
+
+
+ |
+
+(Optional)
+ ResponseHeadersPolicy defines the response headers set/removed on all routes
+ |
+
+
+
+applyToIngress
+
+
+bool
+
+ |
+
+(Optional)
+ ApplyToIngress determines if the Policies will apply to ingress objects
+Contour’s default is false.
+ |
+
+
+
+RateLimitServiceConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
RateLimitServiceConfig defines properties of a global Rate Limit Service.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+extensionService
+
+
+
+NamespacedName
+
+
+ |
+
+ ExtensionService identifies the extension service defining the RLS.
+ |
+
+
+
+domain
+
+
+string
+
+ |
+
+(Optional)
+ Domain is passed to the Rate Limit Service.
+ |
+
+
+
+failOpen
+
+
+bool
+
+ |
+
+(Optional)
+ FailOpen defines whether to allow requests to proceed when the
+Rate Limit Service fails to respond with a valid rate limit
+decision within the timeout defined on the extension service.
+ |
+
+
+
+enableXRateLimitHeaders
+
+
+bool
+
+ |
+
+(Optional)
+ EnableXRateLimitHeaders defines whether to include the X-RateLimit
+headers X-RateLimit-Limit, X-RateLimit-Remaining, and X-RateLimit-Reset
+(as defined by the IETF Internet-Draft linked below), on responses
+to clients when the Rate Limit Service is consulted for a request.
+ref. https://tools.ietf.org/id/draft-polli-ratelimit-headers-03.html
+ |
+
+
+
+enableResourceExhaustedCode
+
+
+bool
+
+ |
+
+(Optional)
+ EnableResourceExhaustedCode enables translating error code 429 to
+grpc code RESOURCE_EXHAUSTED. When disabled it’s translated to UNAVAILABLE
+ |
+
+
+
+defaultGlobalRateLimitPolicy
+
+
+
+GlobalRateLimitPolicy
+
+
+ |
+
+(Optional)
+ DefaultGlobalRateLimitPolicy allows setting a default global rate limit policy for every HTTPProxy.
+HTTPProxy can overwrite this configuration.
+ |
+
+
+
+
+
+(Appears on:
+EnvoyListenerConfig)
+
+
+
ServerHeaderTransformation defines the action to be applied to the Server header on the response path
+
+
+
+
+Value |
+Description |
+
+
+"append_if_absent" |
+If no Server header is present, set it to “envoy”.
+If a Server header is present, pass it through.
+ |
+
"overwrite" |
+Overwrite any Server header with “envoy”.
+This is the default value.
+ |
+
"pass_through" |
+Pass through the value of the Server header, and do not append a header
+if none is present.
+ |
+
+
+SocketOptions
+
+
+(Appears on:
+EnvoyListenerConfig)
+
+
+
SocketOptions defines configurable socket options for Envoy listeners.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+tos
+
+
+int32
+
+ |
+
+(Optional)
+ Defines the value for IPv4 TOS field (including 6 bit DSCP field) for IP packets originating from Envoy listeners.
+Single value is applied to all listeners.
+If listeners are bound to IPv6-only addresses, setting this option will cause an error.
+ |
+
+
+
+trafficClass
+
+
+int32
+
+ |
+
+(Optional)
+ Defines the value for IPv6 Traffic Class field (including 6 bit DSCP field) for IP packets originating from the Envoy listeners.
+Single value is applied to all listeners.
+If listeners are bound to IPv4-only addresses, setting this option will cause an error.
+ |
+
+
+
+TLS
+
+
+(Appears on:
+XDSServerConfig)
+
+
+
TLS holds TLS file config details.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+caFile
+
+
+string
+
+ |
+
+(Optional)
+ CA filename.
+ |
+
+
+
+certFile
+
+
+string
+
+ |
+
+(Optional)
+ Client certificate filename.
+ |
+
+
+
+keyFile
+
+
+string
+
+ |
+
+(Optional)
+ Client key filename.
+ |
+
+
+
+insecure
+
+
+bool
+
+ |
+
+(Optional)
+ Allow serving the xDS gRPC API without TLS.
+ |
+
+
+
+TimeoutParameters
+
+
+(Appears on:
+EnvoyConfig)
+
+
+
TimeoutParameters holds various configurable proxy timeout values.
+
+
+TracingConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
TracingConfig defines properties for exporting trace data to OpenTelemetry.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+includePodDetail
+
+
+bool
+
+ |
+
+(Optional)
+ IncludePodDetail defines a flag.
+If it is true, contour will add the pod name and namespace to the span of the trace.
+the default is true.
+Note: The Envoy pods MUST have the HOSTNAME and CONTOUR_NAMESPACE environment variables set for this to work properly.
+ |
+
+
+
+serviceName
+
+
+string
+
+ |
+
+ ServiceName defines the name for the service.
+contour’s default is contour.
+ |
+
+
+
+overallSampling
+
+
+string
+
+ |
+
+(Optional)
+ OverallSampling defines the sampling rate of trace data.
+contour’s default is 100.
+ |
+
+
+
+maxPathTagLength
+
+
+uint32
+
+ |
+
+(Optional)
+ MaxPathTagLength defines maximum length of the request path
+to extract and include in the HttpUrl tag.
+contour’s default is 256.
+ |
+
+
+
+customTags
+
+
+
+[]*github.com/projectcontour/contour/apis/projectcontour/v1alpha1.CustomTag
+
+
+ |
+
+(Optional)
+ CustomTags defines a list of custom tags with unique tag name.
+ |
+
+
+
+extensionService
+
+
+
+NamespacedName
+
+
+ |
+
+ ExtensionService identifies the extension service defining the otel-collector.
+ |
+
+
+
+WorkloadType
+(string
alias)
+
+(Appears on:
+EnvoySettings)
+
+
+
WorkloadType is the type of Kubernetes workload to use for a component.
+
+XDSServerConfig
+
+
+(Appears on:
+ContourConfigurationSpec)
+
+
+
XDSServerConfig holds the config for the Contour xDS server.
+
+
+
+
+Field |
+Description |
+
+
+
+
+
+type
+
+
+
+XDSServerType
+
+
+ |
+
+(Optional)
+ Defines the XDSServer to use for contour serve .
+Values: contour (default), envoy .
+Other values will produce an error.
+ |
+
+
+
+address
+
+
+string
+
+ |
+
+(Optional)
+ Defines the xDS gRPC API address which Contour will serve.
+Contour’s default is “0.0.0.0”.
+ |
+
+
+
+port
+
+
+int
+
+ |
+
+(Optional)
+ Defines the xDS gRPC API port which Contour will serve.
+Contour’s default is 8001.
+ |
+
+
+
+tls
+
+
+
+TLS
+
+
+ |
+
+(Optional)
+ TLS holds TLS file config details.
+Contour’s default is { caFile: “/certs/ca.crt”, certFile: “/certs/tls.cert”, keyFile: “/certs/tls.key”, insecure: false }.
+ |
+
+
+
+XDSServerType
+(string
alias)
+
+(Appears on:
+XDSServerConfig)
+
+
+
XDSServerType is the type of xDS server implementation.
+
+
+
+
+Value |
+Description |
+
+
+"contour" |
+Use Contour’s xDS server.
+ |
+
"envoy" |
+Use the upstream go-control-plane -based xDS server.
+ |
+
+
+
+
+Generated with gen-crd-api-reference-docs
.
+
diff --git a/site/content/docs/1.28/config/api.md b/site/content/docs/1.28/config/api.md
new file mode 100644
index 00000000000..99809537d11
--- /dev/null
+++ b/site/content/docs/1.28/config/api.md
@@ -0,0 +1,3 @@
+# Contour API Reference
+
+{{% include-html api-reference.html %}}
diff --git a/site/content/docs/1.28/config/client-authorization.md b/site/content/docs/1.28/config/client-authorization.md
new file mode 100644
index 00000000000..4db2b932eff
--- /dev/null
+++ b/site/content/docs/1.28/config/client-authorization.md
@@ -0,0 +1,123 @@
+# Client Authorization
+
+Contour supports integrating external servers to authorize client requests.
+
+Envoy implements external authorization in the [ext_authz][1] filter.
+This filter intercepts client requests and holds them while it sends a check
+request to an external server.
+The filter uses the check result to either allow the request to proceed, or to
+deny or redirect the request.
+
+The diagram below shows the sequence of requests involved in the successful
+authorization of a HTTP request:
+
+
+
+
+
+The [external authorization][7] guides demonstrates how to deploy HTTP basic
+authentication using Contour and [contour-authserver](https://github.com/projectcontour/contour-authserver).
+
+## Extension Services
+
+The starting point for external authorization in Contour is the
+[ExtensionService][2] API.
+This API creates a cluster which Envoy can use to send requests to an external server.
+In principle, the Envoy cluster can be used for any purpose, but in this
+document we are concerned only with how to use it as an authorization service.
+
+An authorization service is a gRPC service that implements the Envoy [CheckRequest][3] protocol.
+Note that Contour requires the extension to implement the "v3" version of the protocol.
+Contour is compatible with any authorization server that implements this protocol.
+
+The primary field of interest in the `ExtensionService` CRD is the
+`.spec.services` field.
+This field lists the Kubernetes Services that will receive the check requests.
+The `.spec.services[].name` field contains the name of the Service, which must
+exist in the same namespace as the `ExtensionService` object.
+The `ExtensionService` object must exist in the same namespace as the
+Services they target to ensure that both objects are under the same
+administrative control.
+
+### Load Balancing for Extension Services
+
+An `ExtensionService` can be configured to send traffic to multiple Kubernetes Services.
+In this case, requests are divided proportionally across the Services according
+to the weight in the `.spec.services[].weight` field.
+The service weight can be used to flexibly shift traffic between Services for
+reasons like implementing blue-green deployments.
+The `.spec.loadBalancerPolicy` field configures how Envoy will load balance
+requests to the endpoints within each Service.
+
+### TLS Validation for Extension Services
+
+Since authorizing a client request may involve passing sensitive credentials
+from a HTTP request to the authorization service, the connection to the
+authorization server should be as secure as possible.
+Contour defaults the `.spec.protocol` field to "h2", which configures
+Envoy to use HTTP/2 over TLS for the authorization service connection.
+
+The [.spec.validation][4] field configures how Envoy should verify the TLS
+identity of the authorization server.
+This is a critical protection against accidentally sending credentials to an
+imposter service and should be enabled for all production deployments.
+The `.spec.validation` field should specify the expected server name
+from the authorization server's TLS certificate, and the trusted CA bundle
+that can be used to validate the TLS chain of trust.
+
+## Authorizing Virtual Hosts
+
+The [.spec.virtualhost.authorization][5] field in the Contour `HTTPProxy`
+API connects a virtual host to an authorization server that is bound by an
+`ExtensionService` object.
+Each virtual host can use a different `ExtensionService`, but only one
+`ExtensionService` can be used by a single virtual host.
+Authorization servers can only be attached to `HTTPProxy` objects that have TLS
+termination enabled.
+
+### Migrating from Application Authorization
+
+When applications perform their own authorization, migrating to centralized
+authorization may need some planning.
+The `.spec.virtualhost.authorization.failOpen` field controls how client
+requests should be handled when the authorization server fails.
+During a migration process, this can be set to `true`, so that if the
+authorization server becomes unavailable, clients can gracefully fall back to
+the existing application authorization mechanism.
+
+### Scoping Authorization Policy Settings
+
+It is common for services to contain some HTTP request paths that require
+authorization and some that do not.
+The HTTPProxy [authorization policy][6] allows authorization to be
+disabled for both an entire virtual host and for specific routes.
+
+The initial authorization policy is set on the HTTPProxy virtual host
+in the `.spec.virtualhost.authorization.authPolicy` field.
+This configures whether authorization is enabled, and the default authorization policy context.
+If authorization is disabled on the virtual host, it is also disabled by
+default on all the routes for that virtual host that do not specify an authorization policy.
+However, a route can configure its own authorization policy (in the
+`.spec.routes[].authPolicy` field) that can configure whether authorization
+is enabled, irrespective of the virtual host setting.
+
+The authorization policy context is a way to configure a set of key/value
+pairs that will be sent to the authorization server with each request check
+request.
+The keys and values that should be specified here depend on which authorization
+server has been configured.
+This facility is intended for configuring authorization-specific information, such as
+the basic authentication realm, or OIDC parameters.
+
+The initial context map can be set on the virtual host.
+This sets the context keys that will be sent on every check request.
+A route can overwrite the value for a context key by setting it in the
+context field of authorization policy for the route.
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/ext_authz_filter
+[2]: api/#projectcontour.io/v1alpha1.ExtensionService
+[3]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/service/auth/v3/external_auth.proto
+[4]: api/#projectcontour.io/v1.UpstreamValidation
+[5]: api/#projectcontour.io/v1.AuthorizationServer
+[6]: api/#projectcontour.io/v1.AuthorizationPolicy
+[7]: guides/external-authorization.md
diff --git a/site/content/docs/1.28/config/cookie-rewriting.md b/site/content/docs/1.28/config/cookie-rewriting.md
new file mode 100644
index 00000000000..480fc34125c
--- /dev/null
+++ b/site/content/docs/1.28/config/cookie-rewriting.md
@@ -0,0 +1,109 @@
+# Cookie Rewriting
+
+Contour now enables users to customize attributes on HTTP `Set-Cookie` response headers.
+Application specific cookies and cookies generated by Contour's ["cookie" load balancing strategy](https://projectcontour.io/docs/v1.19.0/config/request-routing/#session-affinity) can be rewritten either per HTTPProxy `Route` or `Service`.
+Users can choose to rewrite the `Path`, `Domain`, `Secure`, and `SameSite` attributes of the `Set-Cookie` header currently.
+These attributes may be things an application may not be able to accurately set, without prior knowledge of how the application is deployed.
+For example, if Contour is in use to rewrite the path or hostname of a request before it reaches an application backend, the application may not be able to accurately set the `Path` and `Domain` attributes in a `Set-Cookie` response header.
+This feature can be used to apply security settings to ensure browsers treat generated cookies appropriately.
+The `SameSite` and `Secure` attributes are currently not set by Envoy when it generates the `X-Contour-Session-Affinity`, but with this feature, users can customize this cookie further.
+
+## Per-Route Cookie Rewriting
+
+In order to implement separate cookie rewriting policies per-route, we can configure an HTTPProxy as below:
+
+```yaml
+# cookie-rewrite-route.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: cookie-rewrite-route
+spec:
+ virtualhost:
+ fqdn: cookie-rewrite-route.com
+ routes:
+ - conditions:
+ - prefix: /admin
+ services:
+ - name: admin-app
+ port: 80
+ cookieRewritePolicies:
+ - name: X-Admin-Session
+ pathRewrite:
+ value: /admin
+ - conditions:
+ - prefix: /payments
+ services:
+ - name: payment-app
+ port: 80
+ cookieRewritePolicies:
+ - name: X-User-Session
+ pathRewrite:
+ value: /payments
+ sameSite: Lax
+ - name: X-User-Data
+ sameSite: Lax
+```
+
+This HTTPProxy allows us to rewrite the `Path` attribute of the `X-Admin-Session` cookie on the `/admin` route.
+In addition on the `/payments` route we rewrite the `Path` and `SameSite` attributes of the `X-User-Session` cookie and the `SameSite` attribute of the additional `X-User-Data` cookie.
+If the backing services `payment-app` and `admin-app` return the specified cookies in `Set-Cookie` response headers, they will be rewritten with the values specified above.
+
+## Per-Service Cookie Rewriting
+
+Similar to the above, if we have more than one `Service` configured per `Route` but want to customize cookies separately between them we can:
+
+```yaml
+# cookie-rewrite-service.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: cookie-rewrite-service
+spec:
+ virtualhost:
+ fqdn: cookie-rewrite-service.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: backend-1
+ port: 80
+ cookieRewritePolicies:
+ - name: X-User-Data-1
+ domainRewrite:
+ value: cookie-rewrite-service.com
+ - name: backend-2
+ port: 80
+ cookieRewritePolicies:
+ - name: X-User-Data-2
+ domainRewrite:
+ value: cookie-rewrite-service.com
+```
+
+## Rewriting Contour Session Affinity Cookie
+
+As mentioned above, users can use Contour's cookie load balancing strategy to enable session affinity.
+Envoy generates a pretty bare-bones cookie but Contour's cookie rewriting feature can be used to customize this cookie to add security attributes:
+
+```yaml
+# cookie-rewrite-session-affinity.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: cookie-rewrite-session-affinity
+spec:
+ virtualhost:
+ fqdn: cookie-rewrite-session-affinity.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: backend
+ port: 80
+ loadBalancerPolicy:
+ strategy: Cookie
+ cookieRewritePolicies:
+ - name: X-Contour-Session-Affinity
+ sameSite: Strict
+ secure: true
+```
diff --git a/site/content/docs/1.28/config/cors.md b/site/content/docs/1.28/config/cors.md
new file mode 100644
index 00000000000..8f468aeaec7
--- /dev/null
+++ b/site/content/docs/1.28/config/cors.md
@@ -0,0 +1,82 @@
+# CORS
+
+A CORS (Cross-origin resource sharing) policy can be set for a HTTPProxy in order to allow cross-domain requests for trusted sources.
+If a policy is set, it will be applied to all the routes of the virtual host.
+
+Contour allows configuring the headers involved in responses to cross-domain requests.
+These include the `Access-Control-Allow-Origin`, `Access-Control-Allow-Methods`, `Access-Control-Allow-Headers`, `Access-Control-Expose-Headers`, `Access-Control-Max-Age`, `Access-Control-Allow-Private-Network` and `Access-Control-Allow-Credentials` headers in responses.
+
+In this example, cross-domain requests will be allowed for any domain (note the `*` value), with the methods `GET`, `POST`, or `OPTIONS`.
+Headers `Authorization` and `Cache-Control` will be passed to the upstream server and headers `Content-Length` and `Content-Range` will be made available to the cross-origin request client.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: cors-example
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ corsPolicy:
+ allowCredentials: true
+ allowPrivateNetwork: true
+ allowOrigin:
+ - "*" # allows any origin
+ allowMethods:
+ - GET
+ - POST
+ - OPTIONS
+ allowHeaders:
+ - authorization
+ - cache-control
+ exposeHeaders:
+ - Content-Length
+ - Content-Range
+ maxAge: "10m" # preflight requests can be cached for 10 minutes.
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: cors-example
+ port: 80
+```
+
+The `allowOrigin` list may also be configured with exact origin matches or regex patterns.
+In the following example, cross-domain requests must originate from the domain `https://client.example.com` or domains that match the regex `http[s]?:\/\/some-site-[a-z0-9]+\.example\.com` (e.g. request with `Origin` header `https://some-site-abc456.example.com`)
+
+*Note:* Patterns for matching `Origin` headers must be valid regex, simple "globbing" patterns (e.g. `*.foo.com`) will not be accepted or may produce incorrect matches.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: cors-example
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ corsPolicy:
+ allowCredentials: true
+ allowOrigin:
+ - https://client.example.com
+ - http[s]?:\/\/some-site-[a-z0-9]+\.example\.com
+ allowMethods:
+ - GET
+ - POST
+ - OPTIONS
+ allowHeaders:
+ - authorization
+ - cache-control
+ exposeHeaders:
+ - Content-Length
+ - Content-Range
+ maxAge: "10m"
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: cors-example
+ port: 80
+```
+
+`MaxAge` durations are expressed in the Go [duration format](https://godoc.org/time#ParseDuration).
+Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h". Only positive values are allowed and 0 disables the cache requiring a preflight `OPTIONS` check for all cross-origin requests.
diff --git a/site/content/docs/1.28/config/external-service-routing.md b/site/content/docs/1.28/config/external-service-routing.md
new file mode 100644
index 00000000000..7da431dd06b
--- /dev/null
+++ b/site/content/docs/1.28/config/external-service-routing.md
@@ -0,0 +1,47 @@
+# External Service Routing
+
+HTTPProxy supports routing traffic to `ExternalName` service types, but this is disabled by default, as it can lead
+to inadvertent exposure of the Envoy Admin UI, allowing remote shutdown and restart of Envoy.
+Please see [this security advisory](https://github.com/projectcontour/contour/security/advisories/GHSA-5ph6-qq5x-7jwc) for all the details.
+It can also be used to expose services in namespaces a user does not have access to, using an ExternalName of `service.namespace.svc.cluster.local`.
+Please see [this Kubernetes security advisory](https://github.com/kubernetes/kubernetes/issues/103675) for more details.
+
+We do *not* recommend enabling ExternalName Services without a strong use case, and understanding of the security implications.
+
+However, To enable ExternalName processing, you must set the `enableExternalNameService` configuration file setting to `true`.
+This will allow the following configuration to be valid.
+
+## ExternalName Support
+
+Contour looks at the `spec.externalName` field of the service and configures the route to use that DNS name instead of utilizing EDS.
+
+Note that hostnames of `localhost` or some other synonyms will be rejected (because of the aforementioned security issues).
+
+There's nothing specific in the HTTPProxy object that needs to be configured other than referencing a service of type `ExternalName`.
+HTTPProxy supports the `requestHeadersPolicy` field to rewrite the `Host` header after first handling a request and before proxying to an upstream service.
+This field can be used to ensure that the forwarded HTTP request contains the hostname that the external resource is expecting.
+
+_**Note:** The ports are required to be specified._
+
+```yaml
+# httpproxy-externalname.yaml
+apiVersion: v1
+kind: Service
+metadata:
+ labels:
+ run: externaldns
+ name: externaldns
+ namespace: default
+spec:
+ externalName: foo-basic.bar.com
+ ports:
+ - name: http
+ port: 80
+ protocol: TCP
+ targetPort: 80
+ type: ExternalName
+```
+
+To proxy to another resource outside the cluster (e.g. A hosted object store bucket for example), configure that external resource in a service type `externalName`.
+Then define a `requestHeadersPolicy` which replaces the `Host` header with the value of the external name service defined previously.
+Finally, if the upstream service is served over TLS, set the `protocol` field on the service to `tls` or annotate the external name service with: `projectcontour.io/upstream-protocol.tls: 443,https`, assuming your service had a port 443 and name `https`.
diff --git a/site/content/docs/1.28/config/fundamentals.md b/site/content/docs/1.28/config/fundamentals.md
new file mode 100644
index 00000000000..0bdac65f77f
--- /dev/null
+++ b/site/content/docs/1.28/config/fundamentals.md
@@ -0,0 +1,197 @@
+# HTTPProxy Fundamentals
+
+The [Ingress][1] object was added to Kubernetes in version 1.1 to describe properties of a cluster-wide reverse HTTP proxy.
+Since that time, the Ingress API has remained relatively unchanged, and the need to express implementation-specific capabilities has inspired an [explosion of annotations][2].
+
+The goal of the HTTPProxy Custom Resource Definition (CRD) is to expand upon the functionality of the Ingress API to allow for a richer user experience as well addressing the limitations of the latter's use in multi tenant environments.
+
+## Key HTTPProxy Benefits
+
+- Safely supports multi-team Kubernetes clusters, with the ability to limit which Namespaces may configure virtual hosts and TLS credentials.
+- Enables including of routing configuration for a path or domain from another HTTPProxy, possibly in another Namespace.
+- Accepts multiple services within a single route and load balances traffic across them.
+- Natively allows defining service weighting and load balancing strategy without annotations.
+- Validation of HTTPProxy objects at creation time and status reporting for post-creation validity.
+
+## Ingress to HTTPProxy
+
+A minimal Ingress object might look like:
+
+```yaml
+# ingress.yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: basic
+spec:
+ rules:
+ - host: foo-basic.bar.com
+ http:
+ paths:
+ - backend:
+ service:
+ name: s1
+ port:
+ number: 80
+ pathType: Prefix
+```
+
+This Ingress object, named `basic`, will route incoming HTTP traffic with a `Host:` header for `foo-basic.bar.com` to a Service named `s1` on port `80`.
+Implementing similar behavior using an HTTPProxy looks like this:
+
+```yaml
+# httpproxy.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: basic
+spec:
+ virtualhost:
+ fqdn: foo-basic.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+```
+
+**Lines 1-5**: As with all other Kubernetes objects, an HTTPProxy needs apiVersion, kind, and metadata fields.
+
+**Lines 7-8**: The presence of the `virtualhost` field indicates that this is a root HTTPProxy that is the top level entry point for this domain.
+
+
+## Interacting with HTTPProxies
+
+As with all Kubernetes objects, you can use `kubectl` to create, list, describe, edit, and delete HTTPProxy CRDs.
+
+Creating an HTTPProxy:
+
+```bash
+$ kubectl create -f basic.httpproxy.yaml
+httpproxy "basic" created
+```
+
+Listing HTTPProxies:
+
+```bash
+$ kubectl get httpproxy
+NAME AGE
+basic 24s
+```
+
+Describing HTTPProxy:
+
+```bash
+$ kubectl describe httpproxy basic
+Name: basic
+Namespace: default
+Labels:
+API Version: projectcontour.io/v1
+Kind: HTTPProxy
+Metadata:
+ Cluster Name:
+ Creation Timestamp: 2019-07-05T19:26:54Z
+ Resource Version: 19373717
+ Self Link: /apis/projectcontour.io/v1/namespaces/default/httpproxy/basic
+ UID: 6036a9d7-8089-11e8-ab00-f80f4182762e
+Spec:
+ Routes:
+ Conditions:
+ Prefix: /
+ Services:
+ Name: s1
+ Port: 80
+ Virtualhost:
+ Fqdn: foo-basic.bar.com
+Events:
+```
+
+Deleting HTTPProxies:
+
+```bash
+$ kubectl delete httpproxy basic
+httpproxy "basic" deleted
+```
+
+## Status Reporting
+
+There are many misconfigurations that could cause an HTTPProxy or delegation to be invalid.
+Contour will make its best effort to process even partially valid configuration and allow traffic to be served for the valid parts.
+To aid users in resolving any issues, Contour updates a `status` field in all HTTPProxy objects.
+
+If an HTTPProxy object is valid, it will have a status property that looks like this:
+
+```yaml
+status:
+ currentStatus: valid
+ description: valid HTTPProxy
+```
+
+If the HTTPProxy is invalid, the `currentStatus` field will be `invalid` and the `description` field will provide a description of the issue.
+
+As an example, if an HTTPProxy object has specified a negative value for weighting, the HTTPProxy status will be:
+
+```yaml
+status:
+ currentStatus: invalid
+ description: "route '/foo': service 'home': weight must be greater than or equal to zero"
+```
+
+Some examples of invalid configurations that Contour provides statuses for:
+
+- Negative weight provided in the route definition.
+- Invalid port number provided for service.
+- Prefix in parent does not match route in delegated route.
+- Root HTTPProxy created in a namespace other than the allowed root namespaces.
+- A given Route of an HTTPProxy both delegates to another HTTPProxy and has a list of services.
+- Orphaned route.
+- Delegation chain produces a cycle.
+- Root HTTPProxy does not specify fqdn.
+- Multiple prefixes cannot be specified on the same set of route conditions.
+- Multiple header conditions of type "exact match" with the same header key.
+- Contradictory header conditions on a route, e.g. a "contains" and "notcontains" condition for the same header and value.
+
+Invalid configuration is ignored and will be not used in the ingress routing configuration.
+Envoy will respond with an error when HTTP request is received on route with invalid configuration on following cases:
+
+* `502 Bad Gateway` response is sent when HTTPProxy has an include that refers to an HTTPProxy that does not exist.
+* `503 Service Unavailable` response is sent when HTTPProxy refers to a service that does not exist.
+
+### Example
+
+Following example has two routes: the first one is valid, the second one refers to a service that does not exist.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-routes-with-a-missing-service
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: valid-service
+ port: 80
+ - conditions:
+ - prefix: /subpage
+ services:
+ - name: service-that-does-not-exist
+ port: 80
+```
+
+The `HTTPProxy` will have condition `Valid=false` with detailed error message: `Spec.Routes unresolved service reference: service "default/service-that-does-not-exist" not found`.
+Requests received for `http://www.example.com/` will be forwarded to `valid-service` but requests received for `http://www.example.com/subpage` will result in error `503 Service Unavailable` response from Envoy.
+
+## HTTPProxy API Specification
+
+The full HTTPProxy specification is described in detail in the [API documentation][4].
+There are a number of working examples of HTTPProxy objects in the [`examples/example-workload`][3] directory of the Contour Github repository.
+
+ [1]: https://kubernetes.io/docs/concepts/services-networking/ingress/
+ [2]: https://github.com/kubernetes/ingress-nginx/blob/master/docs/user-guide/nginx-configuration/annotations.md
+ [3]: {{< param github_url>}}/tree/{{< param branch >}}/examples/example-workload/httpproxy
+ [4]: api.md
diff --git a/site/content/docs/1.28/config/gateway-api.md b/site/content/docs/1.28/config/gateway-api.md
new file mode 100644
index 00000000000..af67fea4339
--- /dev/null
+++ b/site/content/docs/1.28/config/gateway-api.md
@@ -0,0 +1,221 @@
+# Gateway API
+
+## Introduction
+
+[Gateway API][1] is an open source project managed by the SIG Network community.
+It is a collection of resources that model service networking in Kubernetes.
+These resources - GatewayClass, Gateway, HTTPRoute, TCPRoute, Service, etc - aim to evolve Kubernetes service networking through expressive, extensible, and role-oriented interfaces that are implemented by many vendors and have broad industry support.
+
+Contour implements Gateway API in addition to supporting HTTPProxy and Ingress.
+In particular, Contour aims to support all [core and extended features][2] in Gateway API.
+
+Gateway API has a comprehensive [website and docs][1], so this document focuses primarily on unique aspects of Contour's Gateway API implementation, rather than attempting to reproduce all of the content available on the Gateway API website.
+The reader is suggested to familiarize themselves with the basics of Gateway API before continuing with this doc.
+
+In Contour's Gateway API implementation, a Gateway corresponds 1:1 with a single deployment of Contour + Envoy.
+In other words, each Gateway has its own control plane (Contour) and data plane (Envoy).
+
+The remainder of this document delves into more detail regarding configuration options when using Contour with Gateway API.
+If you are looking for a way to get started with Gateway API and Contour, see the [Gateway API guide][12], a step-by-step tutorial on getting Contour installed with Gateway API and using it to route traffic to a service.
+
+## Enabling Gateway API in Contour
+
+There are two ways to deploy Contour with Gateway API support: **static** provisioning and **dynamic** provisioning.
+
+In **static** provisioning, the platform operator defines a `Gateway` resource, and then manually deploys a Contour instance corresponding to that `Gateway` resource.
+It is up to the platform operator to ensure that all configuration matches between the `Gateway` and the Contour/Envoy resources.
+With static provisioning, Contour can be configured with either a [controller name][8], or a specific gateway (see the [API documentation][7].)
+If configured with a controller name, Contour will process the oldest `GatewayClass`, its oldest `Gateway`, and that `Gateway's` routes, for the given controller name.
+If configured with a specific gateway, Contour will process that `Gateway` and its routes.
+
+**Note:** configuring Contour with a controller name is deprecated and will be removed in a future release. Use a specific gateway reference or dynamic provisioning instead.
+
+In **dynamic** provisioning, the platform operator first deploys Contour's Gateway provisioner. Then, the platform operator defines a `Gateway` resource, and the provisioner automatically deploys a Contour instance that corresponds to the `Gateway's` configuration and will process that `Gateway` and its routes.
+
+Static provisioning makes sense for users who:
+- prefer the traditional model of deploying Contour
+- have only a single Gateway
+- want to use just the standard listener ports (80/443)
+- have highly customized YAML for deploying Contour.
+
+Dynamic provisioning makes sense for users who:
+- have many Gateways
+- want to use additional listener ports
+- prefer a simple declarative API for provisioning Contour instances
+- want a fully conformant Gateway API implementation
+
+### Static Provisioning
+
+To statically provision Contour with Gateway API enabled:
+
+1. Install the [Gateway API experimental channel][3].
+1. Create a GatewayClass, with a controller name of `projectcontour.io/gateway-controller`.
+1. Create a Gateway using the above GatewayClass.
+1. In the Contour config file, add a reference to the above Gateway via `gateway.gatewayRef` (see https://projectcontour.io/docs/1.25/configuration/#gateway-configuration)
+1. Install Contour using the above config file.
+
+Contour provides an example manifest for this at https://projectcontour.io/quickstart/contour-gateway.yaml.
+
+### Dynamic Provisioning
+
+To dynamically provision Contour with Gateway API enabled:
+
+1. Install the [Contour Gateway Provisioner][9], which includes the Gateway API experimental channel.
+1. Create a GatewayClass, with a controller name of `projectcontour.io/gateway-controller`.
+1. Create a Gateway using the above GatewayClass.
+
+The Contour Gateway Provisioner will deploy an instance of Contour in the Gateway's namespace implementing the Gateway spec.
+
+**Note:** Gateway names must be 63 characters or shorter, to avoid issues when generating dependent resources. See [projectcontour/contour#5970][13] and [kubernetes-sigs/gateway-api#2592][14] for more information.
+
+## Gateway Listeners
+
+Each unique Gateway Listener port requires the Envoy service to expose that port, and to map it to an underlying port in the Envoy daemonset/deployment that Envoy is configured to listen on.
+For example, the following Gateway Listener configuration (abridged) requires service ports of 80 and 443, mapped to underlying container ports 8080 and 8443:
+
+```yaml
+listeners:
+- name: http
+ protocol: HTTP
+ port: 80
+- name: https
+ protocol: HTTPS
+ port: 443
+```
+
+In dynamic provisioning, the Contour Gateway Provisioner will continuously ensure that the Envoy service and daemonset/deployment are kept in sync with the Gateway Listener configuration.
+In static provisioning, it is up to the platform operator to keep the Envoy resources in sync with the Gateway Listeners.
+
+To get from the Gateway Listener port to the port that Envoy will be configured to listen on, i.e. the container port:
+- add 8000 to the Listener port number
+- if the result is greater than 65535, subtract 65535
+- if the result is less than or equal to 1023, add 1023.
+
+Note that, in rare corner cases, it's possible to have port conflicts.
+Check the Gateway status to ensure that Listeners have been properly provisioned.
+
+## Routing
+
+Gateway API defines multiple route types.
+Each route type is appropriate for a different type of traffic being proxied to a backend service.
+Contour implements `HTTPRoute`, `TLSRoute`, `GRPCRoute` and `TCPRoute`.
+The details of each of these route types are covered in extensive detail on the Gateway API website; the [route resources overview][11] is a good place to start learning about them.
+
+### Routing with HTTPProxy or Ingress
+
+When Gateway API is enabled in Contour, it's still possible to use HTTPProxy or Ingress to define routes, with some limitations.
+This is useful for users who:
+- are in the process of migrating to Gateway API
+- want to use the Contour Gateway Provisioner for dynamic provisioning, but need the advanced features of HTTPProxy
+
+To use HTTPProxy or Ingress with Gateway API, define a Gateway with the following Listeners:
+
+```yaml
+listeners:
+- name: http
+ protocol: HTTP
+ port: 80
+ allowedRoutes:
+ namespaces:
+ from: All
+- name: https
+ protocol: projectcontour.io/https
+ port: 443
+ allowedRoutes:
+ namespaces:
+ from: All
+```
+
+Note that for the second Listener, a Contour-specific protocol is used, and no TLS details are specified.
+Instead, TLS details continue to be configured on the HTTPProxy or Ingress resource.
+
+This is an area of active development and further work will be done in upcoming releases to better support migrations and mixed modes of operation.
+
+## Contour Gateway Provisioner
+
+### Customizing a GatewayClass
+
+Gateway API [supports attaching parameters to a GatewayClass][5], which can customize the Gateways that are provisioned for that GatewayClass.
+
+Contour defines a CRD called `ContourDeployment`, which can be used as `GatewayClass` parameters.
+
+A simple example of a parameterized Contour GatewayClass that provisions Envoy as a Deployment instead of the default DaemonSet looks like:
+
+```yaml
+kind: GatewayClass
+apiVersion: gateway.networking.k8s.io/v1
+metadata:
+ name: contour-with-envoy-deployment
+spec:
+ controllerName: projectcontour.io/gateway-controller
+ parametersRef:
+ kind: ContourDeployment
+ group: projectcontour.io
+ name: contour-with-envoy-deployment-params
+ namespace: projectcontour
+---
+kind: ContourDeployment
+apiVersion: projectcontour.io/v1alpha1
+metadata:
+ namespace: projectcontour
+ name: contour-with-envoy-deployment-params
+spec:
+ envoy:
+ workloadType: Deployment
+```
+
+All Gateways provisioned using the `contour-with-envoy-deployment` GatewayClass would get an Envoy Deployment.
+
+See [the API documentation][6] for all `ContourDeployment` options.
+
+It's important to note that, per the [GatewayClass spec][10]:
+
+> It is recommended that [GatewayClass] be used as a template for Gateways.
+> This means that a Gateway is based on the state of the GatewayClass at the time it was created and changes to the GatewayClass or associated parameters are not propagated down to existing Gateways.
+> This recommendation is intended to limit the blast radius of changes to GatewayClass or associated parameters.
+> If implementations choose to propagate GatewayClass changes to existing Gateways, that MUST be clearly documented by the implementation.
+
+Contour follows the recommended behavior, meaning changes to a GatewayClass and its parameters are not propagated down to existing Gateways.
+
+### Upgrades
+
+When the Contour Gateway Provisioner is upgraded to a new version, it will upgrade all Gateways it controls (both the control plane and the data plane).
+
+## Disabling Experimental Resources
+
+Some users may want to use Contour with the [Gateway API standard channel][4] instead of the experimental channel, to avoid installing alpha resources into their clusters.
+To do this, Contour must be told to disable informers for the experimental resources.
+In the Contour (control plane) deployment, use the `--disable-feature` flag for `contour serve` to disable informers for the experimental resources:
+
+```yaml
+containers:
+- name: contour
+ image: ghcr.io/projectcontour/contour:
+ command: ["contour"]
+ args:
+ - serve
+ - --incluster
+ - --xds-address=0.0.0.0
+ - --xds-port=8001
+ - --contour-cafile=/certs/ca.crt
+ - --contour-cert-file=/certs/tls.crt
+ - --contour-key-file=/certs/tls.key
+ - --config-path=/config/contour.yaml
+ - --disable-feature=tlsroutes
+ - --disable-feature=grpcroutes
+```
+
+[1]: https://gateway-api.sigs.k8s.io/
+[2]: https://gateway-api.sigs.k8s.io/concepts/conformance/#2-support-levels
+[3]: https://gateway-api.sigs.k8s.io/guides/#install-experimental-channel
+[4]: https://gateway-api.sigs.k8s.io/guides/#install-standard-channel
+[5]: https://gateway-api.sigs.k8s.io/api-types/gatewayclass/#gatewayclass-parameters
+[6]: https://projectcontour.io/docs/main/config/api/#projectcontour.io/v1alpha1.ContourDeployment
+[7]: https://projectcontour.io/docs/main/config/api/#projectcontour.io/v1alpha1.GatewayConfig
+[8]: https://gateway-api.sigs.k8s.io/api-types/gatewayclass/#gatewayclass-controller-selection
+[9]: https://projectcontour.io/quickstart/contour-gateway-provisioner.yaml
+[10]: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1.GatewayClass
+[11]: https://gateway-api.sigs.k8s.io/concepts/api-overview/#route-resources
+[12]: /docs/{{< param version >}}/guides/gateway-api
+[13]: https://github.com/projectcontour/contour/issues/5970
+[14]: https://github.com/kubernetes-sigs/gateway-api/issues/2592
\ No newline at end of file
diff --git a/site/content/docs/1.28/config/health-checks.md b/site/content/docs/1.28/config/health-checks.md
new file mode 100644
index 00000000000..6dd1aac619d
--- /dev/null
+++ b/site/content/docs/1.28/config/health-checks.md
@@ -0,0 +1,160 @@
+# Upstream Health Checks
+
+## HTTP Proxy Health Checking
+
+Active health checking can be configured on a per route basis.
+Contour supports HTTP health checking and can be configured with various settings to tune the behavior.
+
+During HTTP health checking Envoy will send an HTTP request to the upstream Endpoints.
+It expects a 200 response by default if the host is healthy (see `expectedStatuses` below for configuring the "healthy" status codes).
+The upstream host can return 503 if it wants to immediately notify Envoy to no longer forward traffic to it.
+It is important to note that these are health checks which Envoy implements and are separate from any other system such as those that exist in Kubernetes.
+
+```yaml
+# httpproxy-health-checks.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: health-check
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: health.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ healthCheckPolicy:
+ path: /healthy
+ intervalSeconds: 5
+ timeoutSeconds: 2
+ unhealthyThresholdCount: 3
+ healthyThresholdCount: 5
+ services:
+ - name: s1-health
+ port: 80
+ - name: s2-health
+ port: 80
+```
+
+Health check configuration parameters:
+
+- `path`: HTTP endpoint used to perform health checks on upstream service (e.g. `/healthz`). It expects a 200 response if the host is healthy. The upstream host can return 503 if it wants to immediately notify downstream hosts to no longer forward traffic to it.
+- `host`: The value of the host header in the HTTP health check request. If left empty (default value), the name "contour-envoy-healthcheck" will be used.
+- `intervalSeconds`: The interval (seconds) between health checks. Defaults to 5 seconds if not set.
+- `timeoutSeconds`: The time to wait (seconds) for a health check response. If the timeout is reached the health check attempt will be considered a failure. Defaults to 2 seconds if not set.
+- `unhealthyThresholdCount`: The number of unhealthy health checks required before a host is marked unhealthy. Note that for http health checking if a host responds with 503 this threshold is ignored and the host is considered unhealthy immediately. Defaults to 3 if not defined.
+- `healthyThresholdCount`: The number of healthy health checks required before a host is marked healthy. Note that during startup, only a single successful health check is required to mark a host healthy.
+- `expectedStatuses`: An optional list of HTTP status ranges that are considered healthy. Ranges follow half-open semantics, meaning the start is inclusive and the end is exclusive. Statuses must be between 100 (inclusive) and 600 (exclusive).
+
+### Non-default expected statuses
+
+By default, only responses with a 200 status code will be considered healthy.
+The set of response codes considered healthy can be customized by specifying ranges in `expectedStatuses`.
+Ranges follow half-open semantics, meaning the start is inclusive and the end is exclusive.
+Statuses must be between 100 (inclusive) and 600 (exclusive).
+For example:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: health-check
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: health.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ healthCheckPolicy:
+ path: /healthy
+ intervalSeconds: 5
+ timeoutSeconds: 2
+ unhealthyThresholdCount: 3
+ healthyThresholdCount: 5
+ # Status codes 200 and 250-299 will be considered healthy.
+ expectedStatuses:
+ - start: 200
+ end: 201
+ - start: 250
+ end: 300
+ services:
+ - name: s1-health
+ port: 80
+ - name: s2-health
+ port: 80
+```
+
+Note that if `expectedStatuses` is specified, `200` must be explicitly included in one of the specified ranges if it is desired as a healthy status code.
+
+## TCP Proxy Health Checking
+
+Contour also supports TCP health checking and can be configured with various settings to tune the behavior.
+
+During TCP health checking Envoy will send a connect-only health check to the upstream Endpoints.
+It is important to note that these are health checks which Envoy implements and are separate from any
+other system such as those that exist in Kubernetes.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: tcp-health-check
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: health.bar.com
+ tcpproxy:
+ healthCheckPolicy:
+ intervalSeconds: 5
+ timeoutSeconds: 2
+ unhealthyThresholdCount: 3
+ healthyThresholdCount: 5
+ services:
+ - name: s1-health
+ port: 80
+ - name: s2-health
+ port: 80
+```
+
+TCP Health check policy configuration parameters:
+
+- `intervalSeconds`: The interval (seconds) between health checks. Defaults to 5 seconds if not set.
+- `timeoutSeconds`: The time to wait (seconds) for a health check response. If the timeout is reached the health check attempt will be considered a failure. Defaults to 2 seconds if not set.
+- `unhealthyThresholdCount`: The number of unhealthy health checks required before a host is marked unhealthy. Note that for http health checking if a host responds with 503 this threshold is ignored and the host is considered unhealthy immediately. Defaults to 3 if not defined.
+- `healthyThresholdCount`: The number of healthy health checks required before a host is marked healthy. Note that during startup, only a single successful health check is required to mark a host healthy.
+
+## Specify the service health check port
+
+contour supports configuring an optional health check port for services.
+
+By default, the service's health check port is the same as the service's routing port.
+If the service's health check port and routing port are different, you can configure the health check port separately.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: health-check
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: health.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ healthCheckPolicy:
+ path: /healthy
+ intervalSeconds: 5
+ timeoutSeconds: 2
+ unhealthyThresholdCount: 3
+ healthyThresholdCount: 5
+ services:
+ - name: s1-health
+ port: 80
+ healthPort: 8998
+ - name: s2-health
+ port: 80
+```
+
+In this example, envoy will send a health check request to port `8998` of the `s1-health` service and port `80` of the `s2-health` service respectively . If the host is healthy, envoy will forward traffic to the `s1-health` service on port `80` and to the `s2-health` service on port `80`.
diff --git a/site/content/docs/1.28/config/inclusion-delegation.md b/site/content/docs/1.28/config/inclusion-delegation.md
new file mode 100644
index 00000000000..b9364ff1fcd
--- /dev/null
+++ b/site/content/docs/1.28/config/inclusion-delegation.md
@@ -0,0 +1,139 @@
+# HTTPProxy Inclusion
+
+HTTPProxy permits the splitting of a system's configuration into separate HTTPProxy instances using **inclusion**.
+
+Inclusion, as the name implies, allows for one HTTPProxy object to be included in another, optionally with some conditions inherited from the parent.
+Contour reads the inclusion tree and merges the included routes into one big object internally before rendering Envoy config.
+Importantly, the included HTTPProxy objects do not have to be in the same namespace.
+
+Each tree of HTTPProxy starts with a root, the top level object of the configuration for a particular virtual host.
+Each root HTTPProxy defines a `virtualhost` key, which describes properties such as the fully qualified name of the virtual host, TLS configuration, etc.
+
+HTTPProxies included from the root must not contain a virtualhost key.
+Root objects cannot include other roots either transitively or directly.
+This permits the owner of an HTTPProxy root to allow the inclusion of a portion of the route space inside a virtual host, and to allow that route space to be further subdivided with inclusions.
+Because the path is not necessarily used as the only key, the route space can be multi-dimensional.
+
+## Conditions and Inclusion
+
+Like Routes, Inclusion may specify a set of [conditions][1].
+These conditions are added to any conditions on the routes included.
+This process is recursive.
+
+Conditions are sets of individual condition statements, for example `prefix: /blog` is the condition that the matching request's path must start with `/blog`.
+When conditions are combined through inclusion Contour merges the conditions inherited via inclusion with any conditions specified on the route.
+This may result in duplicates, for example two `prefix:` conditions, mix of both `prefix:` and `exact` or `prefix` and `regex` conditions, or two header match conditions with the same name and value.
+To resolve this Contour applies the following logic.
+
+- `prefix:` conditions are concatenated together in the order they were applied from the root object. For example the conditions, `prefix: /api`, `prefix: /v1` becomes a single `prefix: /api/v1` conditions. Note: Multiple prefixes cannot be supplied on a single set of Route conditions.
+- `exact:` conditions are also concatenated just like `prefix:` conditions, but `exact:` conditions are not allowed in include match conditions. If the child httpproxy has `exact:` condition then after concatenation, it becomes a single `exact:` condition. For example, `prefix: /static` and `exact: /main.js` become a single `exact: /static/main.js` condition.
+- `regex:` conditions are also concatenated just like `prefix:` conditions, but `regex:` conditions are not allowed in include match conditions. If the child httpproxy has `regex:` condition then after concatenation, it becomes a single `regex:` condition. For example, `prefix: /static` and `regex: /.*/main.js` become a single `regex: /static/.*/main.js` condition.
+- Proxies with repeated identical `header:` conditions of type "exact match" (the same header keys exactly) are marked as "Invalid" since they create an un-routable configuration.
+
+## Configuring Inclusion
+
+Inclusion is a top-level field in the HTTPProxy [spec][2] element.
+It requires one field, `name`, and has two optional fields:
+
+- `namespace`. This will assume the included HTTPProxy is in the same namespace if it's not specified.
+- a `conditions` block.
+
+## Inclusion Within the Same Namespace
+
+HTTPProxies can include other HTTPProxy objects in the namespace by specifying the name of the object and its namespace in the top-level `includes` block.
+Note that `includes` is a list, and so it must use the YAML list construct.
+
+In this example, the HTTPProxy `include-root` has included the configuration for paths matching `/service2` from the HTTPProxy named `service2` in the same namespace as `include-root` (the `default` namespace).
+It's important to note that `service2` HTTPProxy has not defined a `virtualhost` property as it is NOT a root HTTPProxy.
+
+```yaml
+# httpproxy-inclusion-samenamespace.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: include-root
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: root.bar.com
+ includes:
+ # Includes the /service2 path from service2 in the same namespace
+ - name: service2
+ namespace: default
+ conditions:
+ - prefix: /service2
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: service2
+ namespace: default
+spec:
+ routes:
+ - services: # matches /service2
+ - name: s2
+ port: 80
+ - conditions:
+ - prefix: /blog # matches /service2/blog
+ services:
+ - name: blog
+ port: 80
+```
+
+## Inclusion Across Namespaces
+
+Inclusion can also happen across Namespaces by specifying a `namespace` in the `inclusion`.
+This is a particularly powerful paradigm for enabling multi-team Ingress management.
+
+If the `--watch-namespaces` configuration flag is used, it must define all namespaces that will be referenced by the inclusion.
+
+In this example, the root HTTPProxy has included configuration for paths matching `/blog` to the `blog` HTTPProxy object in the `marketing` namespace.
+
+```yaml
+# httpproxy-inclusion-across-namespaces.yaml
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: namespace-include-root
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: ns-root.bar.com
+ includes:
+ # delegate the subpath, `/blog` to the HTTPProxy object in the marketing namespace with the name `blog`
+ - name: blog
+ namespace: marketing
+ conditions:
+ - prefix: /blog
+ routes:
+ - services:
+ - name: s1
+ port: 80
+
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: blog
+ namespace: marketing
+spec:
+ routes:
+ - services:
+ - name: s2
+ port: 80
+```
+
+## Orphaned HTTPProxy children
+
+It is possible for HTTPProxy objects to exist that have not been delegated to by another HTTPProxy.
+These objects are considered "orphaned" and will be ignored by Contour in determining ingress configuration.
+
+[1]: request-routing#conditions
+[2]: api/#projectcontour.io/v1.HTTPProxySpec
diff --git a/site/content/docs/1.28/config/ingress.md b/site/content/docs/1.28/config/ingress.md
new file mode 100644
index 00000000000..22e65bb0255
--- /dev/null
+++ b/site/content/docs/1.28/config/ingress.md
@@ -0,0 +1,94 @@
+# k8s Ingress Resource Support in Contour
+
+
+
+
+
+This document describes Contour's implementation of specific Ingress resource fields and features.
+As the Ingress specification has evolved between v1beta1 and v1, any differences between versions are highlighted to ensure clarity for Contour users.
+
+**Note: As of Contour version 1.16.0, Contour is not compatible with Kubernetes versions that predate Ingress v1. This means Contour 1.16.0 and above require Kubernetes 1.19 and above. The Ingress v1beta1 resource is still available in Kubernetes 1.19 (but will be removed in 1.22) and the API server will convert such resources to Ingress v1 for Contour to subscribe to.**
+
+## Kubernetes Versions
+
+Contour is [validated against Kubernetes release versions N through N-2][1] (with N being the latest release).
+For Kubernetes version 1.19+, the API server translates any Ingress v1beta1 resources to Ingress v1 and Contour watches Ingress v1 resources.
+
+## IngressClass and IngressClass Name
+
+In order to support differentiating between Ingress controllers or multiple instances of a single Ingress controller, users can create an [IngressClass resource][2] and specify an IngressClass name on a Ingress to reference it.
+The IngressClass resource can be used to provide configuration to an Ingress controller watching resources it governs.
+Contour supports watching an IngressClass resource specified with the `--ingress-class-name` flag to the `contour serve` command.
+Contour does not require an IngressClass resource with the name passed in the aforementioned flag to exist, the name can just be used as an identifier for filtering which Ingress resources Contour reconciles into actual route configuration.
+
+Ingresses may specify an IngressClass name via the original annotation method or via the `ingressClassName` spec field.
+As the `ingressClassName` field has been introduced on Ingress v1beta1, there should be no differences in IngressClass name filtering between the two available versions of the resource.
+Contour uses its configured IngressClass name to filter Ingresses.
+If the `--ingress-class-name` flag is provided, Contour will only accept Ingress resources that exactly match the specified IngressClass name via annotation or spec field, with the value in the annotation taking precedence. (The `--ingress-class-name` value can be a comma-separated list of class names to match against.)
+If the flag is not passed to `contour serve` Contour will accept any Ingress resource that specifies the IngressClass name `contour` in annotation or spec fields or does not specify one at all.
+
+## Default Backend
+
+Contour supports the `defaultBackend` Ingress v1 spec field and equivalent `backend` v1beta1 version of the field.
+See upstream [documentation][3] on this field.
+Any requests that do not match an Ingress rule will be forwarded to this backend.
+As TLS secrets on Ingresses are scoped to specific hosts, this default backend cannot serve TLS as it could match an unbounded set of hosts and configuring a matching set of TLS secrets would not be possible.
+As is the case on Ingress rules, Contour only supports configuring a Service as a backend and does not support any other Kubernetes resource.
+
+## Ingress Rules
+
+See upstream [documentation][4] on Ingress rules.
+
+As with default backends, Contour only supports configuring a Service as a backend and does not support any other Kubernetes resource.
+
+Contour supports [wildcard hostnames][5] as documented by the upstream API as well as precise hostnames.
+Wildcard hostnames are limited to the whole first DNS label of the hostname, e.g. `*.foo.com` is valid but `*foo.com`, `foo*.com`, `foo.*.com` are not.
+`*` is also not a valid hostname.
+Precise hostnames in Ingress or HTTPProxy configuration take higher precedence over wildcards.
+For example, given an Ingress rule with the hostname `*.foo.com` routing to `service-a` and another Ingress rule or HTTPProxy route containing a subdomain (say `bar.foo.com`) routing to `service-b`, requests to `bar.foo.com` will be routed to `service-b`.
+The Ingress admission controller validation ensures valid hostnames are present when creating an Ingress resource.
+
+Contour supports all of the various [path matching][6] types described by the Ingress spec.
+Prior to Contour 1.14.0, path match types were ignored and path matching was performed with a Contour specific implementation.
+Paths specified with any regex meta-characters (any of `^+*[]%`) were implemented as regex matches.
+Any other paths were programmed in Envoy as "string prefix" matches.
+This behavior is preserved in the `ImplementationSpecific` match type in Contour 1.14.0+ to ensure backwards compatibility.
+`Exact` path matches will now result in matching requests to the given path exactly.
+The `Prefix` patch match type will now result in matching requests with a "segment prefix" rather than a "string prefix" according to the spec (e.g. the prefix `/foo/bar` will match requests with paths `/foo/bar`, `/foo/bar/`, and `/foo/bar/baz`, but not `/foo/barbaz`).
+
+## TLS
+
+See upstream [documentation][7] on TLS configuration.
+
+A secret specified in an Ingress TLS element will only be applied to Ingress rules with `Host` configuration that exactly matches an element of the TLS `Hosts` field.
+Any secrets that do not match an Ingress rule `Host` will be ignored.
+
+In Ingress v1beta1, the `secretName` field could contain a string with a full `namespace/name` identifier.
+When used with Contour's [TLS certificate delegation][8], this allowed Ingresses to use a TLS certificate from a different namespace.
+However, Ingress v1 does not allow the `secretName` field to contain a string with a full `namespace/name` identifier, because the field validation disallows the `/` character.
+Instead, Ingress v1 resources can now use the `projectcontour.io/tls-cert-namespace` annotation, to define the namespace that contains the TLS certificate (if different than the Ingress's namespace).
+This enables the TLS certificate delegation functionality to continue working for Ingress v1.
+For more information and an example, see the [TLS certificate delegation documentation][8].
+
+## Status
+
+In order to inform users of the address the Services their Ingress resources can be accessed at, Contour sets status on Ingress resources.
+If `contour serve` is run with the `--ingress-status-address` flag, Contour will use the provided value to set the Ingress status address accordingly.
+If not provided, Contour will use the address of the Envoy service using the passed in `--envoy-service-name` and `--envoy-service-namespace` flags.
+
+## Header Manipulation
+
+The Ingress resource does not allow adding or removing HTTP headers on requests or responses.
+However, Contour does allow users to set a global HTTP header [policy configuration][9] which can be optionally applied to configuration generated from Ingress resources.
+Contour enables this behavior with the `applyToIngress` boolean field (set to `true` to enable).
+
+[0]: https://github.com/kubernetes-sigs/ingress-controller-conformance
+[1]: /resources/compatibility-matrix/
+[2]: https://kubernetes.io/docs/concepts/services-networking/ingress/#ingress-class
+[3]: https://kubernetes.io/docs/concepts/services-networking/ingress/#default-backend
+[4]: https://kubernetes.io/docs/concepts/services-networking/ingress/#ingress-rules
+[5]: https://kubernetes.io/docs/concepts/services-networking/ingress/#hostname-wildcards
+[6]: https://kubernetes.io/docs/concepts/services-networking/ingress/#path-types
+[7]: https://kubernetes.io/docs/concepts/services-networking/ingress/#tls
+[8]: /docs/{{< param version >}}/config/tls-delegation/
+[9]: /docs/{{< param version >}}/configuration/#policy-configuration
diff --git a/site/content/docs/1.28/config/ip-filtering.md b/site/content/docs/1.28/config/ip-filtering.md
new file mode 100644
index 00000000000..161d39bc228
--- /dev/null
+++ b/site/content/docs/1.28/config/ip-filtering.md
@@ -0,0 +1,80 @@
+# IP Filtering
+
+Contour supports filtering requests based on the incoming ip address using Envoy's [RBAC Filter][1].
+
+Requests can be either allowed or denied based on a CIDR range specified on the virtual host and/or individual routes.
+
+If the request's IP address is allowed, the request will be proxied to the appropriate upstream.
+If the request's IP address is denied, an HTTP 403 (Forbidden) will be returned to the client.
+
+## Specifying Rules
+
+Rules are specified with the `ipAllowPolicy` and `ipDenyPolicy` fields on `virtualhost` and `route`:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: basic
+spec:
+ virtualhost:
+ fqdn: foo-basic.bar.com
+ ipAllowPolicy:
+ # traffic is allowed if it came from localhost (i.e. co-located reverse proxy)
+ - cidr: 127.0.0.1/32
+ source: Peer
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ # route-level ip filters override the virtualhost-level filters
+ ipAllowPolicy:
+ # traffic is allowed if it came from localhost (i.e. co-located reverse proxy)
+ - cidr: 127.0.0.1/32
+ source: Peer
+ # and the request originated from an IP in this range
+ - cidr: 99.99.0.0/16
+ source: Remote
+```
+
+### Specifying CIDR Ranges
+
+CIDR ranges may be ipv4 or ipv6. Bare IP addresses are interpreted as the CIDR range containing that one ip address only.
+
+Examples:
+- `1.1.1.1/24`
+- `127.0.0.1`
+- `2001:db8::68/24`
+- `2001:db8::68`
+
+### Allow vs Deny
+
+Filters are specified as either allow or deny:
+
+- `ipAllowPolicy` only allows requests that match the ip filters.
+- `ipDenyPolicy` denies all requests unless they match the ip filters.
+
+Allow and deny policies cannot both be specified at the same time for a virtual host or route.
+
+### IP Source
+
+The `source` field controls how the ip address is selected from the request for filtering.
+
+- `source: Peer` filter rules will filter using Envoy's [direct_remote_ip][2], which is always the physical peer.
+- `source: Remote` filter rules will filter using Envoy's [remote_ip][3], which may be inferred from the X-Forwarded-For header or proxy protocol.
+
+If using `source: Remote` with `X-Forwarded-For`, it may be necessary to configure Contour's `numTrustedHops` in [Network Parameters][4].
+
+### Virtual Host and Route Filter Precedence
+
+IP filters on the virtual host apply to all routes included in the virtual host, unless the route specifies its own rules.
+
+Rules specified on a route override any rules defined on the virtual host, they are not additive.
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/rbac_filter.html
+[2]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/rbac/v3/rbac.proto#envoy-v3-api-field-config-rbac-v3-principal-direct-remote-ip
+[3]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/rbac/v3/rbac.proto#envoy-v3-api-field-config-rbac-v3-principal-remote-ip
+[4]: api/#projectcontour.io/v1.NetworkParameters
+
diff --git a/site/content/docs/1.28/config/jwt-verification.md b/site/content/docs/1.28/config/jwt-verification.md
new file mode 100644
index 00000000000..3f884ad2aef
--- /dev/null
+++ b/site/content/docs/1.28/config/jwt-verification.md
@@ -0,0 +1,182 @@
+# JWT Verification
+
+Contour supports verifying JSON Web Tokens (JWTs) on incoming requests, using Envoy's [jwt_authn HTTP filter][1].
+Specifically, the following properties can be checked:
+- issuer field
+- audiences field
+- signature, using a configured JSON Web Key Store (JWKS)
+- time restrictions (e.g. expiration, not before time)
+
+If verification succeeds, the request will be proxied to the appropriate upstream.
+If verification fails, an HTTP 401 (Unauthorized) will be returned to the client.
+
+JWT verification is only supported on TLS-terminating virtual hosts.
+
+## Configuring providers and rules
+
+A JWT provider is configured for an HTTPProxy's virtual host, and defines how to verify JWTs:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: jwt-verification
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: example.com
+ tls:
+ secretName: example-com-tls-cert
+ jwtProviders:
+ - name: provider-1
+ issuer: example.com
+ audiences:
+ - audience-1
+ - audience-2
+ remoteJWKS:
+ uri: https://example.com/jwks.json
+ timeout: 1s
+ cacheDuration: 5m
+ forwardJWT: true
+ routes:
+ ...
+```
+
+The provider above requires JWTs to have an issuer of example.com, an audience of either audience-1 or audience-2, and a signature that can be verified using the configured JWKS.
+It also forwards the JWT to the backend via the `Authorization` header after successful verification.
+
+To apply a JWT provider as a requirement to a given route, specify a `jwtVerificationPolicy` for the route:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: jwt-verification
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: example.com
+ tls:
+ secretName: example-com-tls-cert
+ jwtProviders:
+ - name: provider-1
+ ...
+ routes:
+ - conditions:
+ - prefix: /
+ jwtVerificationPolicy:
+ require: provider-1
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /css
+ services:
+ - name: s1
+ port: 80
+```
+
+In the above example, the default route requires requests to carry JWTs that can be verified using provider-1.
+The second route _excludes_ requests to paths starting with `/css` from JWT verification, because it does not have a JWT verification policy.
+
+### Configuring TLS validation for the JWKS server
+
+By default, the JWKS server's TLS certificate will not be validated, but validation can be requested by setting the `spec.virtualhost.jwtProviders[].remoteJWKS.validation` field.
+This field has mandatory `caSecret` and `subjectName` fields, which specify the trusted root certificates with which to validate the server certificate and the expected server name.
+The `caSecret` can be a namespaced name of the form `/`.
+If the CA secret's namespace is not the same namespace as the `HTTPProxy` resource, [TLS Certificate Delegation][5] must be used to allow the owner of the CA certificate secret to delegate, for the purposes of referencing the CA certificate in a different namespace, permission to Contour to read the Secret object from another namespace.
+
+**Note:** If `spec.virtualhost.jwtProviders[].remoteJWKS.validation` is present, `spec.virtualhost.jwtProviders[].remoteJWKS.uri` must have a scheme of `https`.
+
+## Setting a default provider
+
+The previous section showed how to explicitly require JWT providers for specific routes.
+An alternate approach is to define a JWT provider as the default by specifying `default: true` for it, in which case it is automatically applied to all routes unless they disable JWT verification.
+The example from the previous section could alternately be configured as follows:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: jwt-verification
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: example.com
+ tls:
+ secretName: example-com-tls-cert
+ jwtProviders:
+ - name: provider-1
+ default: true
+ ...
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /css
+ jwtVerificationPolicy:
+ disabled: true
+ services:
+ - name: s1
+ port: 80
+```
+
+In this case, the default route automatically has provider-1 applied, while the `/css` route explicitly disables JWT verification.
+
+One scenario where setting a default provider can be particularly useful is when using [HTTPProxy inclusion][2].
+Setting a default provider in the root HTTPProxy allows all routes in the child HTTPProxies to automatically have JWT verification applied.
+For example:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: jwt-verification-root
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: example.com
+ tls:
+ secretName: example-com-tls-cert
+ jwtProviders:
+ - name: provider-1
+ default: true
+ ...
+ includes:
+ - name: jwt-verification-child
+ namespace: default
+ conditions:
+ - prefix: /blog
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: jwt-verification-child
+ namespace: default
+spec:
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+```
+
+In this case, all routes in the child HTTPProxy will automatically have JWT verification applied, without the owner of this HTTPProxy needing to configure it explicitly.
+
+## API documentation
+
+For more information on the HTTPProxy API for JWT verification, see:
+
+- [JWTProvider][3]
+- [JWTVerificationPolicy][4]
+
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/jwt_authn_filter
+[2]: /docs/{{< param version >}}/config/inclusion-delegation/
+[3]: /docs/{{< param version >}}/config/api/#projectcontour.io/v1.JWTProvider
+[4]: /docs/{{< param version >}}/config/api/#projectcontour.io/v1.JWTVerificationPolicy
+[5]: tls-delegation.md
diff --git a/site/content/docs/1.28/config/overload-manager.md b/site/content/docs/1.28/config/overload-manager.md
new file mode 100644
index 00000000000..33c96532eba
--- /dev/null
+++ b/site/content/docs/1.28/config/overload-manager.md
@@ -0,0 +1,30 @@
+# Overload Manager
+
+Envoy uses heap memory when processing requests.
+When the system runs out of memory or memory resource limit for the container is reached, Envoy process is terminated abruptly.
+To avoid this, Envoy [overload manager][1] can be enabled.
+Overload manager controls how much memory Envoy will allocate at maximum and what actions it takes when the limit is reached.
+
+Overload manager is disabled by default.
+It can be enabled at deployment time by using `--overload-max-heap=[MAX_BYTES]` command line flag in [`contour bootstrap`][2] command.
+The bootstrap command is executed in [init container of Envoy pod][3] to generate initial configuration for Envoy.
+To enable overload manager, modify the deployment manifest and add for example `--overload-max-heap=2147483648` to set maximum heap size to 2 GiB.
+The appropriate number of bytes can be different from system to system.
+
+After the feature is enabled, following two overload actions are configured to Envoy:
+
+* Shrink heap action is executed when 95% of the maximum heap size is reached.
+* Envoy will stop accepting requests when 98% of the maximum heap size is reached.
+
+When requests are denied due to high memory pressure, `503 Service Unavailable` will be returned with a response body containing text `envoy overloaded`.
+Shrink heap action will try to free unused heap memory, eventually allowing requests to be processed again.
+
+**NOTE:**
+The side effect of overload is that Envoy will deny also requests `/ready` and `/stats` endpoints.
+This is due to the way how Contour secures Envoy's admin API and exposes only selected admin API endpoints by proxying itself.
+When readiness probe fails, the overloaded Envoy will be removed from the list of service endpoints.
+If the maximum heap size is set too low, Envoy may be unable to free enough memory and never become ready again.
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/configuration/operations/overload_manager/overload_manager
+[2]: ../configuration#bootstrap-flags
+[3]: https://github.com/projectcontour/contour/blob/cbec8eca9e8b639318588c5aa7ec0b5b751938c5/examples/render/contour.yaml#L5204-L5216
diff --git a/site/content/docs/1.28/config/rate-limiting.md b/site/content/docs/1.28/config/rate-limiting.md
new file mode 100644
index 00000000000..7a69c22c079
--- /dev/null
+++ b/site/content/docs/1.28/config/rate-limiting.md
@@ -0,0 +1,366 @@
+# Rate Limiting
+
+- [Overview](#overview)
+- [Local Rate Limiting](#local-rate-limiting)
+- [Global Rate Limiting](#global-rate-limiting)
+
+## Overview
+
+Rate limiting is a means of protecting backend services against unwanted traffic.
+This can be useful for a variety of different scenarios:
+
+- Protecting against denial-of-service (DoS) attacks by malicious actors
+- Protecting against DoS incidents due to bugs in client applications/services
+- Enforcing usage quotas for different classes of clients, e.g. free vs. paid tiers
+- Controlling resource consumption/cost
+
+Envoy supports two forms of HTTP rate limiting: **local** and **global**.
+
+In local rate limiting, rate limits are enforced by each Envoy instance, without any communication with other Envoys or any external service.
+
+In global rate limiting, an external rate limit service (RLS) is queried by each Envoy via gRPC for rate limit decisions.
+
+Contour supports both forms of Envoy's rate limiting.
+
+## Local Rate Limiting
+
+The `HTTPProxy` API supports defining local rate limit policies that can be applied to either individual routes or entire virtual hosts.
+Local rate limit policies define a maximum number of requests per unit of time that an Envoy should proxy to the upstream service.
+Requests beyond the defined limit will receive a `429 (Too Many Requests)` response by default.
+Local rate limit policies program Envoy's [HTTP local rate limit filter][1].
+
+It's important to note that local rate limit policies apply *per Envoy pod*.
+For example, a local rate limit policy of 100 requests per second for a given route will result in *each Envoy pod* allowing up to 100 requests per second for that route.
+
+### Defining a local rate limit
+
+Local rate limit policies can be defined for either routes or virtual hosts. A local rate limit policy requires a `requests` and a `units` field, defining the *number of requests per unit of time* that are allowed. `Requests` must be a positive integer, and `units` can be `second`, `minute`, or `hour`. Optionally, a `burst` parameter can also be provided, defining the number of requests above the baseline rate that are allowed in a short period of time. This would allow occasional larger bursts of traffic not to be rate limited.
+
+Local rate limiting for the virtual host:
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: ratelimited-vhost
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ rateLimitPolicy:
+ local:
+ requests: 100
+ unit: hour
+ burst: 20
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /s2
+ services:
+ - name: s2
+ port: 80
+```
+
+Local rate limiting for the route:
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: ratelimited-route
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ rateLimitPolicy:
+ local:
+ requests: 20
+ unit: minute
+ - conditions:
+ - prefix: /s2
+ services:
+ - name: s2
+ port: 80
+```
+
+### Customizing the response
+
+#### Response code
+
+By default, Envoy returns a `429 (Too Many Requests)` when a request is rate limited.
+A non-default response code can optionally be configured as part of the local rate limit policy, in the `responseStatusCode` field.
+The value must be in the 400-599 range.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: custom-ratelimit-response
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ rateLimitPolicy:
+ local:
+ requests: 20
+ unit: minute
+ responseStatusCode: 503 # Service Unavailable
+```
+
+#### Headers
+
+Headers can optionally be added to rate limited responses, by configuring the `responseHeadersToAdd` field.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: custom-ratelimit-response
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ rateLimitPolicy:
+ local:
+ requests: 20
+ unit: minute
+ responseHeadersToAdd:
+ - name: x-contour-ratelimited
+ value: "true"
+```
+
+## Global Rate Limiting
+
+The `HTTPProxy` API also supports defining global rate limit policies on routes and virtual hosts.
+
+In order to use global rate limiting, you must first select and deploy an external rate limit service (RLS).
+There is an [Envoy rate limit service implementation][2], but any service that implements the [RateLimitService gRPC interface][3] is supported.
+
+### Configuring an external RLS with Contour
+
+Once you have deployed your RLS, you must configure it with Contour.
+
+Define an extension service for it (substituting values as appropriate):
+```yaml
+apiVersion: projectcontour.io/v1alpha1
+kind: ExtensionService
+metadata:
+ namespace: projectcontour
+ name: ratelimit
+spec:
+ protocol: h2
+ services:
+ - name: ratelimit
+ port: 8081
+```
+
+Now add a reference to it in the Contour config file:
+```yaml
+rateLimitService:
+ # The namespace/name of the extension service.
+ extensionService: projectcontour/ratelimit
+ # The domain value to pass to the RLS for all rate limit
+ # requests. Acts as a container for a set of rate limit
+ # definitions within the RLS.
+ domain: contour
+ # Whether to allow requests to proceed when the rate limit
+ # service fails to respond with a valid rate limit decision
+ # within the timeout defined on the extension service.
+ failOpen: true
+```
+
+### Defining a global rate limit policy
+
+Global rate limit policies can be defined for either routes or virtual hosts. Unlike local rate limit policies, global rate limit policies do not directly define a rate limit. Instead, they define a set of request descriptors that will be generated and sent to the external RLS for each request. The external RLS then makes the rate limit decision based on the descriptors and returns a response to Envoy.
+
+A global rate limit policy for the virtual host:
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: ratelimited-vhost
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ rateLimitPolicy:
+ global:
+ descriptors:
+ # the first descriptor has a single key-value pair:
+ # [ remote_address= ].
+ - entries:
+ - remoteAddress: {}
+ # the second descriptor has two key-value pairs:
+ # [ remote_address=, vhost=local.projectcontour.io ].
+ - entries:
+ - remoteAddress: {}
+ - genericKey:
+ key: vhost
+ value: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /s2
+ services:
+ - name: s2
+ port: 80
+```
+
+A global rate limit policy for the route:
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ namespace: default
+ name: ratelimited-route
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /s1
+ services:
+ - name: s1
+ port: 80
+ rateLimitPolicy:
+ global:
+ descriptors:
+ # the first descriptor has a single key-value pair:
+ # [ remote_address= ].
+ - entries:
+ - remoteAddress: {}
+ # the second descriptor has two key-value pairs:
+ # [ remote_address=, prefix=/s1 ].
+ - entries:
+ - remoteAddress: {}
+ - genericKey:
+ key: prefix
+ value: /s1
+ - conditions:
+ - prefix: /s2
+ services:
+ - name: s2
+ port: 80
+```
+
+#### Descriptors & descriptor entries
+
+A descriptor is a list of key-value pairs, i.e. entries, that are generated for a request. The entries can be generated based on different criteria. If any entry in a descriptor cannot generate a key-value pair for a given request, then the entire descriptor is not generated (see the [Envoy documentation][8] for more information). When a global rate limit policy defines multiple descriptors, then *all* descriptors that can be generated will be generated and sent to the rate limit service for consideration.
+
+Below are the supported types of descriptor entries.
+
+##### GenericKey
+
+A `GenericKey` descriptor entry defines a static key-value pair. For example:
+
+```yaml
+rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - genericKey:
+ key: virtual-host-name
+ value: foo.bar.com
+```
+
+Produces a descriptor entry of `virtual-host-name=foo.bar.com`.
+
+The `key` field is optional and defaults to a value of `generic_key` if not specified.
+
+See the [Envoy documentation][4] for more information and examples.
+
+##### RemoteAddress
+
+A `RemoteAddress` descriptor entry has a key of `remote_address` and a value of the client IP address (using the trusted address from `x-forwarded-for`). For example:
+
+```yaml
+rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - remoteAddress: {}
+```
+
+Produces a descriptor entry of `remote_address=`.
+
+See the [Envoy documentation][5] for more information and examples.
+
+##### RequestHeader
+
+A `RequestHeader` descriptor entry has a static key and a value equal to the value of a specified header on the client request. If the header is not present, the descriptor entry is not generated. For example:
+
+```yaml
+rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - requestHeader:
+ headerName: My-Header
+ descriptorKey: my-header-value
+```
+
+Produces a descriptor entry of `my-header-value=`, for a client request that has the `My-Header` header.
+
+See the [Envoy documentation][6] for more information and examples.
+
+##### RequestHeaderValueMatch
+
+A `RequestHeaderValueMatch` descriptor entry has a key of `header_match` and a static value. The entry is only generated if the client request's headers match a specified set of criteria. For example:
+
+```yaml
+rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - requestHeaderValueMatch:
+ headers:
+ - name: My-Header
+ notpresent: true
+ - name: My-Other-Header
+ contains: contour
+ expectMatch: true
+ value: foo
+```
+
+Produces a descriptor entry of `header_match=foo`, for a client request that does not have the `My-Header` header, and does have the `My-Other-Header` header, with a value containing the substring "contour".
+
+Contour supports `present`, `notpresent`, `contains`, `notcontains`, `exact`, and `notexact` header match operators.
+
+The `expectMatch` field defaults to true if not specified. If true, the client request's headers must positively match the specified criteria in order for the descriptor entry to be generated. If false, the client request's header must *not* match the specified criteria in order for the descriptor entry to be generated.
+
+See the [Envoy documentation][7] for more information and examples.
+
+
+
+[1]: https://www.envoyproxy.io/docs/envoy/v1.17.0/configuration/http/http_filters/local_rate_limit_filter#config-http-filters-local-rate-limit
+[2]: https://github.com/envoyproxy/ratelimit
+[3]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/service/ratelimit/v3/rls.proto
+[4]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-msg-config-route-v3-ratelimit-action-generickey
+[5]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#config-route-v3-ratelimit-action-remoteaddress
+[6]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#config-route-v3-ratelimit-action-requestheaders
+[7]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#config-route-v3-ratelimit-action-headervaluematch
+[8]: https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/rate_limit_filter#composing-actions
diff --git a/site/content/docs/1.28/config/request-rewriting.md b/site/content/docs/1.28/config/request-rewriting.md
new file mode 100644
index 00000000000..88fa3cc2508
--- /dev/null
+++ b/site/content/docs/1.28/config/request-rewriting.md
@@ -0,0 +1,337 @@
+# Request Rewriting
+
+## Path Rewriting
+
+HTTPProxy supports rewriting the HTTP request URL path prior to delivering the request to the backend service.
+Rewriting is performed after a routing decision has been made, and never changes the request destination.
+
+The `pathRewritePolicy` field specifies how the path prefix should be rewritten.
+The `replacePrefix` rewrite policy specifies a replacement string for a HTTP request path prefix match.
+When this field is present, the path prefix that the request matched is replaced by the text specified in the `replacement` field.
+If the HTTP request path is longer than the matched prefix, the remainder of the path is unchanged.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: rewrite-example
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: rewrite.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ pathRewritePolicy:
+ replacePrefix:
+ - replacement: /new/prefix
+```
+
+The `replacePrefix` field accepts an array of possible replacements.
+When more than one `replacePrefix` array element is present, the `prefix` field can be used to disambiguate which replacement to apply.
+
+If no `prefix` field is present, the replacement is applied to all prefix matches made against the route.
+If a `prefix` field is present, the replacement is applied only to routes that have an exactly matching prefix condition.
+Specifying more than one `replacePrefix` entry is mainly useful when a HTTPProxy document is included into multiple parent documents.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: rewrite-example
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: rewrite.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ conditions:
+ - prefix: /v1/api
+ pathRewritePolicy:
+ replacePrefix:
+ - prefix: /v1/api
+ replacement: /app/api/v1
+ - prefix: /
+ replacement: /app
+```
+
+## Header Rewriting
+
+HTTPProxy supports rewriting HTTP request and response headers.
+The `Set` operation sets a HTTP header value, creating it if it doesn't already exist or overwriting it if it does.
+The `Remove` operation removes a HTTP header.
+The `requestHeadersPolicy` field is used to rewrite headers on a HTTP request, and the `responseHeadersPolicy` is used to rewrite headers on a HTTP response.
+These fields can be specified on a route or on a specific service, depending on the rewrite granularity you need.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: header-rewrite-example
+spec:
+ virtualhost:
+ fqdn: header.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ requestHeadersPolicy:
+ set:
+ - name: Host
+ value: external.dev
+ remove:
+ - Some-Header
+ - Some-Other-Header
+```
+
+Manipulating headers is also supported per-Service or per-Route. Headers can be set or
+removed from the request or response as follows:
+
+per-Service:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: header-manipulation
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: headers.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ requestHeadersPolicy:
+ set:
+ - name: X-Foo
+ value: bar
+ remove:
+ - X-Baz
+ responseHeadersPolicy:
+ set:
+ - name: X-Service-Name
+ value: s1
+ remove:
+ - X-Internal-Secret
+```
+
+per-Route:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: header-manipulation
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: headers.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ requestHeadersPolicy:
+ set:
+ - name: X-Foo
+ value: bar
+ remove:
+ - X-Baz
+ responseHeadersPolicy:
+ set:
+ - name: X-Service-Name
+ value: s1
+ remove:
+ - X-Internal-Secret
+```
+
+In these examples we are setting the header `X-Foo` with value `baz` on requests
+and stripping `X-Baz`. We are then setting `X-Service-Name` on the response with
+value `s1`, and removing `X-Internal-Secret`.
+
+### Dynamic Header Values
+
+It is sometimes useful to set a header value using a dynamic value such as the
+hostname where the Envoy Pod is running (`%HOSTNAME%`) or the subject of the
+TLS client certificate (`%DOWNSTREAM_PEER_SUBJECT%`) or based on another header
+(`%REQ(header)%`).
+
+Examples:
+```
+ requestHeadersPolicy:
+ set:
+ - name: X-Envoy-Hostname
+ value: "%HOSTNAME%"
+ - name: X-Host-Protocol
+ value: "%REQ(Host)% - %PROTOCOL%"
+ responseHeadersPolicy:
+ set:
+ - name: X-Envoy-Response-Flags
+ value: "%RESPONSE_FLAGS%"
+```
+
+Contour supports most of the custom request/response header variables offered
+by Envoy - see the Envoy
+documentation for details of what each of these resolve to:
+
+* `%DOWNSTREAM_REMOTE_ADDRESS%`
+* `%DOWNSTREAM_REMOTE_ADDRESS_WITHOUT_PORT%`
+* `%DOWNSTREAM_LOCAL_ADDRESS%`
+* `%DOWNSTREAM_LOCAL_ADDRESS_WITHOUT_PORT%`
+* `%DOWNSTREAM_LOCAL_PORT%`
+* `%DOWNSTREAM_LOCAL_URI_SAN%`
+* `%DOWNSTREAM_PEER_URI_SAN%`
+* `%DOWNSTREAM_LOCAL_SUBJECT%`
+* `%DOWNSTREAM_PEER_SUBJECT%`
+* `%DOWNSTREAM_PEER_ISSUER%`
+* `%DOWNSTREAM_TLS_SESSION_ID%`
+* `%DOWNSTREAM_TLS_CIPHER%`
+* `%DOWNSTREAM_TLS_VERSION%`
+* `%DOWNSTREAM_PEER_FINGERPRINT_256%`
+* `%DOWNSTREAM_PEER_FINGERPRINT_1%`
+* `%DOWNSTREAM_PEER_SERIAL%`
+* `%DOWNSTREAM_PEER_CERT%`
+* `%DOWNSTREAM_PEER_CERT_V_START%`
+* `%DOWNSTREAM_PEER_CERT_V_END%`
+* `%HOSTNAME%`
+* `%REQ(header-name)%`
+* `%PROTOCOL%`
+* `%RESPONSE_FLAGS%`
+* `%RESPONSE_CODE_DETAILS%`
+* `%UPSTREAM_REMOTE_ADDRESS%`
+
+Note that Envoy passes variables that can't be expanded through unchanged or
+skips them entirely - for example:
+* `%UPSTREAM_REMOTE_ADDRESS%` as a request header remains as
+ `%UPSTREAM_REMOTE_ADDRESS%` because as noted in the Envoy docs: "The upstream
+ remote address cannot be added to request headers as the upstream host has not
+ been selected when custom request headers are generated."
+* `%DOWNSTREAM_TLS_VERSION%` is skipped if TLS is not in use
+* Envoy ignores REQ headers that refer to an non-existent header - for example
+ `%REQ(Host)%` works as expected but `%REQ(Missing-Header)%` is skipped
+
+Contour already sets the `X-Request-Start` request header to
+`t=%START_TIME(%s.%3f)%` which is the Unix epoch time when the request
+started.
+
+To enable setting header values based on the destination service Contour also supports:
+
+* `%CONTOUR_NAMESPACE%`
+* `%CONTOUR_SERVICE_NAME%`
+* `%CONTOUR_SERVICE_PORT%`
+
+For example, with the following HTTPProxy object that has a per-Service requestHeadersPolicy using these variables:
+```
+# httpproxy.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: basic
+ namespace: myns
+spec:
+ virtualhost:
+ fqdn: foo-basic.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ requestHeadersPolicy:
+ set:
+ - name: l5d-dst-override
+ value: "%CONTOUR_SERVICE_NAME%.%CONTOUR_NAMESPACE%.svc.cluster.local:%CONTOUR_SERVICE_PORT%"
+```
+the values would be:
+* `CONTOUR_NAMESPACE: "myns"`
+* `CONTOUR_SERVICE_NAME: "s1"`
+* `CONTOUR_SERVICE_PORT: "80"`
+
+and the `l5-dst-override` header would be set to `s1.myns.svc.cluster.local:80`.
+
+For per-Route requestHeadersPolicy only `%CONTOUR_NAMESPACE%` is set and using
+`%CONTOUR_SERVICE_NAME%` and `%CONTOUR_SERVICE_PORT%` will end up as the
+literal values `%%CONTOUR_SERVICE_NAME%%` and `%%CONTOUR_SERVICE_PORT%%`,
+respectively.
+
+### Manipulating the Host header
+
+Contour allows users to manipulate the host header in two ways, using the `requestHeadersPolicy`.
+
+#### Static rewrite
+
+You can set the host to a static value. This can be done on the route and service level.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: static-host-header-rewrite-route
+spec:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ - requestHeaderPolicy:
+ set:
+ - name: host
+ value: foo.com
+```
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: static-host-header-rewrite-service
+spec:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ - requestHeaderPolicy:
+ set:
+ - name: host
+ value: "foo.com"
+```
+
+#### Dynamic rewrite
+
+You can also set the host header dynamically with the content of an existing header.
+The format has to be `"%REQ()%"`. If the header is empty, it is ignored.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: dynamic-host-header-rewrite-route
+spec:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+ - requestHeaderPolicy:
+ set:
+ - name: host
+ value: "%REQ(x-rewrite-header)%"
+```
+
+Note: Only one of static or dynamic host rewrite can be specified.
+
+Note: Dynamic rewrite is only available at the route level and not possible on the service level.
+
+Note: Pay attention to the potential security implications of using this option, the provided header must come from a trusted source.
+
+Note: The header rewrite is only done while forwarding and has no bearing on the routing decision.
diff --git a/site/content/docs/1.28/config/request-routing.md b/site/content/docs/1.28/config/request-routing.md
new file mode 100644
index 00000000000..19ef5386e86
--- /dev/null
+++ b/site/content/docs/1.28/config/request-routing.md
@@ -0,0 +1,535 @@
+# Request Routing
+
+A HTTPProxy object must have at least one route or include defined.
+In this example, any requests to `multi-path.bar.com/blog` or `multi-path.bar.com/blog/*` will be routed to the Service `s2` using the prefix conditions. Requests to `multi-path.bar.com/feed` will be routed to Service `s2` using exact match condition.
+All other requests to the host `multi-path.bar.com` will be routed to the Service `s1`.
+
+```yaml
+# httpproxy-multiple-paths.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-paths
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: multi-path.bar.com
+ routes:
+ - conditions:
+ - prefix: / # matches everything else
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /blog # matches `multi-path.bar.com/blog` or `multi-path.bar.com/blog/*`
+ services:
+ - name: s2
+ port: 80
+ - conditions:
+ - exact: /feed # matches `multi-path.bar.com/feed` only
+ services:
+ - name: s2
+ port: 80
+```
+
+In the following example, we match on headers and query parameters and send to different services, with a default route if those do not match.
+
+```yaml
+# httpproxy-multiple-headers.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-paths
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: multi-path.bar.com
+ routes:
+ - conditions:
+ - header:
+ name: x-os
+ contains: ios
+ services:
+ - name: s1
+ port: 80
+ - conditions:
+ - header:
+ name: x-os
+ contains: android
+ services:
+ - name: s2
+ port: 80
+ - conditions:
+ - queryParameter:
+ name: os
+ exact: other
+ ignoreCase: true
+ services:
+ - name: s3
+ port: 80
+ - services:
+ - name: s4
+ port: 80
+```
+
+## Conditions
+
+Each Route entry in a HTTPProxy **may** contain one or more conditions.
+These conditions are combined with an AND operator on the route passed to Envoy.
+Conditions can be either a `prefix`, `exact`, `regex`, `header` or a `queryParameter` condition. At most one of `prefix`, `exact` or `regex` can be used in one condition block.
+
+#### Prefix conditions
+
+Paths defined are matched using prefix conditions.
+Up to one prefix condition may be present in any condition block.
+
+Prefix conditions **must** start with a `/` if they are present.
+
+#### Exact conditions
+
+Paths defined are matched using exact conditions.
+Up to one exact condition may be present in any condition block. Any condition block can
+either have a regex condition, exact condition or prefix condition, but not multiple together. Exact conditions are
+only allowed in route match conditions and not in include match conditions.
+
+Exact conditions **must** start with a `/` if they are present.
+
+#### Regex conditions
+
+Paths defined are matched using regex expressions.
+Up to one regex condition may be present in any condition block. Any condition block can
+either have a regex condition, exact condition or prefix condition, but not multiple together. Regex conditions are
+only allowed in route match conditions and not in include match conditions.
+
+Regex conditions **must** start with a `/` if they are present.
+
+#### Header conditions
+
+For `header` conditions there is the following structure:
+
+1. one required field, `name`
+2. six operator fields: `present`, `notpresent`, `contains`, `notcontains`, `exact`, and `notexact`
+3. two optional modifiers: `ignoreCase` and `treatMissingAsEmpty`
+
+Operators:
+- `present` is a boolean and checks that the header is present. The value will not be checked.
+
+- `notpresent` similarly checks that the header is *not* present.
+
+- `contains` is a string, and checks that the header contains the string. `notcontains` similarly checks that the header does *not* contain the string.
+
+- `exact` is a string, and checks that the header exactly matches the whole string. `notexact` checks that the header does *not* exactly match the whole string.
+
+- `regex` is a string representing a regular expression, and checks that the header value matches against the given regular expression.
+
+Modifiers:
+- `ignoreCase`: IgnoreCase specifies that string matching should be case insensitive. It has no effect on the `Regex` parameter.
+- `treatMissingAsEmpty`: specifies if the header match rule specified header does not exist, this header value will be treated as empty. Defaults to false. Unlike the underlying Envoy implementation this is **only** supported for negative matches (e.g. NotContains, NotExact).
+
+#### Query parameter conditions
+
+Similar to the `header` conditions, `queryParameter` conditions also require the
+`name` field to be specified, which represents the name of the query parameter
+e.g. `search` when the query string looks like `/?search=term` and `term`
+representing the value.
+
+There are six operator fields: `exact`, `prefix`, `suffix`, `regex`, `contains`
+and `present` and a modifier `ignoreCase` which can be used together with all of
+the operator fields except `regex` and `present`.
+
+- `exact` is a string, and checks that the query parameter value exactly matches
+ the whole string.
+
+- `prefix` is a string, and checks that the query parameter value is prefixed by
+ the given value.
+
+- `suffix` is a string, and checks that the query parameter value is suffixed by
+ the given value.
+
+- `regex` is a string representing a regular expression, and checks that the
+ query parameter value matches against the given regular expression.
+
+- `contains` is a string, and checks that the query parameter value contains
+ the given string.
+
+- `present` is a boolean, and checks that the query parameter is present. The
+ value will not be checked.
+
+- `ignoreCase` is a boolean, and if set to `true` it will enable case
+ insensitive matching for any of the string operator matching methods.
+
+## Request Redirection
+
+HTTP redirects can be implemented in HTTPProxy using `requestRedirectPolicy` on a route.
+In the following basic example, requests to `example.com` are redirected to `www.example.com`.
+We configure a root HTTPProxy for `example.com` that contains redirect configuration.
+We also configure a root HTTPProxy for `www.example.com` that represents the destination of the redirect.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: example-com
+spec:
+ virtualhost:
+ fqdn: example.com
+ routes:
+ - conditions:
+ - prefix: /
+ requestRedirectPolicy:
+ hostname: www.example.com
+```
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: www-example-com
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1
+ port: 80
+```
+
+In addition to specifying the hostname to set in the `location` header, the scheme, port, and returned status code of the redirect response can be configured.
+Configuration of the path or a path prefix replacement to modify the path of the returned `location` can be included as well.
+See [the API specification][3] for more detail.
+
+## Multiple Upstreams
+
+One of the key HTTPProxy features is the ability to support multiple services for a given path:
+
+```yaml
+# httpproxy-multiple-upstreams.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-upstreams
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: multi.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ - name: s2
+ port: 80
+```
+
+In this example, requests for `multi.bar.com/` will be load balanced across two Kubernetes Services, `s1`, and `s2`.
+This is helpful when you need to split traffic for a given URL across two different versions of an application.
+
+### Upstream Weighting
+
+Building on multiple upstreams is the ability to define relative weights for upstream Services.
+This is commonly used for canary testing of new versions of an application when you want to send a small fraction of traffic to a specific Service.
+
+```yaml
+# httpproxy-weight-shifting.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: weight-shifting
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: weights.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ weight: 10
+ - name: s2
+ port: 80
+ weight: 90
+```
+
+In this example, we are sending 10% of the traffic to Service `s1`, while Service `s2` receives the remaining 90% of traffic.
+
+HTTPProxy weighting follows some specific rules:
+
+- If no weights are specified for a given route, it's assumed even distribution across the Services.
+- Weights are relative and do not need to add up to 100. If all weights for a route are specified, then the "total" weight is the sum of those specified. As an example, if weights are 20, 30, 20 for three upstreams, the total weight would be 70. In this example, a weight of 30 would receive approximately 42.9% of traffic (30/70 = .4285).
+- If some weights are specified but others are not, then it's assumed that upstreams without weights have an implicit weight of zero, and thus will not receive traffic.
+
+### Traffic mirroring
+
+Per route, a service can be nominated as a mirror.
+The mirror service will receive a copy of the read traffic sent to any non mirror service.
+The mirror traffic is considered _read only_, any response by the mirror will be discarded.
+
+This service can be useful for recording traffic for later replay or for smoke testing new deployments.
+
+`weight` can be optionally set (in the space of integers 1-100) to mirror the corresponding percent of traffic (ie. `weight: 5` mirrors 5% of traffic). Omitting the `weight` field results in 100% traffic mirroring. There is unexpected behavior if `weight` is explicitly set to 0, 100% traffic will be mirrored. This occurs because we cannot distinguish undefined variables from explicitly setting them to default values, and omission of a `weight` must mirror full traffic.
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: traffic-mirror
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: www
+ port: 80
+ - name: www-mirror
+ port: 80
+ mirror: true
+```
+
+## Response Timeouts
+
+Each Route can be configured to have a timeout policy and a retry policy as shown:
+
+```yaml
+# httpproxy-response-timeout.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: response-timeout
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: timeout.bar.com
+ routes:
+ - timeoutPolicy:
+ response: 1s
+ idle: 10s
+ idleConnection: 60s
+ retryPolicy:
+ count: 3
+ perTryTimeout: 150ms
+ services:
+ - name: s1
+ port: 80
+```
+
+In this example, requests to `timeout.bar.com/` will have a response timeout policy of 1s.
+This refers to the time that spans between the point at which complete client request has been processed by the proxy, and when the response from the server has been completely processed.
+
+- `timeoutPolicy.response` Timeout for receiving a response from the server after processing a request from client.
+If not supplied, Envoy's default value of 15s applies.
+More information can be found in [Envoy's documentation][4].
+- `timeoutPolicy.idle` Timeout for how long the proxy should wait while there is no activity during single request/response (for HTTP/1.1) or stream (for HTTP/2).
+Timeout will not trigger while HTTP/1.1 connection is idle between two consecutive requests.
+If not specified, there is no per-route idle timeout, though a connection manager-wide stream idle timeout default of 5m still applies.
+More information can be found in [Envoy's documentation][6].
+- `timeoutPolicy.idleConnection` Timeout for how long connection from the proxy to the upstream service is kept when there are no active requests.
+If not supplied, Envoy’s default value of 1h applies.
+More information can be found in [Envoy's documentation][8].
+
+TimeoutPolicy durations are expressed in the Go [Duration format][5].
+Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h".
+The string "infinity" is also a valid input and specifies no timeout.
+A value of "0s" will be treated as if the field were not set, i.e. by using Envoy's default behavior.
+Example input values: "300ms", "5s", "1m".
+
+- `retryPolicy`: A retry will be attempted if the server returns an error code in the 5xx range, or if the server takes more than `retryPolicy.perTryTimeout` to process a request.
+
+- `retryPolicy.count` specifies the maximum number of retries allowed. This parameter is optional and defaults to 1. Set to -1 to disable. If set to 0, the Envoy default of 1 is used.
+
+- `retryPolicy.perTryTimeout` specifies the timeout per retry. If this field is greater than the request timeout, it is ignored. This parameter is optional.
+ If left unspecified, `timeoutPolicy.request` will be used.
+
+## Load Balancing Strategy
+
+Each route can have a load balancing strategy applied to determine which of its Endpoints is selected for the request.
+The following list are the options available to choose from:
+
+- `RoundRobin`: Each healthy upstream Endpoint is selected in round-robin order (Default strategy if none selected).
+- `WeightedLeastRequest`: The least request load balancer uses different algorithms depending on whether hosts have the same or different weights in an attempt to route traffic based upon the number of active requests or the load at the time of selection.
+- `Random`: The random strategy selects a random healthy Endpoints.
+- `RequestHash`: The request hashing strategy allows for load balancing based on request attributes. An upstream Endpoint is selected based on the hash of an element of a request. For example, requests that contain a consistent value in an HTTP request header will be routed to the same upstream Endpoint. Currently, only hashing of HTTP request headers, query parameters and the source IP of a request is supported.
+- `Cookie`: The cookie load balancing strategy is similar to the request hash strategy and is a convenience feature to implement session affinity, as described below.
+
+More information on the load balancing strategy can be found in [Envoy's documentation][7].
+
+The following example defines the strategy for the route `/` as `WeightedLeastRequest`.
+
+```yaml
+# httpproxy-lb-strategy.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: lb-strategy
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: strategy.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: s1-strategy
+ port: 80
+ - name: s2-strategy
+ port: 80
+ loadBalancerPolicy:
+ strategy: WeightedLeastRequest
+```
+
+The below example demonstrates how request hash load balancing policies can be configured:
+
+Request hash headers
+```yaml
+# httpproxy-lb-request-hash.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: lb-request-hash
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: request-hash.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: httpbin
+ port: 8080
+ loadBalancerPolicy:
+ strategy: RequestHash
+ requestHashPolicies:
+ - headerHashOptions:
+ headerName: X-Some-Header
+ terminal: true
+ - headerHashOptions:
+ headerName: User-Agent
+ - hashSourceIP: true
+```
+In this example, if a client request contains the `X-Some-Header` header, the value of the header will be hashed and used to route to an upstream Endpoint. This could be used to implement a similar workflow to cookie-based session affinity by passing a consistent value for this header. If it is present, because it is set as a `terminal` hash option, Envoy will not continue on to process to `User-Agent` header or source IP to calculate a hash. If `X-Some-Header` is not present, Envoy will use the `User-Agent` header value to make a routing decision along with the source IP of the client making the request. These policies can be used alone or as shown for an advanced routing decision.
+
+
+Request hash source ip
+```yaml
+# httpproxy-lb-request-hash-ip.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: lb-request-hash
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: request-hash.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: httpbin
+ port: 8080
+ loadBalancerPolicy:
+ strategy: RequestHash
+ requestHashPolicies:
+ - hashSourceIP: true
+```
+
+Request hash query parameters
+```yaml
+# httpproxy-lb-request-hash.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: lb-request-hash
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: request-hash.bar.com
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: httpbin
+ port: 8080
+ loadBalancerPolicy:
+ strategy: RequestHash
+ requestHashPolicies:
+ - queryParameterHashOptions:
+ prameterName: param1
+ terminal: true
+ - queryParameterHashOptions:
+ parameterName: param2
+```
+
+## Session Affinity
+
+Session affinity, also known as _sticky sessions_, is a load balancing strategy whereby a sequence of requests from a single client are consistently routed to the same application backend.
+Contour supports session affinity on a per-route basis with `loadBalancerPolicy` `strategy: Cookie`.
+
+```yaml
+# httpproxy-sticky-sessions.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: httpbin
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: httpbin.davecheney.com
+ routes:
+ - services:
+ - name: httpbin
+ port: 8080
+ loadBalancerPolicy:
+ strategy: Cookie
+```
+
+Session affinity is based on the premise that the backend servers are robust, do not change ordering, or grow and shrink according to load.
+None of these properties are guaranteed by a Kubernetes cluster and will be visible to applications that rely heavily on session affinity.
+
+Any perturbation in the set of pods backing a service risks redistributing backends around the hash ring.
+
+## Internal Redirects
+
+HTTPProxy supports handling 3xx redirects internally, that is capturing a configurable 3xx redirect response, synthesizing a new request, sending it to the upstream specified by the new route match, and returning the redirected response as the response to the original request.
+
+Internal redirects can be enabled in HTTPProxy by defining an `internalRedirectPolicy` on a route.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: myservice
+ namespace: prod
+spec:
+ virtualhost:
+ fqdn: foo.com
+ routes:
+ - conditions:
+ - prefix: /download
+ services:
+ - name: foo
+ port: 8080
+ internalRedirectPolicy:
+ maxInternalRedirects: 5
+ redirectResponseCodes: [ 302 ]
+ allowCrossSchemeRedirect: SafeOnly
+ denyRepeatedRouteRedirect: true
+```
+
+In this example, a sample redirect flow might look like this:
+
+1. Client sends a `GET` request for http://foo.com/download.
+2. Upstream `foo` returns a `302` response with `location: http://foo.com/myfile`.
+3. Envoy lookups a route for http://foo.com/myfile and sends a new `GET` request to the corresponding upstream with the additional request header `x-envoy-original-url: http://foo.com/download`.
+4. Envoy proxies the response data for http://foo.com/myfile to the client as the response to the original request.
+
+See [the API specification][9] and [Envoy's documentation][10] for more detail.
+
+[3]: /docs/{{< param version >}}/config/api/#projectcontour.io/v1.HTTPRequestRedirectPolicy
+[4]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-routeaction-timeout
+[5]: https://godoc.org/time#ParseDuration
+[6]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/route/v3/route_components.proto#envoy-v3-api-field-config-route-v3-routeaction-idle-timeout
+[7]: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/upstream/load_balancing/overview
+[8]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/core/v3/protocol.proto#envoy-v3-api-field-config-core-v3-httpprotocoloptions-idle-timeout
+[9] /docs/{{< param version >}}/config/api/#projectcontour.io/v1.HTTPInternalRedirectPolicy
+[10] https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/http/http_connection_management.html#internal-redirects
diff --git a/site/content/docs/1.28/config/slow-start.md b/site/content/docs/1.28/config/slow-start.md
new file mode 100644
index 00000000000..b44cc18fdc3
--- /dev/null
+++ b/site/content/docs/1.28/config/slow-start.md
@@ -0,0 +1,39 @@
+# Slow Start Mode
+
+Slow start mode is a configuration setting that is used to gradually increase the amount of traffic targeted to a newly added upstream endpoint.
+By default, the amount of traffic will increase linearly for the duration of time window set by `window` field, starting from 10% of the target load balancing weight and increasing to 100% gradually.
+The easing function for the traffic increase can be adjusted by setting optional field `aggression`.
+A value above 1.0 results in a more aggressive increase initially, slowing down when nearing the end of the time window.
+Value below 1.0 results in slow initial increase, picking up speed when nearing the end of the time window.
+Optional field `minWeightPercent` can be set to change the minimum percent of target weight.
+It is used to avoid too small new weight, which may cause endpoint to receive no traffic in beginning of the slow start window.
+
+Slow start mode can be useful for example with JVM based applications, that might otherwise get overwhelmed during JIT warm-up period.
+Such applications may respond to requests slowly or return errors immediately after pod start or after container restarts.
+User impact of this behavior can be mitigated by using slow start configuration to gradually increase traffic to recently started service endpoints.
+
+The following example configures slow start mode for a service:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: slow-start
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ routes:
+ - services:
+ - name: java-app
+ port: 80
+ slowStartPolicy:
+ window: 3s
+ aggression: "1.0"
+ minWeightPercent: 10
+```
+
+Slow start mode works only with `RoundRobin` and `WeightedLeastRequest` [load balancing strategies][2].
+For more details see [Envoy documentation][1].
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/upstream/load_balancing/slow_start
+[2]: api/#projectcontour.io/v1.LoadBalancerPolicy
diff --git a/site/content/docs/1.28/config/tls-delegation.md b/site/content/docs/1.28/config/tls-delegation.md
new file mode 100644
index 00000000000..155796fe7eb
--- /dev/null
+++ b/site/content/docs/1.28/config/tls-delegation.md
@@ -0,0 +1,79 @@
+# TLS Certificate Delegation
+
+In order to support wildcard certificates, TLS certificates for a `*.somedomain.com`, which are stored in a namespace controlled by the cluster administrator, Contour supports a facility known as TLS Certificate Delegation.
+This facility allows the owner of a TLS certificate to delegate, for the purposes of referencing the TLS certificate, permission to Contour to read the Secret object from another namespace.
+Delegation works for both HTTPProxy and Ingress resources, however it needs an annotation to work with Ingress v1.
+
+If the `--watch-namespaces` configuration flag is used, it must define all namespaces that will be referenced by the delegation.
+
+The [`TLSCertificateDelegation`][1] resource defines a set of `delegations` in the `spec`.
+Each delegation references a `secretName` from the namespace where the `TLSCertificateDelegation` is created as well as describing a set of `targetNamespaces` in which the certificate can be referenced.
+If all namespaces should be able to reference the secret, then set `"*"` as the value of `targetNamespaces` (see example below).
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: TLSCertificateDelegation
+metadata:
+ name: example-com-wildcard
+ namespace: www-admin
+spec:
+ delegations:
+ - secretName: example-com-wildcard
+ targetNamespaces:
+ - example-com
+ - secretName: another-com-wildcard
+ targetNamespaces:
+ - "*"
+```
+
+In this example, the permission for Contour to reference the Secret `example-com-wildcard` in the `www-admin` namespace has been delegated to HTTPProxy and Ingress objects in the `example-com` namespace.
+Also, the permission for Contour to reference the Secret `another-com-wildcard` from all namespaces has been delegated to all HTTPProxy and Ingress objects in the cluster.
+
+To reference the secret from an HTTPProxy or Ingress v1beta1 you must use the slash syntax in the `secretName`:
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: www
+ namespace: example-com
+spec:
+ virtualhost:
+ fqdn: foo2.bar.com
+ tls:
+ secretName: www-admin/example-com-wildcard
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+To reference the secret from an Ingress v1 you must use the `projectcontour.io/tls-cert-namespace` annotation:
+```yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ annotations:
+ projectcontour.io/tls-cert-namespace: www-admin
+ name: www
+ namespace: example-com
+spec:
+ rules:
+ - host: foo2.bar.com
+ http:
+ paths:
+ - path: /
+ pathType: Prefix
+ backend:
+ service:
+ name: s1
+ port:
+ number: 80
+ tls:
+ - hosts:
+ - foo2.bar.com
+ secretName: example-com-wildcard
+```
+
+
+[0]: https://github.com/projectcontour/contour/issues/3544
+[1]: /docs/{{< param version >}}/config/api/#projectcontour.io/v1.TLSCertificateDelegation
diff --git a/site/content/docs/1.28/config/tls-termination.md b/site/content/docs/1.28/config/tls-termination.md
new file mode 100644
index 00000000000..d1b26dc2f4e
--- /dev/null
+++ b/site/content/docs/1.28/config/tls-termination.md
@@ -0,0 +1,353 @@
+# TLS Termination
+
+HTTPProxy follows a similar pattern to Ingress for configuring TLS credentials.
+
+You can secure a HTTPProxy by specifying a Secret that contains TLS private key and certificate information.
+If multiple HTTPProxies utilize the same Secret, the certificate must include the necessary Subject Authority Name (SAN) for each fqdn.
+
+Contour (via Envoy) requires that clients send the Server Name Indication (SNI) TLS extension so that requests can be routed to the correct virtual host.
+Virtual hosts are strongly bound to SNI names.
+This means that the Host header in HTTP requests must match the SNI name that was sent at the start of the TLS session.
+
+Contour also follows a "secure first" approach.
+When TLS is enabled for a virtual host, any request to the insecure port is redirected to the secure interface with a 301 redirect.
+Specific routes can be configured to override this behavior and handle insecure requests by enabling the `spec.routes.permitInsecure` parameter on a Route.
+
+The TLS secret must:
+- be a Secret of type `kubernetes.io/tls`. This means that it must contain keys named `tls.crt` and `tls.key` that contain the certificate and private key to use for TLS, in PEM format.
+
+The TLS secret may also:
+- add any chain CA certificates required for validation into the `tls.crt` PEM bundle. If this is the case, the serving certificate must be the first certificate in the bundle and the intermediate CA certificates must be appended in issuing order.
+
+```yaml
+# ingress-tls.secret.yaml
+apiVersion: v1
+data:
+ tls.crt: base64 encoded cert
+ tls.key: base64 encoded key
+kind: Secret
+metadata:
+ name: testsecret
+ namespace: default
+type: kubernetes.io/tls
+```
+
+The HTTPProxy can be configured to use this secret using `tls.secretName` property:
+
+```yaml
+# httpproxy-tls.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: tls-example
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: foo2.bar.com
+ tls:
+ secretName: testsecret
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+If the `tls.secretName` property contains a slash, eg. `somenamespace/somesecret` then, subject to TLS Certificate Delegation, the TLS certificate will be read from `somesecret` in `somenamespace`.
+See TLS Certificate Delegation below for more information.
+
+The TLS **Minimum Protocol Version** a virtual host should negotiate can be specified by setting the `spec.virtualhost.tls.minimumProtocolVersion`:
+
+- 1.3
+- 1.2 (Default)
+
+## Fallback Certificate
+
+Contour provides virtual host based routing, so that any TLS request is routed to the appropriate service based on both the server name requested by the TLS client and the HOST header in the HTTP request.
+
+Since the HOST Header is encrypted during TLS handshake, it can’t be used for virtual host based routing unless the client sends HTTPS requests specifying hostname using the TLS server name, or the request is first decrypted using a default TLS certificate.
+
+Some legacy TLS clients do not send the server name, so Envoy does not know how to select the right certificate. A fallback certificate is needed for these clients.
+
+_**Note:**
+The minimum TLS protocol version for any fallback request is defined by the `minimum TLS protocol version` set in the Contour configuration file.
+Enabling the fallback certificate is not compatible with TLS client authentication._
+
+### Fallback Certificate Configuration
+
+First define the `namespace/name` in the [Contour configuration file][1] of a Kubernetes secret which will be used as the fallback certificate.
+Any HTTPProxy which enables fallback certificate delegation must have the fallback certificate delegated to the namespace in which the HTTPProxy object resides.
+
+To do that, configure `TLSCertificateDelegation` to delegate the fallback certificate to specific or all namespaces (e.g. `*`) which should be allowed to enable the fallback certificate.
+Finally, for each root HTTPProxy, set the `Spec.TLS.enableFallbackCertificate` parameter to allow that HTTPProxy to opt-in to the fallback certificate routing.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: fallback-tls-example
+ namespace: defaultub
+spec:
+ virtualhost:
+ fqdn: fallback.bar.com
+ tls:
+ secretName: testsecret
+ enableFallbackCertificate: true
+ routes:
+ - services:
+ - name: s1
+ port: 80
+---
+apiVersion: projectcontour.io/v1
+kind: TLSCertificateDelegation
+metadata:
+ name: fallback-delegation
+ namespace: www-admin
+spec:
+ delegations:
+ - secretName: fallback-secret-name
+ targetNamespaces:
+ - "*"
+```
+
+## Permitting Insecure Requests
+
+A HTTPProxy can be configured to permit insecure requests to specific Routes.
+In this example, any request to `foo2.bar.com/blog` will not receive a 301 redirect to HTTPS, but the `/` route will:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: tls-example-insecure
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: foo2.bar.com
+ tls:
+ secretName: testsecret
+ routes:
+ - services:
+ - name: s1
+ port: 80
+ - conditions:
+ - prefix: /blog
+ permitInsecure: true
+ services:
+ - name: s2
+ port: 80
+```
+
+## Client Certificate Validation
+
+It is possible to protect the backend service from unauthorized external clients by requiring the client to present a valid TLS certificate.
+Envoy will validate the client certificate by verifying that it is not expired and that a chain of trust can be established to the configured trusted root CA certificate.
+Only those requests with a valid client certificate will be accepted and forwarded to the backend service.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-client-auth
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+The preceding example enables validation by setting the optional `clientValidation` attribute.
+Its mandatory attribute `caSecret` contains a name of an existing Kubernetes Secret that must be of type "Opaque" and have only a data key named `ca.crt`.
+The data value of the key `ca.crt` must be a PEM-encoded certificate bundle and it must contain all the trusted CA certificates that are to be used for validating the client certificate.
+If the Opaque Secret also contains one of either `tls.crt` or `tls.key` keys, it will be ignored.
+
+By default, client certificates are required but some applications might support different authentication schemes. In that case you can set the `optionalClientCertificate` field to `true`. A client certificate will be requested, but the connection is allowed to continue if the client does not provide one. If a client certificate is sent, it will be verified according to the other properties, which includes disabling validations if `skipClientCertValidation` is set.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-optional-client-auth
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ optionalClientCertificate: true
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+When using external authorization, it may be desirable to use an external authorization server to validate client certificates on requests, rather than the Envoy proxy.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-client-auth-and-ext-authz
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ authorization:
+ # external authorization server configuration
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ skipClientCertValidation: true
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+In the above example, setting the `skipClientCertValidation` field to `true` will configure Envoy to require client certificates on requests and pass them along to a configured authorization server.
+Failed validation of client certificates by Envoy will be ignored and the `fail_verify_error` [Listener statistic][2] incremented.
+If the `caSecret` field is omitted, Envoy will request but not require client certificates to be present on requests.
+
+Optionally, you can enable certificate revocation check by providing one or more Certificate Revocation Lists (CRLs).
+Attribute `crlSecret` contains a name of an existing Kubernetes Secret that must be of type "Opaque" and have a data key named `crl.pem`.
+The data value of the key `crl.pem` must be one or more PEM-encoded CRLs concatenated together.
+Large CRL lists are not supported since individual Secrets are limited to 1MiB in size.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-client-auth-and-crl-check
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ crlSecret: client-crl
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+CRLs must be available from all relevant CAs, including intermediate CAs.
+Otherwise clients will be denied access, since the revocation status cannot be checked for the full certificate chain.
+This behavior can be controlled by `crlOnlyVerifyLeafCert` field.
+If the option is set to `true`, only the certificate at the end of the certificate chain will be subject to validation by CRL.
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-client-auth-and-crl-check-only-leaf
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ crlSecret: client-crl
+ crlOnlyVerifyLeafCert: true
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+## Client Certificate Details Forwarding
+
+HTTPProxy supports passing certificate data through the `x-forwarded-client-cert` header to let applications use details from client certificates (e.g. Subject, SAN...). Since the certificate (or the certificate chain) could exceed the web server header size limit, you have the ability to select what specific part of the certificate to expose in the header through the `forwardClientCertificate` field. Read more about the supported values in the [Envoy documentation](https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_conn_man/headers#x-forwarded-client-cert).
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: with-client-auth
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ tls:
+ secretName: secret
+ clientValidation:
+ caSecret: client-root-ca
+ forwardClientCertificate:
+ subject: true
+ cert: true
+ chain: true
+ dns: true
+ uri: true
+ routes:
+ - services:
+ - name: s1
+ port: 80
+```
+
+## TLS Session Proxying
+
+HTTPProxy supports proxying of TLS encapsulated TCP sessions.
+
+_Note_: The TCP session must be encrypted with TLS.
+This is necessary so that Envoy can use SNI to route the incoming request to the correct service.
+
+If `spec.virtualhost.tls.secretName` is present then that secret will be used to decrypt the TCP traffic at the edge.
+
+```yaml
+# httpproxy-tls-termination.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: example
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: tcp.example.com
+ tls:
+ secretName: secret
+ tcpproxy:
+ services:
+ - name: tcpservice
+ port: 8080
+ - name: otherservice
+ port: 9999
+ weight: 20
+```
+
+The `spec.tcpproxy` key indicates that this _root_ HTTPProxy will forward the de-encrypted TCP traffic to the backend service.
+
+### TLS Session Passthrough
+
+If you wish to handle the TLS handshake at the backend service set `spec.virtualhost.tls.passthrough: true` indicates that once SNI demuxing is performed, the encrypted connection will be forwarded to the backend service.
+The backend service is expected to have a key which matches the SNI header received at the edge, and be capable of completing the TLS handshake. This is called SSL/TLS Passthrough.
+
+```yaml
+# httpproxy-tls-passthrough.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: example
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: tcp.example.com
+ tls:
+ passthrough: true
+ tcpproxy:
+ services:
+ - name: tcpservice
+ port: 8080
+ - name: otherservice
+ port: 9999
+ weight: 20
+```
+
+[1]: ../configuration#fallback-certificate
+[2]: https://www.envoyproxy.io/docs/envoy/latest/configuration/listeners/stats#tls-statistics
diff --git a/site/content/docs/1.28/config/tracing.md b/site/content/docs/1.28/config/tracing.md
new file mode 100644
index 00000000000..5500c26d547
--- /dev/null
+++ b/site/content/docs/1.28/config/tracing.md
@@ -0,0 +1,117 @@
+# Tracing Support
+
+- [Overview](#overview)
+- [Tracing-config](#tracing-config)
+
+## Overview
+
+Envoy has rich support for [distributed tracing][1],and supports exporting data to third-party providers (Zipkin, Jaeger, Datadog, etc.)
+
+[OpenTelemetry][2] is a CNCF project which is working to become a standard in the space. It was formed as a merger of the OpenTracing and OpenCensus projects.
+
+Contour supports configuring envoy to export data to OpenTelemetry, and allows users to customize some configurations.
+
+- Custom service name, the default is `contour`.
+- Custom sampling rate, the default is `100`.
+- Custom the maximum length of the request path, the default is `256`.
+- Customize span tags from literal or request headers.
+- Customize whether to include the pod's hostname and namespace.
+
+## Tracing-config
+
+In order to use this feature, you must first select and deploy an opentelemetry-collector to receive the tracing data exported by envoy.
+
+First we should deploy an opentelemetry-collector to receive the tracing data exported by envoy
+```bash
+# install operator
+kubectl apply -f https://github.com/open-telemetry/opentelemetry-operator/releases/latest/download/opentelemetry-operator.yaml
+```
+
+Install an otel collector instance, with verbose logging exporter enabled:
+```shell
+kubectl apply -f - </`. If the CA secret's namespace is not the same namespace as the `HTTPProxy` resource, [TLS Certificate Delegation][4] must be used to allow the owner of the CA certificate secret to delegate, for the purposes of referencing the CA certificate in a different namespace, permission to Contour to read the Secret object from another namespace.
+
+_**Note:**
+If `spec.routes.services[].validation` is present, `spec.routes.services[].{name,port}` must point to a Service with a matching `projectcontour.io/upstream-protocol.tls` Service annotation._
+
+In the example below, the upstream service is named `secure-backend` and uses port `8443`:
+
+```yaml
+# httpproxy-example.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: example
+spec:
+ virtualhost:
+ fqdn: www.example.com
+ routes:
+ - services:
+ - name: secure-backend
+ port: 8443
+ validation:
+ caSecret: my-certificate-authority
+ subjectName: backend.example.com
+```
+
+```yaml
+# service-secure-backend.yaml
+apiVersion: v1
+kind: Service
+metadata:
+ name: secure-backend
+ annotations:
+ projectcontour.io/upstream-protocol.tls: "8443"
+spec:
+ ports:
+ - name: https
+ port: 8443
+ selector:
+ app: secure-backend
+
+```
+
+If the `validation` spec is defined on a service, but the secret which it references does not exist, Contour will reject the update and set the status of the HTTPProxy object accordingly.
+This helps prevent the case of proxying to an upstream where validation is requested, but not yet available.
+
+```yaml
+Status:
+ Current Status: invalid
+ Description: route "/": service "tls-nginx": upstreamValidation requested but secret not found or misconfigured
+```
+
+## Upstream Validation
+
+When defining upstream services on a route, it's possible to configure the connection from Envoy to the backend endpoint to communicate over TLS.
+
+A CA certificate and a Subject Name must be provided, which are both used to verify the backend endpoint's identity.
+
+If specifying multiple Subject Names, `SubjectNames` and `SubjectName` must be configured such that `SubjectNames[0] == SubjectName`.
+
+The CA certificate bundle for the backend service should be supplied in a Kubernetes Secret.
+The referenced Secret must be of type "Opaque" and have a data key named `ca.crt`.
+This data value must be a PEM-encoded certificate bundle.
+
+In addition to the CA certificate and the subject name, the Kubernetes service must also be annotated with a Contour specific annotation: `projectcontour.io/upstream-protocol.tls: ` ([see annotations section][1]).
+
+_**Note:** This annotation is applied to the Service not the Ingress or HTTPProxy object._
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: blog
+ namespace: marketing
+spec:
+ routes:
+ - services:
+ - name: s2
+ port: 80
+ validation:
+ caSecret: foo-ca-cert
+ subjectName: foo.marketing
+ subjectNames:
+ - foo.marketing
+ - bar.marketing
+```
+
+## Envoy Client Certificate
+
+Contour can be configured with a `namespace/name` in the [Contour configuration file][3] of a Kubernetes secret which Envoy uses as a client certificate when upstream TLS is configured for the backend.
+Envoy will send the certificate during TLS handshake when the backend applications request the client to present its certificate.
+Backend applications can validate the certificate to ensure that the connection is coming from Envoy.
+
+[1]: annotations.md
+[2]: api/#projectcontour.io/v1.Service
+[3]: ../configuration#fallback-certificate
+[4]: tls-delegation.md
diff --git a/site/content/docs/1.28/config/virtual-hosts.md b/site/content/docs/1.28/config/virtual-hosts.md
new file mode 100644
index 00000000000..b7a138dde6b
--- /dev/null
+++ b/site/content/docs/1.28/config/virtual-hosts.md
@@ -0,0 +1,138 @@
+# Virtual Hosts
+
+
+Similar to Ingress, HTTPProxy support name-based virtual hosting.
+Name-based virtual hosts use multiple host names with the same IP address.
+
+```
+foo.bar.com --| |-> foo.bar.com s1:80
+ | 178.91.123.132 |
+bar.foo.com --| |-> bar.foo.com s2:80
+```
+
+Unlike Ingress however, HTTPProxy only support a single root domain per HTTPProxy object.
+As an example, this Ingress object:
+
+```yaml
+# ingress-name.yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: name-example
+spec:
+ rules:
+ - host: foo1.bar.com
+ http:
+ paths:
+ - backend:
+ service:
+ name: s1
+ port:
+ number: 80
+ pathType: Prefix
+ - host: bar1.bar.com
+ http:
+ paths:
+ - backend:
+ service:
+ name: s2
+ port:
+ number: 80
+ pathType: Prefix
+```
+
+must be represented by two different HTTPProxy objects:
+
+```yaml
+# httpproxy-name.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: name-example-foo
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: foo1.bar.com
+ routes:
+ - services:
+ - name: s1
+ port: 80
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: name-example-bar
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: bar1.bar.com
+ routes:
+ - services:
+ - name: s2
+ port: 80
+```
+
+A HTTPProxy object that contains a [`virtualhost`][2] field is known as a "root proxy".
+
+## Virtualhost aliases
+
+To present the same set of routes under multiple DNS entries (e.g. `www.example.com` and `example.com`), including a service with a `prefix` condition of `/` can be used.
+
+```yaml
+# httpproxy-inclusion-multipleroots.yaml
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-root
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: bar.com
+ includes:
+ - name: main
+ namespace: default
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: multiple-root-www
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: www.bar.com
+ includes:
+ - name: main
+ namespace: default
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: main
+ namespace: default
+spec:
+ routes:
+ - services:
+ - name: s2
+ port: 80
+```
+
+## Restricted root namespaces
+
+HTTPProxy inclusion allows Administrators to limit which users/namespaces may configure routes for a given domain, but it does not restrict where root HTTPProxies may be created.
+Contour has an enforcing mode which accepts a list of namespaces where root HTTPProxy are valid.
+Only users permitted to operate in those namespaces can therefore create HTTPProxy with the [`virtualhost`] field ([see API docs][2]).
+
+This restricted mode is enabled in Contour by specifying a command line flag, `--root-namespaces`, which will restrict Contour to only searching the defined namespaces for root HTTPProxy. This CLI flag accepts a comma separated list of namespaces where HTTPProxy are valid (e.g. `--root-namespaces=default,kube-system,my-admin-namespace`).
+
+HTTPProxy with a defined [virtualhost][2] field that are not in one of the allowed root namespaces will be flagged as `invalid` and will be ignored by Contour.
+
+Additionally, when defined, Contour will only watch for Kubernetes secrets in these namespaces ignoring changes in all other namespaces.
+Proper RBAC rules should also be created to restrict what namespaces Contour has access matching the namespaces passed to the command line flag.
+An example of this is included in the [examples directory][1] and shows how you might create a namespace called `root-httproxy`.
+
+_**Note:** The restricted root namespace feature is only supported for HTTPProxy CRDs.
+`--root-namespaces` does not affect the operation of Ingress objects. In order to limit other resources, see the `--watch-namespaces` configuration flag._
+
+[1]: {{< param github_url>}}/tree/{{< param branch >}}/examples/root-rbac
+[2]: api/#projectcontour.io/v1.VirtualHost
diff --git a/site/content/docs/1.28/config/websockets.md b/site/content/docs/1.28/config/websockets.md
new file mode 100644
index 00000000000..136c0468378
--- /dev/null
+++ b/site/content/docs/1.28/config/websockets.md
@@ -0,0 +1,27 @@
+# Websockets
+
+WebSocket support can be enabled on specific routes using the `enableWebsockets` field:
+
+```yaml
+# httpproxy-websockets.yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: chat
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: chat.example.com
+ routes:
+ - services:
+ - name: chat-app
+ port: 80
+ - conditions:
+ - prefix: /websocket
+ enableWebsockets: true # Setting this to true enables websocket for all paths that match /websocket
+ services:
+ - name: chat-app
+ port: 80
+```
+
+If you are using Gateway API, websockets are enabled by default at the Listener level.
diff --git a/site/content/docs/1.28/configuration.md b/site/content/docs/1.28/configuration.md
new file mode 100644
index 00000000000..f145d67f09c
--- /dev/null
+++ b/site/content/docs/1.28/configuration.md
@@ -0,0 +1,541 @@
+# Contour Configuration Reference
+
+- [Serve Flags](#serve-flags)
+- [Configuration File](#configuration-file)
+- [Environment Variables](#environment-variables)
+- [Bootstrap Config File](#bootstrap-config-file)
+
+## Overview
+
+There are various ways to configure Contour, flags, the configuration file, as well as environment variables.
+Contour has a precedence of configuration for contour serve, meaning anything configured in the config file is overridden by environment vars which are overridden by cli flags.
+
+## Serve Flags
+
+The `contour serve` command is the main command which is used to watch for Kubernetes resource and process them into Envoy configuration which is then streamed to any Envoy via its xDS gRPC connection.
+There are a number of flags that can be passed to this command which further configures how Contour operates.
+Many of these flags are mirrored in the [Contour Configuration File](#configuration-file).
+
+| Flag Name | Description |
+| --------------------------------------------------------------- | --------------------------------------------------------------------------------------- |
+| `--config-path` | Path to base configuration |
+| `--contour-config-name` | Name of the ContourConfiguration resource to use |
+| `--incluster` | Use in cluster configuration |
+| `--kubeconfig=` | Path to kubeconfig (if not in running inside a cluster) |
+| `--xds-address=` | xDS gRPC API address |
+| `--xds-port=` | xDS gRPC API port |
+| `--stats-address=` | Envoy /stats interface address |
+| `--stats-port=` | Envoy /stats interface port |
+| `--debug-http-address=` | Address the debug http endpoint will bind to. |
+| `--debug-http-port=` | Port the debug http endpoint will bind to |
+| `--http-address=` | Address the metrics HTTP endpoint will bind to |
+| `--http-port=` | Port the metrics HTTP endpoint will bind to. |
+| `--health-address=` | Address the health HTTP endpoint will bind to |
+| `--health-port=` | Port the health HTTP endpoint will bind to |
+| `--contour-cafile=` | CA bundle file name for serving gRPC with TLS |
+| `--contour-cert-file=` | Contour certificate file name for serving gRPC over TLS |
+| `--contour-key-file=` | Contour key file name for serving gRPC over TLS |
+| `--insecure` | Allow serving without TLS secured gRPC |
+| `--root-namespaces=` | Restrict contour to searching these namespaces for root ingress routes |
+| `--watch-namespaces=` | Restrict contour to searching these namespaces for all resources |
+| `--ingress-class-name=` | Contour IngressClass name (comma-separated list allowed) |
+| `--ingress-status-address=` | Address to set in Ingress object status |
+| `--envoy-http-access-log=` | Envoy HTTP access log |
+| `--envoy-https-access-log=` | Envoy HTTPS access log |
+| `--envoy-service-http-address=` | Kubernetes Service address for HTTP requests |
+| `--envoy-service-https-address=` | Kubernetes Service address for HTTPS requests |
+| `--envoy-service-http-port=` | Kubernetes Service port for HTTP requests |
+| `--envoy-service-https-port=` | Kubernetes Service port for HTTPS requests |
+| `--envoy-service-name=` | Name of the Envoy service to inspect for Ingress status details. |
+| `--envoy-service-namespace=` | Envoy Service Namespace |
+| `--use-proxy-protocol` | Use PROXY protocol for all listeners |
+| `--accesslog-format=` | Format for Envoy access logs |
+| `--disable-leader-election` | Disable leader election mechanism |
+| `--disable-feature=` | Do not start an informer for the specified resources. Flag can be given multiple times. |
+| `--leader-election-lease-duration` | The duration of the leadership lease. |
+| `--leader-election-renew-deadline` | The duration leader will retry refreshing leadership before giving up. |
+| `--leader-election-retry-period` | The interval which Contour will attempt to acquire leadership lease. |
+| `--leader-election-resource-name` | The name of the resource (Lease) leader election will lease. |
+| `--leader-election-resource-namespace` | The namespace of the resource (Lease) leader election will lease. |
+| `-d, --debug` | Enable debug logging |
+| `--kubernetes-debug=` | Enable Kubernetes client debug logging |
+| `--log-format=` | Log output format for Contour. Either text (default) or json. |
+| `--kubernetes-client-qps=` | QPS allowed for the Kubernetes client. |
+| `--kubernetes-client-burst=` | Burst allowed for the Kubernetes client. |
+
+## Configuration File
+
+A configuration file can be passed to the `--config-path` argument of the `contour serve` command to specify additional configuration to Contour.
+In most deployments, this file is passed to Contour via a ConfigMap which is mounted as a volume to the Contour pod.
+
+The Contour configuration file is optional.
+In its absence, Contour will operate with reasonable defaults.
+Where Contour settings can also be specified with command-line flags, the command-line value takes precedence over the configuration file.
+
+| Field Name | Type | Default | Description |
+|---------------------------| ---------------------- |------------------------------------------------------------------------------------------------------| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| accesslog-format | string | `envoy` | This key sets the global [access log format][2] for Envoy. Valid options are `envoy` or `json`. |
+| accesslog-format-string | string | None | If present, this specifies custom access log format for Envoy. See [Envoy documentation](https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage) for more information about the syntax. This field only has effect if `accesslog-format` is `envoy` |
+| accesslog-level | string | `info` | This field specifies the verbosity level of the access log. Valid options are `info` (default, all requests are logged), `error` (all non-success, i.e. 300+ response code, requests are logged), `critical` (all server error, i.e. 500+ response code, requests are logged) and `disabled`. |
+| debug | boolean | `false` | Enables debug logging. |
+| default-http-versions | string array | HTTP/1.1
HTTP/2
| This array specifies the HTTP versions that Contour should program Envoy to serve. HTTP versions are specified as strings of the form "HTTP/x", where "x" represents the version number. |
+| disableAllowChunkedLength | boolean | `false` | If this field is true, Contour will disable the RFC-compliant Envoy behavior to strip the `Content-Length` header if `Transfer-Encoding: chunked` is also set. This is an emergency off-switch to revert back to Envoy's default behavior in case of failures.
+| disableMergeSlashes | boolean | `false` | This field disables Envoy's non-standard merge_slashes path transformation behavior that strips duplicate slashes from request URL paths.
+| serverHeaderTransformation | string | `overwrite` | This field defines the action to be applied to the Server header on the response path. Values: `overwrite` (default), `append_if_absent`, `pass_through`
+| disablePermitInsecure | boolean | `false` | If this field is true, Contour will ignore `PermitInsecure` field in HTTPProxy documents. |
+| envoy-service-name | string | `envoy` | This sets the service name that will be inspected for address details to be applied to Ingress objects. |
+| envoy-service-namespace | string | `projectcontour` | This sets the namespace of the service that will be inspected for address details to be applied to Ingress objects. If the `CONTOUR_NAMESPACE` environment variable is present, Contour will populate this field with its value. |
+| ingress-status-address | string | None | If present, this specifies the address that will be copied into the Ingress status for each Ingress that Contour manages. It is exclusive with `envoy-service-name` and `envoy-service-namespace`. |
+| incluster | boolean | `false` | This field specifies that Contour is running in a Kubernetes cluster and should use the in-cluster client access configuration. |
+| json-fields | string array | [fields][5] | This is the list the field names to include in the JSON [access log format][2]. This field only has effect if `accesslog-format` is `json`. |
+| kubeconfig | string | `$HOME/.kube/config` | Path to a Kubernetes [kubeconfig file][3] for when Contour is executed outside a cluster. |
+| kubernetesClientQPS | float32 | | QPS allowed for the Kubernetes client. |
+| kubernetesClientBurst | int | | Burst allowed for the Kubernetes client. |
+| policy | PolicyConfig | | The default [policy configuration](#policy-configuration). |
+| tls | TLS | | The default [TLS configuration](#tls-configuration). |
+| timeouts | TimeoutConfig | | The [timeout configuration](#timeout-configuration). |
+| cluster | ClusterConfig | | The [cluster configuration](#cluster-configuration). |
+| network | NetworkConfig | | The [network configuration](#network-configuration). |
+| listener | ListenerConfig | | The [listener configuration](#listener-configuration). |
+| server | ServerConfig | | The [server configuration](#server-configuration) for `contour serve` command. |
+| gateway | GatewayConfig | | The [gateway-api Gateway configuration](#gateway-configuration). |
+| rateLimitService | RateLimitServiceConfig | | The [rate limit service configuration](#rate-limit-service-configuration). |
+| enableExternalNameService | boolean | `false` | Enable ExternalName Service processing. Enabling this has security implications. Please see the [advisory](https://github.com/projectcontour/contour/security/advisories/GHSA-5ph6-qq5x-7jwc) for more details. |
+| metrics | MetricsParameters | | The [metrics configuration](#metrics-configuration) |
+| featureFlags | string array | `[]` | Defines the toggle to enable new contour features. Available toggles are:
1. `useEndpointSlices` - configures contour to fetch endpoint data from k8s endpoint slices. |
+
+### TLS Configuration
+
+The TLS configuration block can be used to configure default values for how
+Contour should provision TLS hosts.
+
+| Field Name | Type | Default | Description |
+| ------------------------ | -------- | ----------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| minimum-protocol-version | string | `1.2` | This field specifies the minimum TLS protocol version that is allowed. Valid options are `1.2` (default) and `1.3`. Any other value defaults to TLS 1.2.
+| maximum-protocol-version | string | `1.3` | This field specifies the maximum TLS protocol version that is allowed. Valid options are `1.2` and `1.3`. Any other value defaults to TLS 1.3. |
+| fallback-certificate | | | [Fallback certificate configuration](#fallback-certificate). |
+| envoy-client-certificate | | | [Client certificate configuration for Envoy](#envoy-client-certificate). |
+| cipher-suites | []string | See [config package documentation](https://pkg.go.dev/github.com/projectcontour/contour/pkg/config#pkg-variables) | This field specifies the TLS ciphers to be supported by TLS listeners when negotiating TLS 1.2. This parameter should only be used by advanced users. Note that this is ignored when TLS 1.3 is in use. The set of ciphers that are allowed is a superset of those supported by default in stock, non-FIPS Envoy builds and FIPS builds as specified [here](https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/transport_sockets/tls/v3/common.proto#envoy-v3-api-field-extensions-transport-sockets-tls-v3-tlsparameters-cipher-suites). Custom ciphers not accepted by Envoy in a standard build are not supported. |
+
+### Upstream TLS Configuration
+
+The Upstream TLS configuration block can be used to configure default values for how Contour establishes TLS for upstream connections.
+
+| Field Name | Type | Default | Description |
+| ------------------------ | -------- | ----------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| minimum-protocol-version | string | `1.2` | This field specifies the minimum TLS protocol version that is allowed. Valid options are `1.2` (default) and `1.3`. Any other value defaults to TLS 1.2. |
+| maximum-protocol-version | string | `1.3` | This field specifies the maximum TLS protocol version that is allowed. Valid options are `1.2` and `1.3`. Any other value defaults to TLS 1.3. |
+| cipher-suites | []string | See [config package documentation](https://pkg.go.dev/github.com/projectcontour/contour/pkg/config#pkg-variables) | This field specifies the TLS ciphers to be supported by TLS listeners when negotiating TLS 1.2. This parameter should only be used by advanced users. Note that this is ignored when TLS 1.3 is in use. The set of ciphers that are allowed is a superset of those supported by default in stock, non-FIPS Envoy builds and FIPS builds as specified [here](https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/transport_sockets/tls/v3/common.proto#envoy-v3-api-field-extensions-transport-sockets-tls-v3-tlsparameters-cipher-suites). Custom ciphers not accepted by Envoy in a standard build are not supported. |
+
+### Fallback Certificate
+
+| Field Name | Type | Default | Description |
+| ---------- | ------ | ------- | ----------------------------------------------------------------------------------------------- |
+| name | string | `""` | This field specifies the name of the Kubernetes secret to use as the fallback certificate. |
+| namespace | string | `""` | This field specifies the namespace of the Kubernetes secret to use as the fallback certificate. |
+
+
+### Envoy Client Certificate
+
+| Field Name | Type | Default | Description |
+| ---------- | ------ | ------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| name | string | `""` | This field specifies the name of the Kubernetes secret to use as the client certificate and private key when establishing TLS connections to the backend service. |
+| namespace | string | `""` | This field specifies the namespace of the Kubernetes secret to use as the client certificate and private key when establishing TLS connections to the backend service. |
+
+
+### Timeout Configuration
+
+The timeout configuration block can be used to configure various timeouts for the proxies. All fields are optional; Contour/Envoy defaults apply if a field is not specified.
+
+| Field Name | Type | Default | Description |
+| -------------------------------- | ------ | ------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| request-timeout | string | none* | This field specifies the default request timeout. Note that this is a timeout for the entire request, not an idle timeout. Must be a [valid Go duration string][4], or omitted or set to `infinity` to disable the timeout entirely. See [the Envoy documentation][12] for more information.
_Note: A value of `0s` previously disabled this timeout entirely. This is no longer the case. Use `infinity` or omit this field to disable the timeout._ |
+| connection-idle-timeout | string | `60s` | This field defines how long the proxy should wait while there are no active requests (for HTTP/1.1) or streams (for HTTP/2) before terminating an HTTP connection. The timeout applies to downstream connections only. Must be a [valid Go duration string][4], or `infinity` to disable the timeout entirely. See [the Envoy documentation][8] for more information. |
+| stream-idle-timeout | string | `5m`* | This field defines how long the proxy should wait while there is no activity during single request/response (for HTTP/1.1) or stream (for HTTP/2). Timeout will not trigger while HTTP/1.1 connection is idle between two consecutive requests. Must be a [valid Go duration string][4], or `infinity` to disable the timeout entirely. See [the Envoy documentation][9] for more information. |
+| max-connection-duration | string | none* | This field defines the maximum period of time after an HTTP connection has been established from the client to the proxy before it is closed by the proxy, regardless of whether there has been activity or not. Must be a [valid Go duration string][4], or omitted or set to `infinity` for no max duration. See [the Envoy documentation][10] for more information. |
+| delayed-close-timeout | string | `1s`* | *Note: this is an advanced setting that should not normally need to be tuned.*
This field defines how long envoy will wait, once connection close processing has been initiated, for the downstream peer to close the connection before Envoy closes the socket associated with the connection. Setting this timeout to 'infinity' will disable it. See [the Envoy documentation][13] for more information. |
+| connection-shutdown-grace-period | string | `5s`* | This field defines how long the proxy will wait between sending an initial GOAWAY frame and a second, final GOAWAY frame when terminating an HTTP/2 connection. During this grace period, the proxy will continue to respond to new streams. After the final GOAWAY frame has been sent, the proxy will refuse new streams. Must be a [valid Go duration string][4]. See [the Envoy documentation][11] for more information. |
+| connect-timeout | string | `2s` | This field defines how long the proxy will wait for the upstream connection to be established.
+
+_This is Envoy's default setting value and is not explicitly configured by Contour._
+
+### Cluster Configuration
+
+The cluster configuration block can be used to configure various parameters for Envoy clusters.
+
+| Field Name | Type | Default | Description |
+|-----------------------------------|--------|---------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| dns-lookup-family | string | auto | This field specifies the dns-lookup-family to use for upstream requests to externalName type Kubernetes services from an HTTPProxy route. Values are: `auto`, `v4`, `v6`, `all` |
+| max-requests-per-connection | int | none | This field specifies the maximum requests for upstream connections. If not specified, there is no limit |
+| circuit-breakers | [CircuitBreakers](#circuit-breakers) | none | This field specifies the default value for [circuit-breaker-annotations](https://projectcontour.io/docs/main/config/annotations/) for services that don't specify them. |
+| per-connection-buffer-limit-bytes | int | 1MiB* | This field specifies the soft limit on size of the cluster’s new connection read and write buffer. If not specified, Envoy defaults of 1MiB apply |
+| upstream-tls | UpstreamTLS | | [Upstream TLS configuration](#upstream-tls) |
+
+_This is Envoy's default setting value and is not explicitly configured by Contour._
+
+
+
+
+### Network Configuration
+
+The network configuration block can be used to configure various parameters network connections.
+
+| Field Name | Type | Default | Description |
+| ---------------- | ---- | ------- | ----------------------------------------------------------------------------------------------------------------------- |
+| num-trusted-hops | int | 0 | Configures the number of additional ingress proxy hops from the right side of the x-forwarded-for HTTP header to trust. |
+| admin-port | int | 9001 | Configures the Envoy Admin read-only listener on Envoy. Set to `0` to disable. |
+
+### Listener Configuration
+
+The listener configuration block can be used to configure various parameters for Envoy listener.
+
+| Field Name | Type | Default | Description |
+|-----------------------------------|--------|---------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| connection-balancer | string | `""` | This field specifies the listener connection balancer. If the value is `exact`, the listener will use the exact connection balancer to balance connections between threads in a single Envoy process. See [the Envoy documentation][14] for more information. |
+| max-requests-per-connection | int | none | This field specifies the maximum requests for downstream connections. If not specified, there is no limit |
+| per-connection-buffer-limit-bytes | int | 1MiB* | This field specifies the soft limit on size of the listener’s new connection read and write buffer. If not specified, Envoy defaults of 1MiB apply |
+| socket-options | SocketOptions | | The [Socket Options](#socket-options) for Envoy listeners. |
+| max-requests-per-io-cycle | int | none | Defines the limit on number of HTTP requests that Envoy will process from a single connection in a single I/O cycle. Requests over this limit are processed in subsequent I/O cycles. Can be used as a mitigation for CVE-2023-44487 when abusive traffic is detected. Configures the `http.max_requests_per_io_cycle` Envoy runtime setting. The default value when this is not set is no limit. |
+| http2-max-concurrent-streams | int | none | Defines the value for SETTINGS_MAX_CONCURRENT_STREAMS Envoy will advertise in the SETTINGS frame in HTTP/2 connections and the limit for concurrent streams allowed for a peer on a single HTTP/2 connection. It is recommended to not set this lower than 100 but this field can be used to bound resource usage by HTTP/2 connections and mitigate attacks like CVE-2023-44487. The default value when this is not set is unlimited. |
+
+_This is Envoy's default setting value and is not explicitly configured by Contour._
+
+### Server Configuration
+
+The server configuration block can be used to configure various settings for the `contour serve` command.
+
+| Field Name | Type | Default | Description |
+| --------------- | ------ | ------- | ----------------------------------------------------------------------------- |
+| xds-server-type | string | contour | This field specifies the xDS Server to use. Options are `contour` or `envoy`. |
+
+### Gateway Configuration
+
+The gateway configuration block is used to configure which gateway-api Gateway Contour should configure:
+
+| Field Name | Type | Default | Description |
+| -------------- | -------------- | ------- | ------------------------------------------------------------------------------ |
+| controllerName | string | | **DEPRECATED**: Use `gatewayRef` or the Gateway provisioner instead. This field will be removed in a future release. Gateway Class controller name (i.e. projectcontour.io/gateway-controller). If set, Contour will reconcile the oldest GatewayClass, and its oldest Gateway, with this controller string. Only one of `controllerName` or `gatewayRef` must be set. |
+| gatewayRef | NamespacedName | | [Gateway namespace and name](#gateway-ref). If set, Contour will reconcile this specific Gateway. Only one of `controllerName` or `gatewayRef` must be set. |
+
+### Gateway Ref
+
+| Field Name | Type | Default | Description |
+| ---------- | ------ | ------- | ----------------------------------------------------------------------------------------------- |
+| name | string | `""` | This field specifies the name of the specific Gateway to reconcile. |
+| namespace | string | `""` | This field specifies the namespace of the specific Gateway to reconcile. |
+
+### Policy Configuration
+
+The Policy configuration block can be used to configure default policy values
+that are set if not overridden by the user.
+
+The `request-headers` field is used to rewrite headers on a HTTP request, and
+the `response-headers` field is used to rewrite headers on a HTTP response.
+
+| Field Name | Type | Default | Description |
+| ---------------- | ------------ | ------- | ------------------------------------------------------------------------------------------------- |
+| request-headers | HeaderPolicy | none | The default request headers set or removed on all service routes if not overridden in the object |
+| response-headers | HeaderPolicy | none | The default response headers set or removed on all service routes if not overridden in the object |
+| applyToIngress | Boolean | false | Whether the global policy should apply to Ingress objects |
+
+#### HeaderPolicy
+
+The `set` field sets an HTTP header value, creating it if it doesn't already exist but not overwriting it if it does.
+The `remove` field removes an HTTP header.
+
+| Field Name | Type | Default | Description |
+| ---------- | ----------------- | ------- | ------------------------------------------------------------------------------- |
+| set | map[string]string | none | Map of headers to set on all service routes if not overridden in the object |
+| remove | []string | none | List of headers to remove on all service routes if not overridden in the object |
+
+Note: the values of entries in the `set` and `remove` fields can be overridden in HTTPProxy objects but it it not possible to remove these entries.
+
+### Rate Limit Service Configuration
+
+The rate limit service configuration block is used to configure an optional global rate limit service:
+
+| Field Name | Type | Default | Description |
+|-----------------------------| ------ | ------- |------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| extensionService | string | | This field identifies the extension service defining the rate limit service, formatted as /. |
+| domain | string | contour | This field defines the rate limit domain value to pass to the rate limit service. Acts as a container for a set of rate limit definitions within the RLS. |
+| failOpen | bool | false | This field defines whether to allow requests to proceed when the rate limit service fails to respond with a valid rate limit decision within the timeout defined on the extension service. |
+| enableXRateLimitHeaders | bool | false | This field defines whether to include the X-RateLimit headers X-RateLimit-Limit, X-RateLimit-Remaining, and X-RateLimit-Reset (as defined by the IETF Internet-Draft https://tools.ietf.org/id/draft-polli-ratelimit-headers-03.html), on responses to clients when the Rate Limit Service is consulted for a request. |
+| enableResourceExhaustedCode | bool | false | This field defines whether to translate status code 429 to gRPC RESOURCE_EXHAUSTED instead of UNAVAILABLE. |
+
+### Metrics Configuration
+
+MetricsParameters holds configurable parameters for Contour and Envoy metrics.
+
+| Field Name | Type | Default | Description |
+| ----------- | ----------------------- | ------- | -------------------------------------------------------------------- |
+| contour | MetricsServerParameters | | [Metrics Server Parameters](#metrics-server-parameters) for Contour. |
+| envoy | MetricsServerParameters | | [Metrics Server Parameters](#metrics-server-parameters) for Envoy. |
+
+### Metrics Server Parameters
+
+MetricsServerParameters holds configurable parameters for Contour and Envoy metrics.
+Metrics are served over HTTPS if `server-certificate-path` and `server-key-path` are set.
+Metrics and health endpoints cannot have the same port number when metrics are served over HTTPS.
+
+| Field Name | Type | Default | Description |
+| ----------------------- | ------ | ---------------------------- | -----------------------------------------------------------------------------|
+| address | string | 0.0.0.0 | Address that metrics server will bind to. |
+| port | int | 8000 (Contour), 8002 (Envoy) | Port that metrics server will bind to. |
+| server-certificate-path | string | none | Optional path to the server certificate file. |
+| server-key-path | string | none | Optional path to the server private key file. |
+| ca-certificate-path | string | none | Optional path to the CA certificate file used to verify client certificates. |
+
+### Socket Options
+
+| Field Name | Type | Default | Description |
+| --------------- | ------ | ------- | ----------------------------------------------------------------------------- |
+| tos | int | 0 | Defines the value for IPv4 TOS field (including 6 bit DSCP field) for IP packets originating from Envoy listeners. Single value is applied to all listeners. The value must be in the range 0-255, 0 means socket option is not set. If listeners are bound to IPv6-only addresses, setting this option will cause an error. |
+| traffic-class | int | 0 | Defines the value for IPv6 Traffic Class field (including 6 bit DSCP field) for IP packets originating from the Envoy listeners. Single value is applied to all listeners. The value must be in the range 0-255, 0 means socket option is not set. If listeners are bound to IPv4-only addresses, setting this option will cause an error. |
+
+
+### Circuit Breakers
+
+| Field Name | Type | Default | Description |
+| --------------- | ------ | ------- | ----------------------------------------------------------------------------- |
+| max-connections | int | 0 | The maximum number of connections that a single Envoy instance allows to the Kubernetes Service; defaults to 1024. |
+| max-pending-requests | int | 0 | The maximum number of pending requests that a single Envoy instance allows to the Kubernetes Service; defaults to 1024. |
+| max-requests | int | 0 | The maximum parallel requests a single Envoy instance allows to the Kubernetes Service; defaults to 1024 |
+| max-retries | int | 0 | The maximum number of parallel retries a single Envoy instance allows to the Kubernetes Service; defaults to 3. This setting only makes sense if the cluster is configured to do retries.|
+
+### Configuration Example
+
+The following is an example ConfigMap with configuration file included:
+
+```yaml
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ name: contour
+ namespace: projectcontour
+data:
+ contour.yaml: |
+ #
+ # server:
+ # determine which XDS Server implementation to utilize in Contour.
+ # xds-server-type: contour
+ #
+ # specify the gateway-api Gateway Contour should configure
+ # gateway:
+ # controllerName: projectcontour.io/gateway-controller
+ #
+ # should contour expect to be running inside a k8s cluster
+ # incluster: true
+ #
+ # path to kubeconfig (if not running inside a k8s cluster)
+ # kubeconfig: /path/to/.kube/config
+ #
+ # Disable RFC-compliant behavior to strip "Content-Length" header if
+ # "Tranfer-Encoding: chunked" is also set.
+ # disableAllowChunkedLength: false
+ # Disable HTTPProxy permitInsecure field
+ disablePermitInsecure: false
+ tls:
+ # minimum TLS version that Contour will negotiate
+ # minimum-protocol-version: "1.2"
+ # TLS ciphers to be supported by Envoy TLS listeners when negotiating
+ # TLS 1.2.
+ # cipher-suites:
+ # - '[ECDHE-ECDSA-AES128-GCM-SHA256|ECDHE-ECDSA-CHACHA20-POLY1305]'
+ # - '[ECDHE-RSA-AES128-GCM-SHA256|ECDHE-RSA-CHACHA20-POLY1305]'
+ # - 'ECDHE-ECDSA-AES256-GCM-SHA384'
+ # - 'ECDHE-RSA-AES256-GCM-SHA384'
+ # Defines the Kubernetes name/namespace matching a secret to use
+ # as the fallback certificate when requests which don't match the
+ # SNI defined for a vhost.
+ fallback-certificate:
+ # name: fallback-secret-name
+ # namespace: projectcontour
+ envoy-client-certificate:
+ # name: envoy-client-cert-secret-name
+ # namespace: projectcontour
+ ### Logging options
+ # Default setting
+ accesslog-format: envoy
+ # The default access log format is defined by Envoy but it can be customized by setting following variable.
+ # accesslog-format-string: "...\n"
+ # To enable JSON logging in Envoy
+ # accesslog-format: json
+ # accesslog-level: info
+ # The default fields that will be logged are specified below.
+ # To customise this list, just add or remove entries.
+ # The canonical list is available at
+ # https://godoc.org/github.com/projectcontour/contour/internal/envoy#JSONFields
+ # json-fields:
+ # - "@timestamp"
+ # - "authority"
+ # - "bytes_received"
+ # - "bytes_sent"
+ # - "downstream_local_address"
+ # - "downstream_remote_address"
+ # - "duration"
+ # - "method"
+ # - "path"
+ # - "protocol"
+ # - "request_id"
+ # - "requested_server_name"
+ # - "response_code"
+ # - "response_flags"
+ # - "uber_trace_id"
+ # - "upstream_cluster"
+ # - "upstream_host"
+ # - "upstream_local_address"
+ # - "upstream_service_time"
+ # - "user_agent"
+ # - "x_forwarded_for"
+ #
+ # default-http-versions:
+ # - "HTTP/2"
+ # - "HTTP/1.1"
+ #
+ # The following shows the default proxy timeout settings.
+ # timeouts:
+ # request-timeout: infinity
+ # connection-idle-timeout: 60s
+ # stream-idle-timeout: 5m
+ # max-connection-duration: infinity
+ # connection-shutdown-grace-period: 5s
+ #
+ # Envoy cluster settings.
+ # cluster:
+ # configure the cluster dns lookup family
+ # valid options are: auto (default), v4, v6, all
+ # dns-lookup-family: auto
+ # the maximum requests for upstream connections.
+ # If not specified, there is no limit.
+ # Setting this parameter to 1 will effectively disable keep alive
+ # max-requests-per-connection: 0
+ # the soft limit on size of the cluster’s new connection read and write buffers
+ # per-connection-buffer-limit-bytes: 32768
+ #
+ # network:
+ # Configure the number of additional ingress proxy hops from the
+ # right side of the x-forwarded-for HTTP header to trust.
+ # num-trusted-hops: 0
+ # Configure the port used to access the Envoy Admin interface.
+ # admin-port: 9001
+ #
+ # Configure an optional global rate limit service.
+ # rateLimitService:
+ # Identifies the extension service defining the rate limit service,
+ # formatted as /.
+ # extensionService: projectcontour/ratelimit
+ # Defines the rate limit domain to pass to the rate limit service.
+ # Acts as a container for a set of rate limit definitions within
+ # the RLS.
+ # domain: contour
+ # Defines whether to allow requests to proceed when the rate limit
+ # service fails to respond with a valid rate limit decision within
+ # the timeout defined on the extension service.
+ # failOpen: false
+ # Defines whether to include the X-RateLimit headers X-RateLimit-Limit,
+ # X-RateLimit-Remaining, and X-RateLimit-Reset (as defined by the IETF
+ # Internet-Draft linked below), on responses to clients when the Rate
+ # Limit Service is consulted for a request.
+ # ref. https://tools.ietf.org/id/draft-polli-ratelimit-headers-03.html
+ # enableXRateLimitHeaders: false
+ # Defines whether to translate status code 429 to grpc code RESOURCE_EXHAUSTED
+ # instead of the default UNAVAILABLE
+ # enableResourceExhaustedCode: false
+ #
+ # Global Policy settings.
+ # policy:
+ # # Default headers to set on all requests (unless set/removed on the HTTPProxy object itself)
+ # request-headers:
+ # set:
+ # # example: the hostname of the Envoy instance that proxied the request
+ # X-Envoy-Hostname: %HOSTNAME%
+ # # example: add a l5d-dst-override header to instruct Linkerd what service the request is destined for
+ # l5d-dst-override: %CONTOUR_SERVICE_NAME%.%CONTOUR_NAMESPACE%.svc.cluster.local:%CONTOUR_SERVICE_PORT%
+ # # default headers to set on all responses (unless set/removed on the HTTPProxy object itself)
+ # response-headers:
+ # set:
+ # # example: Envoy flags that provide additional details about the response or connection
+ # X-Envoy-Response-Flags: %RESPONSE_FLAGS%
+ # Whether or not the policy settings should apply to ingress objects
+ # applyToIngress: true
+ #
+ # metrics:
+ # contour:
+ # address: 0.0.0.0
+ # port: 8000
+ # server-certificate-path: /path/to/server-cert.pem
+ # server-key-path: /path/to/server-private-key.pem
+ # ca-certificate-path: /path/to/root-ca-for-client-validation.pem
+ # envoy:
+ # address: 0.0.0.0
+ # port: 8002
+ # server-certificate-path: /path/to/server-cert.pem
+ # server-key-path: /path/to/server-private-key.pem
+ # ca-certificate-path: /path/to/root-ca-for-client-validation.pem
+ #
+ # listener:
+ # connection-balancer: exact
+ # socket-options:
+ # tos: 64
+ # traffic-class: 64
+```
+
+_Note:_ The default example `contour` includes this [file][1] for easy deployment of Contour.
+
+## Environment Variables
+
+### CONTOUR_NAMESPACE
+
+If present, the value of the `CONTOUR_NAMESPACE` environment variable is used as:
+
+1. The value for the `contour bootstrap --namespace` flag unless otherwise specified.
+1. The value for the `contour certgen --namespace` flag unless otherwise specified.
+1. The value for the `contour serve --envoy-service-namespace` flag unless otherwise specified.
+1. The value for the `contour serve --leader-election-resource-namespace` flag unless otherwise specified.
+
+The `CONTOUR_NAMESPACE` environment variable is set via the [Downward API][6] in the Contour [example manifests][7].
+
+## Bootstrap Config File
+
+The bootstrap configuration file is generated by an initContainer in the Envoy daemonset which runs the `contour bootstrap` command to generate the file.
+This configuration file configures the Envoy container to connect to Contour and receive configuration via xDS.
+
+The next section outlines all the available flags that can be passed to the `contour bootstrap` command which are used to customize
+the configuration file to match the environment in which Envoy is deployed.
+
+### Bootstrap Flags
+
+There are flags that can be passed to `contour bootstrap` that help configure how Envoy
+connects to Contour:
+
+| Flag | Default | Description |
+| -------------------------------------- |-------------------| ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
+| --resources-dir | "" | Directory where resource files will be written. |
+| --admin-address | /admin/admin.sock | Path to Envoy admin unix domain socket. |
+| --admin-port (Deprecated) | 9001 | Deprecated: Port is now configured as a Contour flag. |
+| --xds-address | 127.0.0.1 | Address to connect to Contour xDS server on. |
+| --xds-port | 8001 | Port to connect to Contour xDS server on. |
+| --envoy-cafile | "" | CA filename for Envoy secure xDS gRPC communication. |
+| --envoy-cert-file | "" | Client certificate filename for Envoy secure xDS gRPC communication. |
+| --envoy-key-file | "" | Client key filename for Envoy secure xDS gRPC communication. |
+| --namespace | projectcontour | Namespace the Envoy container will run, also configured via ENV variable "CONTOUR_NAMESPACE". Namespace is used as part of the metric names on static resources defined in the bootstrap configuration file. |
+| --xds-resource-version | v3 | Currently, the only valid xDS API resource version is `v3`. |
+| --dns-lookup-family | auto | Defines what DNS Resolution Policy to use for Envoy -> Contour cluster name lookup. Either v4, v6, auto or all. |
+| --log-format | text | Log output format for Contour. Either text or json. |
+| --overload-max-heap | 0 | Defines the maximum heap memory of the envoy controlled by the overload manager. When the value is greater than 0, the overload manager is enabled, and when envoy reaches 95% of the maximum heap size, it performs a shrink heap operation. When it reaches 98% of the maximum heap size, Envoy Will stop accepting requests. |
+
+
+[1]: {{< param github_url>}}/tree/{{< param branch >}}/examples/contour/01-contour-config.yaml
+[2]: config/access-logging
+[3]: https://kubernetes.io/docs/concepts/configuration/organize-cluster-access-kubeconfig/
+[4]: https://golang.org/pkg/time/#ParseDuration
+[5]: https://godoc.org/github.com/projectcontour/contour/internal/envoy#DefaultFields
+[6]: https://kubernetes.io/docs/tasks/inject-data-application/environment-variable-expose-pod-information/
+[7]: {{< param github_url>}}/tree/{{< param branch >}}/examples/contour
+[8]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/core/v3/protocol.proto#envoy-v3-api-field-config-core-v3-httpprotocoloptions-idle-timeout
+[9]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/filters/network/http_connection_manager/v3/http_connection_manager.proto#envoy-v3-api-field-extensions-filters-network-http-connection-manager-v3-httpconnectionmanager-stream-idle-timeout
+[10]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/core/v3/protocol.proto#envoy-v3-api-field-config-core-v3-httpprotocoloptions-max-connection-duration
+[11]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/filters/network/http_connection_manager/v3/http_connection_manager.proto#envoy-v3-api-field-extensions-filters-network-http-connection-manager-v3-httpconnectionmanager-drain-timeout
+[12]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/filters/network/http_connection_manager/v3/http_connection_manager.proto#envoy-v3-api-field-extensions-filters-network-http-connection-manager-v3-httpconnectionmanager-request-timeout
+[13]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/filters/network/http_connection_manager/v3/http_connection_manager.proto#envoy-v3-api-field-extensions-filters-network-http-connection-manager-v3-httpconnectionmanager-delayed-close-timeout
+[14]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/config/listener/v3/listener.proto#config-listener-v3-listener-connectionbalanceconfig
diff --git a/site/content/docs/1.28/deploy-options.md b/site/content/docs/1.28/deploy-options.md
new file mode 100644
index 00000000000..0ae74a53bd9
--- /dev/null
+++ b/site/content/docs/1.28/deploy-options.md
@@ -0,0 +1,383 @@
+# Deployment Options
+
+The [Getting Started][8] guide shows you a simple way to get started with Contour on your cluster.
+This topic explains the details and shows you additional options.
+Most of this covers running Contour using a Kubernetes Service of `Type: LoadBalancer`.
+If you don't have a cluster with that capability see the [Running without a Kubernetes LoadBalancer][1] section.
+
+## Installation
+
+Contour requires a secret containing TLS certificates that are used to secure the gRPC communication between Contour<>Envoy.
+This secret can be auto-generated by the Contour `certgen` job or provided by an administrator.
+Traffic must be forwarded to Envoy, typically via a Service of `type: LoadBalancer`.
+All other requirements such as RBAC permissions, configuration details, are provided or have good defaults for most installations.
+
+### Setting resource requests and limits
+
+It is recommended that resource requests and limits be set on all Contour and Envoy containers.
+The example YAML manifests used in the [Getting Started][8] guide do not include these, because the appropriate values can vary widely from user to user.
+The table below summarizes the Contour and Envoy containers, and provides some reasonable resource requests to start with (note that these should be adjusted based on observed usage and expected load):
+
+| Workload | Container | Request (mem) | Request (cpu) |
+| ------------------- | ---------------- | ------------- | ------------- |
+| deployment/contour | contour | 128Mi | 250m |
+| daemonset/envoy | envoy | 256Mi | 500m |
+| daemonset/envoy | shutdown-manager | 50Mi | 25m |
+
+
+### Envoy as Daemonset
+
+The recommended installation is for Contour to run as a Deployment and Envoy to run as a Daemonset.
+The example Damonset places a single instance of Envoy per node in the cluster as well as attaches to `hostPorts` on each node.
+This model allows for simple scaling of Envoy instances as well as ensuring even distribution of instances across the cluster.
+
+The [example daemonset manifest][2] or [Contour Gateway Provisioner][12] will create an installation based on these recommendations.
+
+_Note: If the size of the cluster is scaled down, connections can be lost since Kubernetes Damonsets do not follow proper `preStop` hooks._
+
+### Envoy as Deployment
+
+An alternative Envoy deployment model is utilizing a Kubernetes Deployment with a configured `podAntiAffinity` which attempts to mirror the Daemonset deployment model.
+A benefit of this model compared to the Daemonset version is when a node is removed from the cluster, the proper shutdown events are available so connections can be cleanly drained from Envoy before terminating.
+
+The [example deployment manifest][14] will create an installation based on these recommendations.
+
+## Testing your installation
+
+### Get your hostname or IP address
+
+To retrieve the IP address or DNS name assigned to your Contour deployment, run:
+
+```bash
+$ kubectl get -n projectcontour service envoy -o wide
+```
+
+On AWS, for example, the response looks like:
+
+```
+NAME CLUSTER-IP EXTERNAL-IP PORT(S) AGE SELECTOR
+contour 10.106.53.14 a47761ccbb9ce11e7b27f023b7e83d33-2036788482.ap-southeast-2.elb.amazonaws.com 80:30274/TCP 3h app=contour
+```
+
+Depending on your cloud provider, the `EXTERNAL-IP` value is an IP address, or, in the case of Amazon AWS, the DNS name of the ELB created for Contour. Keep a record of this value.
+
+Note that if you are running an Elastic Load Balancer (ELB) on AWS, you must add more details to your configuration to get the remote address of your incoming connections.
+See the [instructions for enabling the PROXY protocol.][4]
+
+#### Minikube
+
+On Minikube, to get the IP address of the Contour service run:
+
+```bash
+$ minikube service -n projectcontour envoy --url
+```
+
+The response is always an IP address, for example `http://192.168.99.100:30588`. This is used as CONTOUR_IP in the rest of the documentation.
+
+#### kind
+
+When creating the cluster on Kind, pass a custom configuration to allow Kind to expose port 80/443 to your local host:
+
+```yaml
+kind: Cluster
+apiVersion: kind.x-k8s.io/v1alpha4
+nodes:
+- role: control-plane
+- role: worker
+ extraPortMappings:
+ - containerPort: 80
+ hostPort: 80
+ listenAddress: "0.0.0.0"
+ - containerPort: 443
+ hostPort: 443
+ listenAddress: "0.0.0.0"
+```
+
+Then run the create cluster command passing the config file as a parameter.
+This file is in the `examples/kind` directory:
+
+```bash
+$ kind create cluster --config examples/kind/kind-expose-port.yaml
+```
+
+Then, your CONTOUR_IP (as used below) will just be `localhost:80`.
+
+_Note: We've created a public DNS record (`local.projectcontour.io`) which is configured to resolve to `127.0.0.1``. This allows you to use a real domain name in your kind cluster._
+
+### Test with Ingress
+
+The Contour repository contains an example deployment of the Kubernetes Up and Running demo application, [kuard][5].
+To test your Contour deployment, deploy `kuard` with the following command:
+
+```bash
+$ kubectl apply -f https://projectcontour.io/examples/kuard.yaml
+```
+
+Then monitor the progress of the deployment with:
+
+```bash
+$ kubectl get po,svc,ing -l app=kuard
+```
+
+You should see something like:
+
+```
+NAME READY STATUS RESTARTS AGE
+po/kuard-370091993-ps2gf 1/1 Running 0 4m
+po/kuard-370091993-r63cm 1/1 Running 0 4m
+po/kuard-370091993-t4dqk 1/1 Running 0 4m
+
+NAME CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+svc/kuard 10.110.67.121 80/TCP 4m
+
+NAME HOSTS ADDRESS PORTS AGE
+ing/kuard * 10.0.0.47 80 4m
+```
+
+... showing that there are three Pods, one Service, and one Ingress that is bound to all virtual hosts (`*`).
+
+In your browser, navigate your browser to the IP or DNS address of the Contour Service to interact with the demo application.
+
+### Test with HTTPProxy
+
+To test your Contour deployment with [HTTPProxy][9], run the following command:
+
+```sh
+$ kubectl apply -f https://projectcontour.io/examples/kuard-httpproxy.yaml
+```
+
+Then monitor the progress of the deployment with:
+
+```sh
+$ kubectl get po,svc,httpproxy -l app=kuard
+```
+
+You should see something like:
+
+```sh
+NAME READY STATUS RESTARTS AGE
+pod/kuard-bcc7bf7df-9hj8d 1/1 Running 0 1h
+pod/kuard-bcc7bf7df-bkbr5 1/1 Running 0 1h
+pod/kuard-bcc7bf7df-vkbtl 1/1 Running 0 1h
+
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+service/kuard ClusterIP 10.102.239.168 80/TCP 1h
+
+NAME FQDN TLS SECRET FIRST ROUTE STATUS STATUS DESCRIPT
+httpproxy.projectcontour.io/kuard kuard.local valid valid HTTPProxy
+```
+
+... showing that there are three Pods, one Service, and one HTTPProxy .
+
+In your terminal, use curl with the IP or DNS address of the Contour Service to send a request to the demo application:
+
+```sh
+$ curl -H 'Host: kuard.local' ${CONTOUR_IP}
+```
+
+## Running without a Kubernetes LoadBalancer
+
+If you can't or don't want to use a Service of `type: LoadBalancer` there are other ways to run Contour.
+
+### NodePort Service
+
+If your cluster doesn't have the capability to configure a Kubernetes LoadBalancer,
+or if you want to configure the load balancer outside Kubernetes,
+you can change the Envoy Service in the [`02-service-envoy.yaml`][7] file and set `type` to `NodePort`.
+
+This will have every node in your cluster listen on the resultant port and forward traffic to Contour.
+That port can be discovered by taking the second number listed in the `PORT` column when listing the service, for example `30274` in `80:30274/TCP`.
+
+Now you can point your browser at the specified port on any node in your cluster to communicate with Contour.
+
+### Host Networking
+
+You can run Contour without a Kubernetes Service at all.
+This is done by having the Envoy pod run with host networking.
+Contour's examples utilize this model in the `/examples` directory.
+To configure, set: `hostNetwork: true` and `dnsPolicy: ClusterFirstWithHostNet` on your Envoy pod definition.
+Next, pass `--envoy-service-http-port=80 --envoy-service-https-port=443` to the contour `serve` command which instructs Envoy to listen directly on port 80/443 on each host that it is running.
+This is best paired with a DaemonSet (perhaps paired with Node affinity) to ensure that a single instance of Contour runs on each Node.
+See the [AWS NLB tutorial][10] as an example.
+
+## Disabling Features
+
+You can run Contour with certain features disabled by passing `--disable-feature` flag to the Contour `serve` command.
+The flag is used to disable the informer for a custom resource, effectively making the corresponding CRD optional in the cluster.
+You can provide the flag multiple times.
+
+For example, to disable ExtensionService CRD, use the flag as follows: `--disable-feature=extensionservices`.
+
+See the [configuration section entry][19] for all options.
+
+## Upgrading Contour/Envoy
+
+At times, it's needed to upgrade Contour, the version of Envoy, or both.
+The included `shutdown-manager` can assist with watching Envoy for open connections while draining and give signal back to Kubernetes as to when it's fine to delete Envoy pods during this process.
+
+See the [redeploy envoy][11] docs for more information about how to not drop active connections to Envoy.
+Also see the [upgrade guides][15] on steps to roll out a new version of Contour.
+
+## Running Multiple Instances of Contour
+
+It's possible to run multiple instances of Contour within a single Kubernetes cluster.
+This can be useful for separating external vs. internal ingress, for having separate ingress controllers for different ingress classes, and more.
+Each Contour instance can also be configured via the `--watch-namespaces` flag to handle their own namespaces. This allows the Kubernetes RBAC objects
+to be restricted further.
+
+The recommended way to deploy multiple Contour instances is to put each instance in its own namespace.
+This avoids most naming conflicts that would otherwise occur, and provides better logical separation between the instances.
+However, it is also possible to deploy multiple instances in a single namespace if needed; this approach requires more modifications to the example manifests to function properly.
+Each approach is described in detail below, using the [examples/contour][17] directory's manifests for reference.
+
+### In Separate Namespaces (recommended)
+
+In general, this approach requires updating the `namespace` of all resources, as well as giving unique names to cluster-scoped resources to avoid conflicts.
+
+- `00-common.yaml`:
+ - update the name of the `Namespace`
+ - update the namespace of both `ServiceAccounts`
+- `01-contour-config.yaml`:
+ - update the namespace of the `ConfigMap`
+ - if you have any namespaced references within the ConfigMap contents (e.g. `fallback-certificate`, `envoy-client-certificate`), ensure those point to the correct namespace as well.
+- `01-crds.yaml` will be shared between the two instances; no changes are needed.
+- `02-job-certgen.yaml`:
+ - update the namespace of all resources
+ - update the namespace of the `ServiceAccount` subject within the `RoleBinding`
+- `02-role-contour.yaml`:
+ - update the name of the `ClusterRole` to be unique
+ - update the namespace of the `Role`
+- `02-rbac.yaml`:
+ - update the name of the `ClusterRoleBinding` to be unique
+ - update the namespace of the `RoleBinding`
+ - update the namespaces of the `ServiceAccount` subject within both resources
+ - update the name of the ClusterRole within the ClusterRoleBinding's roleRef to match the unique name used in `02-role-contour.yaml`
+- `02-service-contour.yaml`:
+ - update the namespace of the `Service`
+- `02-service-envoy.yaml`:
+ - update the namespace of the `Service`
+- `03-contour.yaml`:
+ - update the namespace of the `Deployment`
+ - add an argument to the container, `--ingress-class-name=`, so this instance only processes Ingresses/HTTPProxies with the given ingress class.
+- `03-envoy.yaml`:
+ - update the namespace of the `DaemonSet`
+ - remove the two `hostPort` definitions from the container (otherwise, these would conflict between the two instances)
+
+
+### In The Same Namespace
+
+This approach requires giving unique names to all resources to avoid conflicts, and updating all resource references to use the correct names.
+
+- `00-common.yaml`:
+ - update the names of both `ServiceAccounts` to be unique
+- `01-contour-config.yaml`:
+ - update the name of the `ConfigMap` to be unique
+- `01-crds.yaml` will be shared between the two instances; no changes are needed.
+- `02-job-certgen.yaml`:
+ - update the names of all resources to be unique
+ - update the name of the `Role` within the `RoleBinding`'s roleRef to match the unique name used for the `Role`
+ - update the name of the `ServiceAccount` within the `RoleBinding`'s subjects to match the unique name used for the `ServiceAccount`
+ - update the serviceAccountName of the `Job`
+ - add an argument to the container, `--secrets-name-suffix=`, so the generated TLS secrets have unique names
+ - update the spec.template.metadata.labels on the `Job` to be unique
+- `02-role-contour.yaml`:
+ - update the names of the `ClusterRole` and `Role` to be unique
+- `02-rbac.yaml`:
+ - update the names of the `ClusterRoleBinding` and `RoleBinding` to be unique
+ - update the roleRefs within both resources to reference the unique `Role` and `ClusterRole` names used in `02-role-contour.yaml`
+ - update the subjects within both resources to reference the unique `ServiceAccount` name used in `00-common.yaml`
+- `02-service-contour.yaml`:
+ - update the name of the `Service` to be unique
+ - update the selector to be unique (this must match the labels used in `03-contour.yaml`, below)
+- `02-service-envoy.yaml`:
+ - update the name of the `Service` to be unique
+ - update the selector to be unique (this must match the labels used in `03-envoy.yaml`, below)
+- `03-contour.yaml`:
+ - update the name of the `Deployment` to be unique
+ - update the metadata.labels, the spec.selector.matchLabels, the spec.template.metadata.labels, and the spec.template.spec.affinity.podAntiAffinity labels to match the labels used in `02-service-contour.yaml`
+ - update the serviceAccountName to match the unique name used in `00-common.yaml`
+ - update the `contourcert` volume to reference the unique `Secret` name generated from `02-certgen.yaml` (e.g. `contourcert`)
+ - update the `contour-config` volume to reference the unique `ConfigMap` name used in `01-contour-config.yaml`
+ - add an argument to the container, `--leader-election-resource-name=`, so this Contour instance uses a separate leader election `Lease`
+ - add an argument to the container, `--envoy-service-name=`, referencing the unique name used in `02-service-envoy.yaml`
+ - add an argument to the container, `--ingress-class-name=`, so this instance only processes Ingresses/HTTPProxies with the given ingress class.
+- `03-envoy.yaml`:
+ - update the name of the `DaemonSet` to be unique
+ - update the metadata.labels, the spec.selector.matchLabels, and the spec.template.metadata.labels to match the unique labels used in `02-service-envoy.yaml`
+ - update the `--xds-address` argument to the initContainer to use the unique name of the contour Service from `02-service-contour.yaml`
+ - update the serviceAccountName to match the unique name used in `00-common.yaml`
+ - update the `envoycert` volume to reference the unique `Secret` name generated from `02-certgen.yaml` (e.g. `envoycert`)
+ - remove the two `hostPort` definitions from the container (otherwise, these would conflict between the two instances)
+
+### Using the Gateway provisioner
+
+The Contour Gateway provisioner also supports deploying multiple instances of Contour, either in the same namespace or different namespaces.
+See [Getting Started with the Gateway provisioner][16] for more information on getting started with the Gateway provisioner.
+To deploy multiple Contour instances, you create multiple `Gateways`, either in the same namespace or in different namespaces.
+
+Note that although the provisioning request itself is made via a Gateway API resource (`Gateway`), this method of installation still allows you to use *any* of the supported APIs for defining virtual hosts and routes: `Ingress`, `HTTPProxy`, or Gateway API's `HTTPRoute` and `TLSRoute`.
+
+If you are using `Ingress` or `HTTPProxy`, you will likely want to assign each Contour instance a different ingress class, so they each handle different subsets of `Ingress`/`HTTPProxy` resources.
+To do this, [create two separate GatewayClasses][18], each with a different `ContourDeployment` parametersRef.
+The `ContourDeployment` specs should look like:
+
+```yaml
+kind: ContourDeployment
+apiVersion: projectcontour.io/v1alpha1
+metadata:
+ namespace: projectcontour
+ name: ingress-class-1
+spec:
+ runtimeSettings:
+ ingress:
+ classNames:
+ - ingress-class-1
+---
+kind: ContourDeployment
+apiVersion: projectcontour.io/v1alpha1
+metadata:
+ namespace: projectcontour
+ name: ingress-class-2
+spec:
+ runtimeSettings:
+ ingress:
+ classNames:
+ - ingress-class-2
+```
+
+Then create each `Gateway` with the appropriate `spec.gatewayClassName`.
+
+## Running Contour in tandem with another ingress controller
+
+If you're running multiple ingress controllers, or running on a cloudprovider that natively handles ingress,
+you can specify the annotation `kubernetes.io/ingress.class: "contour"` on all ingresses that you would like Contour to claim.
+You can customize the class name with the `--ingress-class-name` flag at runtime. (A comma-separated list of class names is allowed.)
+If the `kubernetes.io/ingress.class` annotation is present with a value other than `"contour"`, Contour will ignore that ingress.
+
+## Uninstall Contour
+
+To remove Contour or the Contour Gateway Provisioner from your cluster, delete the namespace:
+
+```bash
+$ kubectl delete ns projectcontour
+```
+**Note**: Your namespace may differ from above.
+
+[1]: #running-without-a-kubernetes-loadbalancer
+[2]: {{< param github_url>}}/tree/{{< param branch >}}/examples/render/contour.yaml
+[3]: #host-networking
+[4]: guides/proxy-proto.md
+[5]: https://github.com/kubernetes-up-and-running/kuard
+[7]: {{< param github_url>}}/tree/{{< param branch >}}/examples/contour/02-service-envoy.yaml
+[8]: /getting-started
+[9]: config/fundamentals.md
+[10]: guides/deploy-aws-nlb.md
+[11]: redeploy-envoy.md
+[12]: {{< param github_url>}}/tree/{{< param branch >}}/examples/render/contour-gateway-provisioner.yaml
+[13]: https://projectcontour.io/resources/deprecation-policy/
+[14]: {{< param github_url>}}/tree/{{< param branch >}}/examples/render/contour-deployment.yaml
+[15]: /resources/upgrading/
+[16]: https://projectcontour.io/getting-started/#option-3-contour-gateway-provisioner-alpha
+[17]: {{< param github_url>}}/tree/{{< param branch >}}/examples/contour
+[18]: guides/gateway-api/#next-steps
+[19]: configuration.md
\ No newline at end of file
diff --git a/site/content/docs/1.28/github.md b/site/content/docs/1.28/github.md
new file mode 100644
index 00000000000..8a0f36b4f4d
--- /dev/null
+++ b/site/content/docs/1.28/github.md
@@ -0,0 +1,80 @@
+This document outlines how we use GitHub.
+
+## Milestones
+
+Contour attempts to ship on a quarterly basis.
+These releases are tracked with a milestone.
+The _current_ release is the milestone with the closest delivery date.
+
+Issues which are not assigned to the current milestone _should not be worked on_.
+
+## Priorities
+
+This project has three levels of priority:
+
+- p0 - Must fix immediately.
+This is reserved for bugs and security issues. A milestone cannot ship with open p0 issues.
+- p1 - Should be done.
+p1 issues assigned to a milestone _should_ be completed during that milestone.
+- p2 - May be done.
+p2 issues assigned to a milestone _may_ be completed during that milestone if time permits.
+
+Issues without a priority are _unprioritised_. Priority will be assigned by a PM or release manager during issue triage.
+
+## Questions
+
+We encourage support questions via issues.
+Questions will be tagged `question` and are not assigned a milestone or a priority.
+
+## Waiting for information
+
+Any issue which lacks sufficient information for triage will be tagged `waiting-for-info`.
+Issues with this tag may be closed after a reasonable length of time if further information is not forthcoming.
+
+## Issue tagging
+
+Issues without tags have not be triaged.
+
+During issue triage, usually by a project member, release manager, or pm, one or more tags will be assigned.
+
+- `Needs-Product` indicates the issue needs attention by a product owner or PM.
+- `Needs-design-doc` indicates the issue requires a design document to be circulated.
+
+These are blocking states, these labels must be resolved, either by PM or agreeing on a design.
+
+## Assigning an issue
+
+Issues within a milestone _should_ be assigned to an owner when work commences on them.
+Assigning an issue indicates that you are working on it.
+
+Before you start to work on an issue you should assign yourself.
+From that point onward you are responsible for the issue and you are expected to report timely status on the issue to anyone that asks.
+
+If you cease work on an issue, even if incomplete, you should leave a comment to that effect on the issue and remove yourself as the assignee.
+From that point onward you are no longer responsible for the issue, however you may be approached as a subject matter expert--as the last person to touch the issue--by future assignees.
+
+For infrequent contributors who are not members of the Contour project, assign yourself by leaving a comment to that effect on the issue.
+
+*Do not hoard issues, you won't enjoy it*
+
+## Requesting a review
+
+PRs which are related to issues in the current milestone should be assigned to the current milestone.
+This is an indicator to reviewers that the PR is ready for review and should be reviewed in the current milestone.
+Occasionally PRs may be assigned to the next milestone indicating they are for review at the start of the next development cycle.
+
+All PRs should reference the issue they relate to either by one of the following;
+
+- `Fixes #NNNN` indicating that merging this issue will fix issue #NNNN
+- `Updates #NNNN` indicating that merging this issue will progress issue #NNNN to some degree.
+
+If there is no `Updates` or `Fixes` line in the PR the review will, with the exception of trivial or self evident fixes, be deferred.
+
+[Further reading][1]
+
+## Help wanted and good first issues
+
+The `help wanted` and `good first issue` tags _may_ be assigned to issues _in the current milestone_.
+To limit the amount of work in progress, `help wanted` and `good first issue` should not be used for issues outside the current milestone.
+
+[1]: https://dave.cheney.net/2019/02/18/talk-then-code
\ No newline at end of file
diff --git a/site/content/docs/1.28/grpc-tls-howto.md b/site/content/docs/1.28/grpc-tls-howto.md
new file mode 100644
index 00000000000..51770de950d
--- /dev/null
+++ b/site/content/docs/1.28/grpc-tls-howto.md
@@ -0,0 +1,169 @@
+# Enabling TLS between Envoy and Contour
+
+This document describes the steps required to secure communication between Envoy and Contour.
+The outcome of this is that we will have two Secrets available in the `projectcontour` namespace:
+
+- **contourcert:** contains Contour's keypair which is used for serving TLS secured gRPC, and the CA's public certificate bundle which is used for validating Envoy's client certificate.
+Contour's certificate must be a valid certificate for the name `contour` in order for this to work.
+This is currently hardcoded by Contour.
+- **envoycert:** contains Envoy's keypair which used as a client for connecting to Contour, and the CA's public certificate bundle which is used for validating Contour's server certificate.
+
+Note that both Secrets contain a copy of the CA certificate bundle under the `ca.crt` data key.
+
+## Ways you can get the certificates into your cluster
+
+- Deploy the Job from [certgen.yaml][1].
+This will run `contour certgen --kube --secrets-format=compact` for you.
+- Run `contour certgen --kube` locally.
+- Run the manual procedure below.
+
+## Caveats and warnings
+
+**Be very careful with your production certificates!**
+
+This is intended as an example to help you get started.
+For any real deployment, you should **carefully** manage all the certificates and control who has access to them.
+Make sure you don't commit them to any git repositories either.
+
+## Manual TLS certificate generation process
+
+### Generating a CA keypair
+
+First, we need to generate a keypair:
+
+```
+$ openssl req -x509 -new -nodes \
+ -keyout certs/cakey.pem -sha256 \
+ -days 1825 -out certs/cacert.pem \
+ -subj "/O=Project Contour/CN=Contour CA"
+```
+
+Then, the new CA key will be stored in `certs/cakey.pem` and the cert in `certs/cacert.pem`.
+
+### Generating Contour's keypair
+
+Next, we need to generate a keypair for Contour.
+First, we make a new private key:
+
+```
+$ openssl genrsa -out certs/contourkey.pem 2048
+```
+
+Then, we create a CSR and have our CA sign the CSR and issue a certificate.
+This uses the file [certs/cert-contour.ext][2], which ensures that at least one of the valid names of the certificate is the bareword `contour`.
+This is required for the handshake to succeed, as `contour bootstrap` configures Envoy to pass this as the SNI server name for the connection.
+
+```
+$ openssl req -new -key certs/contourkey.pem \
+ -out certs/contour.csr \
+ -subj "/O=Project Contour/CN=contour"
+
+$ openssl x509 -req -in certs/contour.csr \
+ -CA certs/cacert.pem \
+ -CAkey certs/cakey.pem \
+ -CAcreateserial \
+ -out certs/contourcert.pem \
+ -days 1825 -sha256 \
+ -extfile certs/cert-contour.ext
+```
+
+At this point, the contour certificate and key are in the files `certs/contourcert.pem` and `certs/contourkey.pem` respectively.
+
+### Generating Envoy's keypair
+
+Next, we generate a keypair for Envoy:
+
+```
+$ openssl genrsa -out certs/envoykey.pem 2048
+```
+
+Then, we generate a CSR and have the CA sign it:
+
+```
+$ openssl req -new -key certs/envoykey.pem \
+ -out certs/envoy.csr \
+ -subj "/O=Project Contour/CN=envoy"
+
+$ openssl x509 -req -in certs/envoy.csr \
+ -CA certs/cacert.pem \
+ -CAkey certs/cakey.pem \
+ -CAcreateserial \
+ -out certs/envoycert.pem \
+ -days 1825 -sha256 \
+ -extfile certs/cert-envoy.ext
+```
+
+Like the Contour certificate, this CSR uses the file [certs/cert-envoy.ext][3].
+However, in this case, there are no special names required.
+
+### Putting the certificates in the cluster
+
+Next, we create the required Secrets in the target Kubernetes cluster:
+
+```bash
+$ kubectl create secret -n projectcontour generic contourcert \
+ --from-file=tls.key=./certs/contourkey.pem \
+ --from-file=tls.crt=./certs/contourcert.pem \
+ --from-file=ca.crt=./certs/cacert.pem \
+ --save-config
+
+$ kubectl create secret -n projectcontour generic envoycert \
+ --from-file=tls.key=./certs/envoykey.pem \
+ --from-file=tls.crt=./certs/envoycert.pem \
+ --from-file=ca.crt=./certs/cacert.pem \
+ --save-config
+```
+
+Note that we don't put the CA **key** into the cluster, there's no reason for that to be there, and that would create a security problem.
+
+## Rotating Certificates
+
+Eventually the certificates that Contour and Envoy use will need to be rotated.
+The following steps can be taken to replace the certificates that Contour and Envoy are using:
+
+1. Generate a new keypair for both Contour and Envoy (optionally also for the CA)
+2. Update the Secrets that hold the gRPC TLS keypairs
+3. Contour and Envoy will automatically rotate their certificates after mounted secrets have been updated by the kubelet
+
+The secrets can be updated in-place by running:
+
+```bash
+$ kubectl create secret -n projectcontour generic contourcert \
+ --from-file=tls.key=./certs/contourkey.pem \
+ --from-file=tls.crt=./certs/contourcert.pem \
+ --from-file=ca.crt=./certs/cacert.pem \
+ --dry-run -o json \
+ | kubectl apply -f -
+
+$ kubectl create secret -n projectcontour generic envoycert \
+ --from-file=tls.key=./certs/envoykey.pem \
+ --from-file=tls.crt=./certs/envoycert.pem \
+ --from-file=ca.crt=./certs/cacert.pem \
+ --dry-run -o json \
+ | kubectl apply -f -
+```
+
+There are few preconditions that need to be met before Envoy can automatically reload certificate and key files:
+
+- Envoy must be version v1.14.1 or later
+- The bootstrap configuration must be generated with `contour bootstrap` using the `--resources-dir` argument, see [examples/contour/03-envoy.yaml][4]
+
+### Rotate using the contour-certgen job
+
+When using the built-in Contour certificate generation, the following steps can be used:
+
+1. Delete the contour-certgen job
+ - `kubectl delete job contour-certgen -n projectcontour`
+2. Reapply the contour-certgen job from [certgen.yaml][1]
+
+## Conclusion
+
+Once this process is done, the certificates will be present as Secrets in the `projectcontour` namespace, as required by
+[examples/contour][5].
+
+[1]: {{< param github_url >}}/tree/{{< param branch >}}/examples/contour/02-job-certgen.yaml
+[2]: {{< param github_url >}}/tree/{{< param branch >}}/certs/cert-contour.ext
+[3]: {{< param github_url >}}/tree/{{< param branch >}}/certs/cert-envoy.ext
+[4]: {{< param github_url >}}/tree/{{< param branch >}}/examples/contour/03-envoy.yaml
+[5]: {{< param github_url >}}/tree/{{< param branch >}}/examples/contour
+
diff --git a/site/content/docs/1.28/guides/_index.md b/site/content/docs/1.28/guides/_index.md
new file mode 100644
index 00000000000..8981b8fbd79
--- /dev/null
+++ b/site/content/docs/1.28/guides/_index.md
@@ -0,0 +1,9 @@
+---
+title: Guides
+description: Contour Resources
+id: guides
+---
+## Getting things done with Contour
+
+This page contains links to articles on configuring specific Contour features.
+
diff --git a/site/content/docs/1.28/guides/cert-manager.md b/site/content/docs/1.28/guides/cert-manager.md
new file mode 100644
index 00000000000..0f926946eda
--- /dev/null
+++ b/site/content/docs/1.28/guides/cert-manager.md
@@ -0,0 +1,670 @@
+---
+title: Deploying HTTPS services with Contour and cert-manager
+---
+
+This tutorial shows you how to securely deploy an HTTPS web application on a Kubernetes cluster, using:
+
+- Kubernetes
+- Contour, as the Ingress controller
+- [JetStack's cert-manager][1] to provision TLS certificates from [the Let's Encrypt project][6]
+
+## Prerequisites
+
+- A Kubernetes cluster deployed in either a data center or a cloud provider with a Kubernetes as a service offering. This tutorial was last tested on a GKE cluster running Kubernetes 1.22
+- RBAC enabled on your cluster
+- Your cluster must be able to request a public IP address from your cloud provider, using a load balancer. If you're on AWS or GKE this is automatic if you deploy a Kubernetes service object of type: LoadBalancer. If you're on your own datacenter you must set it up yourself
+- A DNS domain that you control, where you host your web application
+- Administrator permissions for all deployment steps
+
+**NOTE:** To use a local cluster like `minikube` or `kind`, see the instructions in [the deployment guide][7].
+
+## Summary
+
+This tutorial walks you through deploying:
+
+1. [Contour][0]
+2. [Jetstack cert-manager][1]
+3. A sample web application using HTTPProxy
+
+**NOTE:** If you encounter failures related to permissions, make sure the user you are operating as has administrator permissions.
+
+After you've been through the steps the first time, you don't need to repeat deploying Contour and cert-manager for subsequent application deployments. Instead, you can skip to step 3.
+
+## 1. Deploy Contour
+
+Run:
+
+```bash
+$ kubectl apply -f {{< param base_url >}}/quickstart/contour.yaml
+```
+
+to set up Contour as a deployment in its own namespace, `projectcontour`, and tell the cloud provider to provision an external IP that is forwarded to the Contour pods.
+
+Check the progress of the deployment with this command:
+
+```bash
+$ kubectl -n projectcontour get po
+NAME READY STATUS RESTARTS AGE
+contour-5475898957-jh9fm 1/1 Running 0 39s
+contour-5475898957-qlbs2 1/1 Running 0 39s
+contour-certgen-v1.19.0-5xthf 0/1 Completed 0 39s
+envoy-hqbkm 2/2 Running 0 39s
+```
+
+After all the `contour` & `envoy` pods reach `Running` status and fully `Ready`, move on to the next step.
+
+### Access your cluster
+
+Retrieve the external address of the load balancer assigned to Contour's Envoys by your cloud provider:
+
+```bash
+$ kubectl get -n projectcontour service envoy -o wide
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE SELECTOR
+envoy LoadBalancer 10.51.245.99 35.189.26.87 80:30111/TCP,443:30933/TCP 38d app=envoy
+```
+
+The value of `EXTERNAL-IP` varies by cloud provider. In this example GKE gives a bare IP address; AWS gives you a long DNS name.
+
+To make it easier to work with the external load balancer, the tutorial adds a DNS record to a domain we control that points to this load balancer's IP address:
+
+```bash
+$ host gke.davecheney.com
+gke.davecheney.com has address 35.189.26.87
+```
+
+On AWS, you specify a `CNAME`, not an `A` record, and it would look something like this:
+
+```bash
+$ host aws.davecheney.com
+aws.davecheney.com is an alias for a4d1766f6ce1611e7b27f023b7e83d33–1465548734.ap-southeast-2.elb.amazonaws.com.
+a4d1766f6ce1611e7b27f023b7e83d33–1465548734.ap-southeast-2.elb.amazonaws.com has address 52.63.20.117
+a4d1766f6ce1611e7b27f023b7e83d33–1465548734.ap-southeast-2.elb.amazonaws.com has address 52.64.233.204
+```
+
+In your own data center, you need to arrange for traffic from a public IP address to be forwarded to the cluster IP of the Contour service. This is beyond the scope of the tutorial.
+
+### Testing connectivity
+
+You must deploy at least one Ingress object before Contour can configure Envoy to serve traffic.
+Note that as a security feature, Contour does not configure Envoy to expose a port to the internet unless there's a reason it should.
+For this tutorial we deploy a version of Kenneth Reitz's [httpbin.org service][3].
+
+To deploy httpbin to your cluster, run this command:
+
+```bash
+$ kubectl apply -f {{< param base_url >}}/examples/httpbin.yaml
+```
+
+Check that the pods are running:
+
+```bash
+$ kubectl get po -l app=httpbin
+NAME READY STATUS RESTARTS AGE
+httpbin-85777b684b-8sqw5 1/1 Running 0 24s
+httpbin-85777b684b-pb26w 1/1 Running 0 24s
+httpbin-85777b684b-vpgwl 1/1 Running 0 24s
+```
+
+Then type the DNS name you set up in the previous step into a web browser, for example `http://gke.davecheney.com/`. You should see something like:
+
+![httpbin screenshot][8]
+
+You can delete the httpbin service now, or at any time, by running:
+
+```bash
+$ kubectl delete -f {{< param base_url >}}/examples/httpbin.yaml
+```
+
+## 2. Deploy jetstack/cert-manager
+
+**NOTE:** cert-manager is a powerful product that provides more functionality than this tutorial demonstrates.
+There are plenty of [other ways to deploy cert-manager][4], but they are out of scope.
+
+### Fetch the source manager deployment manifest
+
+To keep things simple, we skip cert-manager's Helm installation, and use the [supplied YAML manifests][5].
+
+```bash
+$ kubectl apply -f https://github.com/jetstack/cert-manager/releases/download/v1.5.4/cert-manager.yaml
+```
+
+When cert-manager is up and running you should see something like:
+
+```bash
+$ kubectl -n cert-manager get all
+NAME READY STATUS RESTARTS AGE
+pod/cert-manager-cainjector-74bb68d67c-8lb2f 1/1 Running 0 40s
+pod/cert-manager-f7f8bf74d-65ld9 1/1 Running 0 40s
+pod/cert-manager-webhook-645b8bdb7-2h5t6 1/1 Running 0 40s
+
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+service/cert-manager ClusterIP 10.48.13.252 9402/TCP 40s
+service/cert-manager-webhook ClusterIP 10.48.7.220 443/TCP 40s
+
+NAME READY UP-TO-DATE AVAILABLE AGE
+deployment.apps/cert-manager 1/1 1 1 40s
+deployment.apps/cert-manager-cainjector 1/1 1 1 40s
+deployment.apps/cert-manager-webhook 1/1 1 1 40s
+
+NAME DESIRED CURRENT READY AGE
+replicaset.apps/cert-manager-cainjector-74bb68d67c 1 1 1 40s
+replicaset.apps/cert-manager-f7f8bf74d 1 1 1 40s
+replicaset.apps/cert-manager-webhook-645b8bdb7 1 1 1 40s
+```
+
+### Deploy the Let's Encrypt cluster issuer
+
+cert-manager supports two different CRDs for configuration, an `Issuer`, which is scoped to a single namespace,
+and a `ClusterIssuer`, which is cluster-wide.
+
+For Contour to be able to serve HTTPS traffic for an Ingress in any namespace, use `ClusterIssuer`.
+Create a file called `letsencrypt-staging.yaml` with the following contents:
+
+```yaml
+apiVersion: cert-manager.io/v1
+kind: ClusterIssuer
+metadata:
+ name: letsencrypt-staging
+ namespace: cert-manager
+spec:
+ acme:
+ email: user@example.com
+ privateKeySecretRef:
+ name: letsencrypt-staging
+ server: https://acme-staging-v02.api.letsencrypt.org/directory
+ solvers:
+ - http01:
+ ingress:
+ class: contour
+```
+
+replacing `user@example.com` with your email address.
+This is the email address that Let's Encrypt uses to communicate with you about certificates you request.
+
+The staging Let's Encrypt server is not bound by [the API rate limits of the production server][2].
+This approach lets you set up and test your environment without worrying about rate limits.
+You can then repeat this step for a production Let's Encrypt certificate issuer.
+
+After you edit and save the file, deploy it:
+
+```bash
+$ kubectl apply -f letsencrypt-staging.yaml
+clusterissuer.cert-manager.io/letsencrypt-staging created
+```
+
+Wait for the `ClusterIssuer` to be ready:
+
+```bash
+$ kubectl get clusterissuer letsencrypt-staging
+NAME READY AGE
+letsencrypt-staging True 54s
+```
+
+## 3. Deploy your first HTTPS site using Ingress
+
+For this tutorial we deploy a version of Kenneth Reitz's [httpbin.org service][3].
+We start with the deployment.
+Copy the following to a file called `deployment.yaml`:
+
+```yaml
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ labels:
+ app: httpbin
+ name: httpbin
+spec:
+ replicas: 1
+ selector:
+ matchLabels:
+ app: httpbin
+ strategy:
+ rollingUpdate:
+ maxSurge: 1
+ maxUnavailable: 1
+ type: RollingUpdate
+ template:
+ metadata:
+ labels:
+ app: httpbin
+ spec:
+ containers:
+ - image: docker.io/kennethreitz/httpbin
+ name: httpbin
+ ports:
+ - containerPort: 8080
+ name: http
+ command: ["gunicorn"]
+ args: ["-b", "0.0.0.0:8080", "httpbin:app"]
+ dnsPolicy: ClusterFirst
+```
+
+Deploy to your cluster:
+
+```bash
+$ kubectl apply -f deployment.yaml
+deployment.apps/httpbin created
+$ kubectl get pod -l app=httpbin
+NAME READY STATUS RESTARTS AGE
+httpbin-67fd96d97c-8j2rr 1/1 Running 0 56m
+```
+
+Expose the deployment to the world with a Service. Create a file called `service.yaml` with
+the following contents:
+
+```yaml
+apiVersion: v1
+kind: Service
+metadata:
+ name: httpbin
+spec:
+ ports:
+ - port: 8080
+ protocol: TCP
+ targetPort: 8080
+ selector:
+ app: httpbin
+```
+
+and deploy:
+
+```bash
+$ kubectl apply -f service.yaml
+service/httpbin created
+$ kubectl get service httpbin
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+httpbin ClusterIP 10.48.6.155 8080/TCP 57m
+```
+
+Expose the Service to the world with Contour and an Ingress object. Create a file called `ingress.yaml` with
+the following contents:
+
+```yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: httpbin
+spec:
+ rules:
+ - host: httpbin.davecheney.com
+ http:
+ paths:
+ - pathType: Prefix
+ path: /
+ backend:
+ service:
+ name: httpbin
+ port:
+ number: 8080
+```
+
+The host name, `httpbin.davecheney.com` is a `CNAME` to the `gke.davecheney.com` record that was created in the first section, and must be created in the same place as the `gke.davecheney.com` record was.
+That is, in your cloud provider.
+This lets requests to `httpbin.davecheney.com` resolve to the external IP address of the Contour service.
+They are then forwarded to the Contour pods running in the cluster:
+
+```bash
+$ host httpbin.davecheney.com
+httpbin.davecheney.com is an alias for gke.davecheney.com.
+gke.davecheney.com has address 35.189.26.87
+```
+
+Change the value of `spec.rules.host` to something that you control, and deploy the Ingress to your cluster:
+
+```bash
+$ kubectl apply -f ingress.yaml
+ingress.networking.k8s.io/httpbin created
+$ kubectl get ingress httpbin
+NAME CLASS HOSTS ADDRESS PORTS AGE
+httpbin httpbin.davecheney.com 80 12s
+```
+
+Now you can type the host name of the service into a browser, or use curl, to verify it's deployed and everything is working:
+
+```bash
+$ curl http://httpbin.davecheney.com/get
+{
+ "args": {},
+ "headers": {
+ "Accept": "*/*",
+ "Content-Length": "0",
+ "Host": "htpbin.davecheney.com",
+ "User-Agent": "curl/7.58.0",
+ "X-Envoy-Expected-Rq-Timeout-Ms": "15000",
+ "X-Envoy-Internal": "true"
+ },
+ "origin": "10.152.0.2",
+ "url": "http://httpbin.davecheney.com/get"
+}
+```
+
+Excellent, it looks like everything is up and running serving traffic over HTTP.
+
+### Request a TLS certificate from Let's Encrypt
+
+Now it's time to use cert-manager to request a TLS certificate from Let's Encrypt.
+Do this by adding some annotations and a `tls:` section to the Ingress spec.
+
+We need to add the following annotations:
+
+- `cert-manager.io/cluster-issuer: letsencrypt-staging`: tells cert-manager to use the `letsencrypt-staging` cluster issuer you just created.
+- `kubernetes.io/tls-acme: "true"`: Tells cert-manager to do ACME TLS (what Let's Encrypt uses).
+- `ingress.kubernetes.io/force-ssl-redirect: "true"`: tells Contour to redirect HTTP requests to the HTTPS site.
+- `kubernetes.io/ingress.class: contour`: Tells Contour that it should handle this Ingress object.
+
+Using `kubectl edit ingress httpbin`:
+
+```yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: httpbin
+ annotations:
+ cert-manager.io/cluster-issuer: letsencrypt-staging
+ ingress.kubernetes.io/force-ssl-redirect: "true"
+ kubernetes.io/ingress.class: contour
+ kubernetes.io/tls-acme: "true"
+spec:
+ tls:
+ - secretName: httpbin
+ hosts:
+ - httpbin.davecheney.com
+ rules:
+ - host: httpbin.davecheney.com
+ http:
+ paths:
+ - pathType: Prefix
+ path: /
+ backend:
+ service:
+ name: httpbin
+ port:
+ number: 8080
+```
+
+The certificate is issued in the name of the hosts listed in the `tls:` section, `httpbin.davecheney.com` and stored in the secret `httpbin`.
+Behind the scenes, cert-manager creates a certificate CRD to manage the lifecycle of the certificate, and then a series of other CRDs to handle the challenge process.
+
+You can watch the progress of the certificate as it's issued:
+
+```bash
+$ kubectl describe certificate httpbin | tail -n 12
+Status:
+ Conditions:
+ Last Transition Time: 2019-11-07T00:37:55Z
+ Message: Waiting for CertificateRequest "httpbinproxy-1925286939" to complete
+ Reason: InProgress
+ Status: False
+ Type: Ready
+Events:
+ Type Reason Age From Message
+ ---- ------ ---- ---- -------
+ Normal GeneratedKey 26s cert-manager Generated a new private key
+ Normal Requested 26s cert-manager Created new CertificateRequest resource "httpbinproxy-1925286939"
+```
+
+Wait for the certificate to be issued:
+
+```bash
+$ kubectl describe certificate httpbin | grep -C3 "Certificate is up to date"
+Status:
+ Conditions:
+ Last Transition Time: 2019-11-06T23:47:50Z
+ Message: Certificate is up to date and has not expired
+ Reason: Ready
+ Status: True
+ Type: Ready
+```
+
+A `kubernetes.io/tls` secret is created with the `secretName` specified in the `tls:` field of the Ingress.
+
+```bash
+$ kubectl get secret httpbin
+NAME TYPE DATA AGE
+httpbin kubernetes.io/tls 2 3m
+```
+
+cert-manager manages the contents of the secret as long as the Ingress is present in your cluster.
+
+You can now visit your site, replacing `http://` with `https://` — and you get a huge security warning!
+This is because the certificate was issued by the Let's Encrypt staging servers and has a fake CA.
+This is so you can't accidentally use the staging servers to serve real certificates.
+
+```bash
+$ curl https://httpbin.davecheney.com/get
+curl: (60) SSL certificate problem: unable to get local issuer certificate
+More details here: https://curl.haxx.se/docs/sslcerts.html
+
+curl failed to verify the legitimacy of the server and therefore could not
+establish a secure connection to it. To learn more about this situation and
+how to fix it, please visit the web page mentioned above.
+```
+
+### Switch to Let's Encrypt Production
+
+To request a properly signed certificate from the Let's Encrypt production servers, we create a new `ClusterIssuer`, as before but with some modifications.
+
+Create a file called `letsencrypt-prod.yaml` with the following contents:
+
+```yaml
+apiVersion: cert-manager.io/v1
+kind: ClusterIssuer
+metadata:
+ name: letsencrypt-prod
+ namespace: cert-manager
+spec:
+ acme:
+ email: user@example.com
+ privateKeySecretRef:
+ name: letsencrypt-prod
+ server: https://acme-v02.api.letsencrypt.org/directory
+ solvers:
+ - http01:
+ ingress:
+ class: contour
+```
+
+again replacing `user@example.com` with your email address.
+
+Deploy:
+
+```bash
+$ kubectl apply -f letsencrypt-prod.yaml
+clusterissuer.cert-manager.io/letsencrypt-prod created
+```
+
+Now we use `kubectl edit ingress httpbin` to edit our Ingress to ask for a real certificate from `letsencrypt-prod`:
+
+```yaml
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: httpbin
+ annotations:
+ cert-manager.io/cluster-issuer: letsencrypt-prod
+spec:
+ ...
+```
+
+The certificate resource will transition to `Ready: False` while it's re-provisioned from the Let's Encrypt production servers, and then back to `Ready: True` once it's been provisioned:
+
+```bash
+$ kubectl describe certificate httpbin
+...
+Events:
+ Type Reason Age From Message
+ ---- ------ ---- ---- -------
+ ...
+ Normal Issuing 21s cert-manager Issuing certificate as Secret was previously issued by ClusterIssuer.cert-manager.io/letsencrypt-staging
+ Normal Reused 21s cert-manager Reusing private key stored in existing Secret resource "httpbin"
+ Normal Requested 21s cert-manager Created new CertificateRequest resource "httpbin-sjqbt"
+ Normal Issuing 18s (x2 over 48s) cert-manager The certificate has been successfully issued
+```
+
+Followed by:
+
+```bash
+$ kubectl get certificate httpbin -o wide
+NAME READY SECRET ISSUER STATUS AGE
+httpbin True httpbin letsencrypt-prod Certificate is up to date and has not expired 3m35s
+```
+
+Now revisiting our `https://httpbin.davecheney.com` site should show a valid, trusted, HTTPS certificate.
+
+```bash
+$ curl https://httpbin.davecheney.com/get
+{
+ "args": {},
+ "headers": {
+ "Accept": "*/*",
+ "Content-Length": "0",
+ "Host": "httpbin.davecheney.com",
+ "User-Agent": "curl/7.58.0",
+ "X-Envoy-Expected-Rq-Timeout-Ms": "15000",
+ "X-Envoy-Internal": "true"
+ },
+ "origin": "10.152.0.2",
+ "url": "https://httpbin.davecheney.com/get"
+}
+```
+
+![httpbin.davecheney.com screenshot][9]
+
+## Making cert-manager work with HTTPProxy
+
+cert-manager currently does not have a way to interact directly with HTTPProxy objects in order to respond to the HTTP01 challenge (See [#950][10] and [#951][11] for details).
+cert-manager, however, can be configured to request certificates automatically using a `Certificate` object.
+
+When cert-manager finds a `Certificate` object, it will implement the HTTP01 challenge by creating a new, temporary Ingress object that will direct requests from Let's Encrypt to temporary pods called 'solver pods'.
+These pods know how to respond to Let's Encrypt's challenge process for verifying you control the domain you're issuing certificates for.
+The Ingress resource as well as the solver pods are short lived and will only be available during the certificate request or renewal process.
+
+The result of the work steps described previously is a TLS secret, which can be referenced by a HTTPProxy.
+
+## Details
+
+To do this, we first need to create our HTTPProxy and Certificate objects.
+
+This example uses the hostname `httpbinproxy.davecheney.com`, remember to create that name before starting.
+
+Firstly, the HTTPProxy:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: httpbinproxy
+spec:
+ virtualhost:
+ fqdn: httpbinproxy.davecheney.com
+ tls:
+ secretName: httpbinproxy
+ routes:
+ - services:
+ - name: httpbin
+ port: 8080
+```
+
+This object will be marked as Invalid by Contour, since the TLS secret doesn't exist yet.
+Once that's done, create the Certificate object:
+
+```yaml
+apiVersion: cert-manager.io/v1
+kind: Certificate
+metadata:
+ name: httpbinproxy
+spec:
+ commonName: httpbinproxy.davecheney.com
+ dnsNames:
+ - httpbinproxy.davecheney.com
+ issuerRef:
+ name: letsencrypt-prod
+ kind: ClusterIssuer
+ secretName: httpbinproxy
+```
+
+Wait for the Certificate to be provisioned:
+
+```bash
+$ kubectl get certificate httpbinproxy -o wide
+NAME READY SECRET ISSUER STATUS AGE
+httpbinproxy True httpbinproxy letsencrypt-prod Certificate is up to date and has not expired 39s
+```
+
+Once cert-manager has fulfilled the HTTP01 challenge, you will have a `httpbinproxy` secret, that will contain the keypair.
+Contour will detect that the Secret exists and generate the HTTPProxy config.
+
+After that, you should be able to curl the new site:
+
+```bash
+$ curl https://httpbinproxy.davecheney.com/get
+{
+ "args": {},
+ "headers": {
+ "Accept": "*/*",
+ "Content-Length": "0",
+ "Host": "httpbinproxy.davecheney.com",
+ "User-Agent": "curl/7.54.0",
+ "X-Envoy-Expected-Rq-Timeout-Ms": "15000",
+ "X-Envoy-External-Address": "122.106.57.183"
+ },
+ "origin": "122.106.57.183",
+ "url": "https://httpbinproxy.davecheney.com/get"
+}
+```
+
+## Wrapping up
+
+Now that you've deployed your first HTTPS site using Contour and Let's Encrypt, deploying additional TLS enabled services is much simpler.
+Remember that for each HTTPS website you deploy, cert-manager will create a Certificate CRD that provides the domain name and the name of the target Secret.
+The TLS functionality will be enabled when the HTTPProxy contains the `tls:` stanza, and the referenced secret contains a valid keypair.
+
+See the [cert-manager docs][12] for more information.
+
+## Bonus points
+
+For bonus points, you can use a feature of Contour to automatically upgrade any HTTP request to the corresponding HTTPS site so you are no longer serving any traffic over insecure HTTP.
+
+To enable the automatic redirect from HTTP to HTTPS, add this annotation to your Ingress object.
+
+```
+metadata:
+ annotations:
+ ingress.kubernetes.io/force-ssl-redirect: "true"
+```
+Now any requests to the insecure HTTP version of your site get an unconditional 301 redirect to the HTTPS version:
+
+```
+$ curl -v http://httpbin.davecheney.com/get
+* Trying 35.189.26.87…
+* TCP_NODELAY set
+* Connected to httpbin.davecheney.com (35.189.26.87) port 80 (#0)
+> GET /get HTTP/1.1
+> Host: httpbin.davecheney.com
+> User-Agent: curl/7.58.0
+> Accept: */*
+>
+< HTTP/1.1 301 Moved Permanently
+< location: https://httpbin.davecheney.com/get
+< date: Tue, 20 Feb 2018 04:11:46 GMT
+< server: envoy
+< content-length: 0
+<
+* Connection #0 to host httpbin.davecheney.com left intact
+```
+
+__Note:__ For HTTPProxy resources this happens automatically without the need for an annotation.
+
+[0]: {{< param github_url >}}
+[1]: https://github.com/jetstack/cert-manager
+[2]: https://letsencrypt.org/docs/rate-limits/
+[3]: http://httpbin.org/
+[4]: https://docs.cert-manager.io/en/latest/getting-started/install/kubernetes.html
+[5]: https://github.com/jetstack/cert-manager/releases/download/v1.5.4/cert-manager.yaml
+[6]: https://letsencrypt.org/getting-started/
+[7]: ../deploy-options/#get-your-hostname-or-ip-address
+[8]: /img/cert-manager/httpbinhomepage.png
+[9]: /img/cert-manager/httpbin.png
+[10]: {{< param github_url >}}/issues/950
+[11]: {{< param github_url >}}/issues/951
+[12]: https://cert-manager.io/docs/usage/ingress/
diff --git a/site/content/docs/1.28/guides/deploy-aws-nlb.md b/site/content/docs/1.28/guides/deploy-aws-nlb.md
new file mode 100644
index 00000000000..af3f8df1019
--- /dev/null
+++ b/site/content/docs/1.28/guides/deploy-aws-nlb.md
@@ -0,0 +1,47 @@
+---
+title: Deploying Contour on AWS with NLB
+---
+
+This is an advanced deployment guide to configure Contour on AWS with the [Network Load Balancer (NLB)][1].
+This configuration has several advantages:
+
+1. NLBs are often cheaper. This is especially true for development. Idle LBs do not cost money.
+2. There are no extra network hops. Traffic goes to the NLB, to the node hosting Contour, and then to the target pod.
+3. Source IP addresses are retained. Envoy (running as part of Contour) sees the native source IP address and records this with an `X-Forwarded-For` header.
+
+## Moving parts
+
+- We run Envoy as a DaemonSet across the cluster and Contour as a deployment
+- The Envoy pod runs on host ports 80 and 443 on the node
+- Host networking means that traffic hits Envoy without transitioning through any other fancy networking hops
+- Contour also binds to 8001 for Envoy->Contour config traffic.
+
+## Deploying Contour
+
+1. [Clone the Contour repository][4] and cd into the repo
+2. Edit the Envoy service (`02-service-envoy.yaml`) in the `examples/contour` directory:
+ - Remove the existing annotation: `service.beta.kubernetes.io/aws-load-balancer-backend-protocol: tcp`
+ - Add the following annotation: `service.beta.kubernetes.io/aws-load-balancer-type: nlb`
+3. Run `kubectl apply -f examples/contour`
+
+This creates the `projectcontour` Namespace along with a ServiceAccount, RBAC rules, Contour Deployment and an Envoy DaemonSet.
+It also creates the NLB based loadbalancer for you.
+
+You can get the address of your NLB via:
+
+```
+$ kubectl get service envoy --namespace=projectcontour -o jsonpath='{.status.loadBalancer.ingress[0].hostname}'
+```
+
+## Test
+
+You can now test your NLB.
+
+1. Install a workload (see the kuard example in the [main deployment guide][2]).
+2. Look up the address for your NLB in the AWS console and enter it in your browser.
+ - Notice that Envoy fills out `X-Forwarded-For`, because it was the first to see the traffic directly from the browser.
+
+[1]: https://aws.amazon.com/blogs/aws/new-network-load-balancer-effortless-scaling-to-millions-of-requests-per-second/
+[2]: ../deploy-options/#testing-your-installation
+[3]: https://github.com/kubernetes/kubernetes/issues/52173
+[4]: {{< param github_url >}}/tree/{{< param branch >}}
diff --git a/site/content/docs/1.28/guides/deploy-aws-tls-nlb.md b/site/content/docs/1.28/guides/deploy-aws-tls-nlb.md
new file mode 100644
index 00000000000..7f4f83f685e
--- /dev/null
+++ b/site/content/docs/1.28/guides/deploy-aws-tls-nlb.md
@@ -0,0 +1,135 @@
+---
+title: AWS Network Load Balancer TLS Termination with Contour
+---
+
+## Motivation
+
+![diagram illustrating connection between network load balancer and contour](/img/aws-nlb-tls/fig.jpg){:class="img-fluid"}
+
+Managing TLS certificates (and related configuration) for production cluster workloads is both time consuming, and high risk. For example, storing multiple copies of a certificate secret key in the cluster may increases the chances of it being compromised. Additionally, TLS can be complicated to configure and implement properly.
+
+Traditionally, TLS termination at the load balancer step required using more expensive application load balancers (ALBs). AWS introduced TLS termination for network load balancers (NLBs) for enhanced security and cost effectiveness.
+
+The TLS implementation used by the AWS NLB is formally verified and maintained. Additionally, AWS Certificate Manager (ACM) is used, fully isolating your cluster from access to the private key.
+
+## Solution Overview
+
+An external client transmits a request to the NLB. The request is encrypted with TLS using the production (e.g., client facing) certificate, and on port 443.
+
+The NLB decrypts the request, and transmits it on to Envoy running in your cluster on port 8080. It follows the standard request routing configured within the cluster. Notably, the request received within the cluster includes the actual origin IP address of the external client.
+
+Alternate ports may be configured. End-to-end encryption technically requires the segment between the NLB and cluster pods be encrypted also. A follow-up post will describe the NLB originating TLS based on a cluster certificate.
+
+## Steps
+
+### Prerequisites
+
+1. Access to DNS records for domain name.
+
+[Review the docs on registering domains with AWS's Route 53.](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/registrar.html)
+
+An alternate DNS provider may be used, such as Google Domains or Namecheap.
+
+Later, a subdomain (e.g., demo-service.gcline.us) will be created, pointing to the NLB. Additionally, access to the DNS records is required to generate a TLS certificate for use by the NLB.
+
+3. Verify [Contour is installed in the cluster.](https://projectcontour.io/getting-started/)
+
+4. Install [AWS Load Balancer Controller.]( https://kubernetes-sigs.github.io/aws-load-balancer-controller/latest/deploy/installation/)
+
+Generally, setting up the Load Balancer Controller has two steps: enabling IAM roles for service accounts, and adding the controller to the cluster. The IAM role allows the controller in the Kubernetes cluster to manage AWS resources. [Learn more about IAM roles for service accounts.](https://docs.aws.amazon.com/eks/latest/userguide/iam-roles-for-service-accounts.html)
+
+### Configure
+
+1. Generate TLS Certificate
+
+Create a public TLS certificate for the domain using AWS Certificate Manager (ACM). This is streamlined when the domain is managed by Route 53. Review the [AWS Certificate Manager Docs.](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html#request-public-console)
+
+The domain name on the TLS certificate must correspond to the planned domain name for the kubernetes service. The domain name may be specified explicitly (e.g., tls-demo.gcline.us), or a wildcard certificate can be used (e.g., *.gcline.us).
+
+If the domain is registered with Route53, the TLS certificate request will automatically be approved. Otherwise, follow ACM console the instructions to create a DNS record to validate the domain.
+
+After validation, the certificate will be available for use in your AWS account.
+
+Note the ARN of the certificate, which uniquely identifies it in kubernetes config files.
+
+![screenshot indicating location of ARN value in web console](/img/aws-nlb-tls/acm-arn.png){:class="img-fluid"}
+
+2. Create Envoy Service with new NLB
+
+Contour expects a kubernetes service pointing to Envoy. Add annotations to the service to enable NLB TLS termination, before the traffic reaches Envoy. The annotations are actioned by the load balancer controller. [Review all the NLB annotations on GitHub.](https://kubernetes-sigs.github.io/aws-load-balancer-controller/latest/guide/service/annotations/)
+
+| annotation name | value | meaning |
+| ----- | --- | ----- |
+| service.beta.kubernetes.io/aws-load-balancer-type | external | explicitly requires an NLB, instead of an ALB |
+| service.beta.kubernetes.io/aws-load-balancer-nlb-target-type | ip | route traffic directly to the pod IP |
+| service.beta.kubernetes.io/aws-load-balancer-scheme | internet-facing | An internet-facing load balancer has a publicly resolvable DNS name |
+| service.beta.kubernetes.io/aws-load-balancer-ssl-cert | "arn:aws:acm:..." | identifies the TLS certificate used by the NLB |
+| service.beta.kubernetes.io/aws-load-balancer-ssl-ports | 443 | determines the port the NLB should listen for TLS traffic on|
+
+Example:
+
+```
+apiVersion: v1
+kind: Service
+metadata:
+ name: envoy
+ namespace: projectcontour
+ annotations:
+ service.beta.kubernetes.io/aws-load-balancer-type: external
+ service.beta.kubernetes.io/aws-load-balancer-nlb-target-type: ip
+ service.beta.kubernetes.io/aws-load-balancer-scheme: internet-facing
+ service.beta.kubernetes.io/aws-load-balancer-ssl-cert: "arn:aws:acm:us-east-2:185309785115:certificate/7610ed7d-5a81-4ea2-a18a-7ba1606cca3e"
+ service.beta.kubernetes.io/aws-load-balancer-ssl-ports: "443"
+spec:
+ externalTrafficPolicy: Local
+ ports:
+ - port: 443
+ targetPort: 8080
+ name: http
+ protocol: TCP
+ selector:
+ app: envoy
+ type: LoadBalancer
+```
+
+*Note:* Don't modify an existing service to add NLB TLS termination. This may result in unexpected behavior, such as duplicate NLB resources or incorrect NLB configuration.
+
+3. Configure DNS
+
+**Get domain name using kubectl.**
+
+The service name and namespace were defined above.
+
+```
+kubectl get svc envoy --namespace projectcontour
+```
+
+```
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+envoy LoadBalancer 10.100.24.154 a7ea2bbde8a164036a7e4c1ed5700cdf-154fb911d990bb1f.elb.us-east-2.amazonaws.com 443:31606/TCP 40d
+```
+
+Note the last 4 digits of the domain name for the NLB. For example, "bb1f".
+
+**Setup DNS alias for NLB**
+
+Create a DNS record pointing from a friendly name (e.g., tls-demo.gcline.us) to the NLB domain (e.g., bb1f.elb.us-east-2.amazonaws.com).
+
+For AWS's Route 53, follow the instructions below. If you use a different DNS provider, follow their instructions for [creating a CNAME record](https://docs.digitalocean.com/products/networking/dns/how-to/manage-records/#cname-records).
+
+First, create a new record in Route 53.
+
+Use the "A" record type, and enable the ["alias" option.](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/resource-record-sets-values-alias.html) This option attaches the DNS record to the AWS resource, without requiring an extra lookup step for clients.
+
+Select the NLB resource. Double check the region, and use the last 4 digits (noted earlier) to select the proper resource.
+
+![screenshot of Route 53 New Record Console](/img/aws-nlb-tls/record.png){:class="img-fluid"}
+
+### Verify
+
+Attempt to access the NLB domain at port 443 with HTTPS/TLS. Is the connection successful? What certificate is used? Does it reach the expected endpoint within the cluster?
+
+### Next Steps
+
+Create a second TLS certificate within the cluster, for securing connections between the NLB and pods. A guide on this topic is forthcoming.
+
diff --git a/site/content/docs/1.28/guides/external-authorization.md b/site/content/docs/1.28/guides/external-authorization.md
new file mode 100644
index 00000000000..74076ede518
--- /dev/null
+++ b/site/content/docs/1.28/guides/external-authorization.md
@@ -0,0 +1,538 @@
+---
+title: External Authorization Support
+---
+
+Starting in version 1.9, Contour supports routing client requests to an
+external authorization server. This feature can be used to centralize
+client authorization so that applications don't have to implement their
+own authorization mechanisms.
+
+## Authorization Architecture
+
+An external authorization server is a server that implements the Envoy
+external authorization [GRPC protocol][3]. Contour supports any server
+that implements this protocol.
+
+You can bind an authorization server to Contour by creating a
+[`ExtensionService`][4] resource.
+This resource tells Contour the service exists, and that it should
+program Envoy with an upstream cluster directing traffic to it.
+Note that the `ExtensionService` resource just binds the server; at this
+point Contour doesn't assume that the server is an authorization server.
+
+Once you have created `ExtensionService` resource, you can bind it to a
+particular application by referencing it in a [`HTTPProxy`][5] resource.
+In the `virtualhost` field, a new `authorization` field specifies the name
+of an `ExtensionService` to bind for the virtual host.
+When you specify a resource name here, Contour will program Envoy to
+send authorization checks to the extension service cluster before routing
+the request to the upstream application.
+
+## Authorization Request Flow
+
+It is helpful to have a mental model of how requests flow through the various
+servers involved in authorizing HTTP requests.
+The flow diagram below shows the actors that participate in the successful
+authorization of an HTTP request.
+Note that in some cases, these actors can be combined into a single
+application server.
+For example, there is no requirement for the external authorization server to
+be a separate application from the authorization provider.
+
+
+
+
+
+
+A HTTP Client generates an HTTP request and sends it to
+an Envoy instance that Contour has programmed with an external
+authorization configuration.
+Envoy holds the HTTP request and sends an authorization check request
+to the Authorization server that Contour has bound to the virtual host.
+The Authorization server may be able to verify the request locally, but in
+many cases it will need to make additional requests to an Authorization
+Provider server to verify or obtain an authorization token.
+
+In this flow, the ExtAuth server is able to authorize the request, and sends an
+authorization response back to the Proxy.
+The response includes the authorization status, and a set of HTTP headers
+modifications to make to the HTTP request.
+Since this authorization was successful, the Proxy modifies the request and
+forwards it to the application.
+If the authorization was not successful, the Proxy would have immediately
+responded to the client with an HTTP error.
+
+## Using the Contour Authorization Server
+
+The Contour project has built a simple authorization server named
+[`contour-authserver`][1]. `contour-authserver` supports an authorization
+testing server, and an HTTP basic authorization server that accesses
+credentials stored in [htpasswd][2] format.
+
+To get started, ensure that Contour is deployed and that you have
+[cert-manager][6] installed in your cluster so that you can easily issue
+self-signed TLS certificates.
+
+At this point, we should also create a cluster-wide self-signed certificate
+issuer, just to make it easier to provision TLS certificates later:
+
+```bash
+$ kubectl apply -f - <` with the appropriate version of Go and BoringCrypto, see [here][10] for version specifics):
+
+```bash
+make container BUILD_CGO_ENABLED=1 BUILD_BASE_IMAGE=goboring/golang: BUILD_EXTRA_GO_LDFLAGS="-linkmode=external -extldflags=-static"
+```
+
+The command above can be broken down as follows:
+- `make container` invokes the container image build target
+- `BUILD_CGO_ENABLED=1` ensures `cgo` is enabled in the Contour compilation process
+- `BUILD_BASE_IMAGE=goboring/golang:` ensures we use the BoringCrypto flavor of Go
+- `BUILD_EXTRA_GO_LDFLAGS` contains the additional linker flags we need to perform a static build
+ - `-linkmode=external` tells the Go linker to use an external linker
+ - `-extldflags=-static"` passes the `-static` flag to the external link to ensure a statically linked executable is produced
+
+The container image build process should fail before export of the `contour` binary to the final image if the compiled binary is not statically linked.
+
+### Validation
+
+To be fully sure the produced `contour` binary has been compiled with BoringCrypto you must remove the `-s` flag from the base Contour `Makefile` to stop stripping symbols and run through the build process above.
+Then you will be able to inspect the `contour` binary with `go tool nm` to check for symbols containing the string `_Cfunc__goboringcrypto_`.
+Also, you can use the program [rsc.io/goversion][21]. It will report the crypto implementation used by a given binary when invoked with the `-crypto` flag.
+
+Once you have a `projectcontour/contour` image built, you can re-tag it if needed, push the image to a registry, and reference it in a Contour deployment to use it!
+
+## Building Envoy
+
+Envoy has support for building in a FIPS compliant mode as [documented here][11].
+The upstream project does not distribute a FIPS compliant Envoy container image, but combining the documented process with the processes for building the Envoy executable and container image, we can produce one.
+
+Again we will need the Envoy source code checked out to the version to build and Docker installed on your computer.
+The simplest way to build Envoy without having to learn [Bazel][12] and set up a C++ toolchain on your computer is to build using the Envoy build container image which contains the necessary tools pre-installed.
+Note that if you do build with FIPS mode outside of the build container, you can only do so on a Linux-amd64 architecture.
+
+We can first compile the Envoy binary by running the following in a `bash` shell from the Envoy source directory:
+
+```bash
+BAZEL_BUILD_EXTRA_OPTIONS="--define boringssl=fips" ENVOY_DOCKER_BUILD_DIR= ./ci/run_envoy_docker.sh './ci/do_ci.sh bazel.release //test/exe:envoy_static_test'
+```
+
+*This command mimics the Envoy release CI process with the target `bazel.release` but differs in only running a single test for brevity. You may omit the `//test/exe:envoy_static_test` test entirely to run the full suite of Envoy tests.*
+
+Replace `` with a directory you would like the build output to be placed on your host computer.
+
+Once that build completes, you should have a file named `release.tar.zst` in your specified output directory.
+This file is a [Zstandard](https://github.com/facebook/zstd) compressed archive containing the compiled Envoy release and debug binaries.
+If you would like to build an image with Envoy according to your own specifications, you can unpack the resulting archive and you will find a stripped Envoy binary in the root and an unstripped Envoy binary with debug info in the `dbg` directory.
+
+To build an image matching the canonical Envoy upstream release image ([`envoyproxy/envoy`][13]), run the following:
+
+*Note: You will need a recent version of Docker/BuildKit that supports Zstandard decompression.*
+
+```bash
+# Make ./linux/amd64 directories.
+mkdir -p ./linux/amd64
+# Copy Zstandard archive from build step.
+cp -a /envoy/x64/bin/release.tar.zst ./linux/amd64/release.tar.zst
+# Run the Docker image build.
+docker build -f ./ci/Dockerfile-envoy --target envoy .
+```
+
+Once you have an image built, you can tag it as needed, push the image to a registry, and use it in an Envoy deployment.
+
+## Configuring TLS Ciphers
+
+Now that we have Contour and Envoy compiled with BoringCrypto, we can turn our attention to ensuring encrypted communication paths in Contour are configured to use FIPS approved cryptographic algorithms.
+Using a FIPS flavor of Envoy will do most of the heavy lifting here without any user configuration needed.
+
+The critical communication paths and how they are set up to be FIPS compliant are enumerated below:
+- Contour -> k8s API
+ - Contour uses [`client-go`][14] to communicate with the k8s API
+ - `client-go` uses the default Golang cipher suites configuration
+ - When compiled with BoringCrypto Go, this set of ciphers is FIPS compliant and not configurable by users
+- Envoy -> Contour xDS Server, extension services, upstream services
+ - A FIPS compliant build of Envoy will choose FIPS approved TLS ciphers when negotiating TLS 1.2 as documented [here][15]
+ - The set of ciphers is not configurable
+- TLS client -> Envoy
+ - As of [Contour 1.13.0][16], the ciphers Envoy will accept as a server when negotiating TLS 1.2 are configurable
+ - The [default set of ciphers Contour configures][17] includes some ciphers that are not FIPS approved
+ - Users must configure FIPS approved ciphers from the list [here][15]
+
+[0]: https://csrc.nist.gov/publications/detail/fips/140/2/final
+[1]: https://csrc.nist.gov/projects/testing-laboratories
+[2]: https://boringssl.googlesource.com/boringssl/
+[3]: https://boringssl.googlesource.com/boringssl/+/master/crypto/fipsmodule/FIPS.md
+[4]: https://go.googlesource.com/go/+/dev.boringcrypto/README.boringcrypto.md
+[5]: https://hub.docker.com/r/projectcontour/contour
+[6]: https://www.gnu.org/software/make/
+[7]: https://www.docker.com/
+[8]: {{< param github_url >}}/blob/main/Dockerfile
+[9]: https://hub.docker.com/r/goboring/golang/
+[10]: https://go.googlesource.com/go/+/dev.boringcrypto/misc/boring/README.md#version-strings
+[11]: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/security/ssl.html#fips-140-2
+[12]: https://bazel.build/
+[13]: https://hub.docker.com/r/envoyproxy/envoy
+[14]: https://github.com/kubernetes/client-go
+[15]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/extensions/transport_sockets/tls/v3/common.proto#envoy-v3-api-field-extensions-transport-sockets-tls-v3-tlsparameters-cipher-suites
+[16]: https://github.com/projectcontour/contour/releases/tag/v1.13.0
+[17]: https://pkg.go.dev/github.com/projectcontour/contour/pkg/config#pkg-variables
+[18]: https://pkg.go.dev/internal/goexperiment@go1.19
+[19]: https://go-boringcrypto.storage.googleapis.com/
+[20]: https://go.googlesource.com/go/+/dev.boringcrypto/misc/boring/README.md#releases
+[21]: https://godoc.org/rsc.io/goversion
\ No newline at end of file
diff --git a/site/content/docs/1.28/guides/gatekeeper.md b/site/content/docs/1.28/guides/gatekeeper.md
new file mode 100644
index 00000000000..fa6d01cab41
--- /dev/null
+++ b/site/content/docs/1.28/guides/gatekeeper.md
@@ -0,0 +1,456 @@
+---
+title: Using Gatekeeper as a validating admission controller with Contour
+---
+
+This tutorial demonstrates how to use [Gatekeeper](https://github.com/open-policy-agent/gatekeeper) as a validating admission controller for Contour.
+
+Gatekeeper is a project that enables users to define flexible policies for Kubernetes resources using [Open Policy Agent (OPA)](https://www.openpolicyagent.org/) that are enforced when those resources are created/updated via the Kubernetes API.
+
+The benefits of using Gatekeeper with Contour are:
+- Immediate feedback for the user when they try to create an `HTTPProxy` with an invalid spec. Instead of having to check the `HTTPProxy`'s status after creation for a possible error message, the create is rejected and the user is immediately provided with a reason for the rejection.
+- User-defined policies for `HTTPProxy` specs. For example, the Contour admin can define policies to enforce maximum limits on timeouts and retries, disallow certain FQDNs, etc.
+
+## Prerequisites
+
+- A Kubernetes cluster with a minimum version of 1.14 (to enable webhook timeouts for Gatekeeper).
+- Cluster-admin permissions
+
+## Deploy Contour
+
+Run:
+
+```bash
+$ kubectl apply -f {{< param base_url >}}/quickstart/contour.yaml
+```
+
+This creates a `projectcontour` namespace and sets up Contour as a deployment and Envoy as a daemonset, with communication between them secured by mutual TLS.
+
+Check the status of the Contour pods with this command:
+
+```bash
+$ kubectl -n projectcontour get pods -l app=contour
+NAME READY STATUS RESTARTS AGE
+contour-8596d6dbd7-9nrg2 1/1 Running 0 32m
+contour-8596d6dbd7-mmtc8 1/1 Running 0 32m
+```
+
+If installation was successful, all pods should reach `Running` status shortly.
+
+## Deploy Gatekeeper
+
+The following instructions are summarized from the [Gatekeeper documentation](https://github.com/open-policy-agent/gatekeeper#installation-instructions).
+If you already have Gatekeeper running in your cluster, you can skip this section.
+
+Run:
+
+```bash
+$ kubectl apply -f https://raw.githubusercontent.com/open-policy-agent/gatekeeper/master/deploy/gatekeeper.yaml
+```
+
+This creates a `gatekeeper-system` namespace and sets up the Gatekeeper controller manager and audit deployments using the latest Gatekeeper release.
+
+Check the status of the Gatekeeper pods with this command:
+
+```bash
+$ kubectl -n gatekeeper-system get pods
+NAME READY STATUS RESTARTS AGE
+gatekeeper-audit-67dfc46db6-kjcmc 1/1 Running 0 40m
+gatekeeper-controller-manager-7cbc758844-64hhn 1/1 Running 0 40m
+gatekeeper-controller-manager-7cbc758844-c4dkd 1/1 Running 0 40m
+gatekeeper-controller-manager-7cbc758844-xv9jn 1/1 Running 0 40m
+```
+
+If installation was successful, all pods should reach `Running` status shortly.
+
+## Configure Gatekeeper
+
+### Background
+
+Gatekeeper uses the [OPA Constraint Framework](https://github.com/open-policy-agent/frameworks/tree/master/constraint) to define and enforce policies.
+This framework has two key types: `ConstraintTemplate` and `Constraint`.
+A `ConstraintTemplate` defines a reusable OPA policy, along with the parameters that can be passed to it when it is instantiated.
+When a `ConstraintTemplate` is created, Gatekeeper automatically creates a custom resource definition (CRD) to represent it in the cluster.
+
+A `Constraint` is an instantiation of a `ConstraintTemplate`, which tells Gatekeeper to apply it to specific Kubernetes resource types (e.g. `HTTPProxy`) and provides any relevant parameter values.
+A `Constraint` is defined as an instance of the CRD representing the associated `ConstraintTemplate`.
+
+We'll now look at some examples to make these concepts concrete.
+
+### Configure resource caching
+
+First, Gatekeeper needs to be configured to store all `HTTPProxy` resources in its internal cache, so that existing `HTTPProxy` resources can be referenced within constraint template policies.
+This is essential for being able to define constraints that look across all `HTTPProxies` -- for example, to verify FQDN uniqueness.
+
+Create a file called `config.yml` containing the following YAML:
+
+```yaml
+apiVersion: config.gatekeeper.sh/v1alpha1
+kind: Config
+metadata:
+ name: config
+ namespace: "gatekeeper-system"
+spec:
+ sync:
+ syncOnly:
+ - group: "projectcontour.io"
+ version: "v1"
+ kind: "HTTPProxy"
+```
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f config.yml
+```
+
+Note that if you already had Gatekeeper running in your cluster, you may already have the `Config` resource defined.
+In that case, you'll need to edit the existing resource to add `HTTPProxy` to the `spec.sync.syncOnly` list.
+
+### Configure HTTPProxy validations
+
+The first constraint template and constraint that we'll define are what we'll refer to as a **validation**.
+These are rules for `HTTPProxy` specs that Contour universally requires to be true.
+In this example, we'll define a constraint template and constraint to enforce that all `HTTPProxies` must have a unique FQDN.
+
+Create a file called `unique-fqdn-template.yml` containing the following YAML:
+
+```yaml
+apiVersion: templates.gatekeeper.sh/v1beta1
+kind: ConstraintTemplate
+metadata:
+ name: httpproxyuniquefqdn
+spec:
+ crd:
+ spec:
+ names:
+ kind: HTTPProxyUniqueFQDN
+ listKind: HTTPProxyUniqueFQDNList
+ plural: HTTPProxyUniqueFQDNs
+ singular: HTTPProxyUniqueFQDN
+ targets:
+ - target: admission.k8s.gatekeeper.sh
+ rego: |
+ package httpproxy.uniquefqdn
+
+ violation[{"msg": msg, "other": sprintf("%v/%v", [other.metadata.namespace, other.metadata.name])}] {
+ got := input.review.object.spec.virtualhost.fqdn
+ other := data.inventory.namespace[_]["projectcontour.io/v1"]["HTTPProxy"][_]
+ other.spec.virtualhost.fqdn = got
+
+ not same(other, input.review.object)
+ msg := "HTTPProxy must have a unique spec.virtualhost.fqdn"
+ }
+
+ same(a, b) {
+ a.metadata.namespace == b.metadata.namespace
+ a.metadata.name == b.metadata.name
+ }
+```
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f unique-fqdn-template.yml
+```
+
+Within a few seconds, you'll see that a corresponding CRD has been created in the cluster:
+
+```bash
+$ kubectl get crd httpproxyuniquefqdn.constraints.gatekeeper.sh
+NAME CREATED AT
+httpproxyuniquefqdn.constraints.gatekeeper.sh 2020-08-13T16:08:57Z
+```
+
+Now, create a file called `unique-fqdn-constraint.yml` containing the following YAML:
+
+```yaml
+apiVersion: constraints.gatekeeper.sh/v1beta1
+kind: HTTPProxyUniqueFQDN
+metadata:
+ name: httpproxy-unique-fqdn
+spec:
+ match:
+ kinds:
+ - apiGroups: ["projectcontour.io"]
+ kinds: ["HTTPProxy"]
+```
+
+Note that the `Kind` of this resource corresponds to the new CRD.
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f unique-fqdn-constraint.yml
+```
+
+Now, let's create some `HTTPProxies` to see the validation in action.
+
+Create a file called `httpproxies.yml` containing the following YAML:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: demo
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: demo.projectcontour.io
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: demo2
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: demo.projectcontour.io
+```
+
+Note that both `HTTPProxies` have the same FQDN.
+
+Apply the YAML:
+
+```bash
+$ kubectl apply -f httpproxies.yml
+```
+
+You should see something like:
+```
+httpproxy.projectcontour.io/demo created
+Error from server ([denied by httpproxy-unique-fqdn] HTTPProxy must have a unique FQDN): error when creating "httpproxies.yml": admission webhook "validation.gatekeeper.sh" denied the request: [denied by httpproxy-unique-fqdn] HTTPProxy must have a unique FQDN
+```
+
+The first `HTTPProxy` was created successfully, because there was not already an existing proxy with the `demo.projectcontour.io` FQDN.
+However, when the second `HTTPProxy` was submitted, Gatekeeper rejected its creation because it used the same FQDN as the first one.
+
+### Configure HTTPProxy policies
+
+The next constraint template and constraint that we'll create are what we refer to as a **policy**.
+These are rules for `HTTPProxy` specs that an individual Contour administrator may want to enforce for their cluster, but that are not explicitly required by Contour itself.
+In this example, we'll define a constraint template and constraint to enforce that all `HTTPProxies` can be configured with at most five retries for any route.
+
+Create a file called `retry-count-range-template.yml` containing the following YAML:
+
+```yaml
+apiVersion: templates.gatekeeper.sh/v1beta1
+kind: ConstraintTemplate
+metadata:
+ name: httpproxyretrycountrange
+spec:
+ crd:
+ spec:
+ names:
+ kind: HTTPProxyRetryCountRange
+ listKind: HTTPProxyRetryCountRangeList
+ plural: HTTPProxyRetryCountRanges
+ singular: HTTPProxyRetryCountRange
+ scope: Namespaced
+ validation:
+ openAPIV3Schema:
+ properties:
+ min:
+ type: integer
+ max:
+ type: integer
+ targets:
+ - target: admission.k8s.gatekeeper.sh
+ rego: |
+ package httpproxy.retrycountrange
+
+ # build a set of all the retry count values
+ retry_counts[val] {
+ val := input.review.object.spec.routes[_].retryPolicy.count
+ }
+
+ # is there a retry count value that's greater than the allowed max?
+ violation[{"msg": msg}] {
+ retry_counts[_] > input.parameters.max
+ msg := sprintf("retry count must be less than or equal to %v", [input.parameters.max])
+ }
+
+ # is there a retry count value that's less than the allowed min?
+ violation[{"msg": msg}] {
+ retry_counts[_] < input.parameters.min
+ msg := sprintf("retry count must be greater than or equal to %v", [input.parameters.min])
+ }
+```
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f retry-count-range-template.yml
+```
+
+Again, within a few seconds, you'll see that a corresponding CRD has been created in the cluster:
+
+```bash
+$ kubectl get crd httpproxyretrycountrange.constraints.gatekeeper.sh
+NAME CREATED AT
+httpproxyretrycountrange.constraints.gatekeeper.sh 2020-08-13T16:12:10Z
+```
+
+Now, create a file called `retry-count-range-constraint.yml` containing the following YAML:
+
+```yaml
+apiVersion: constraints.gatekeeper.sh/v1beta1
+kind: HTTPProxyRetryCountRange
+metadata:
+ name: httpproxy-retry-count-range
+spec:
+ match:
+ kinds:
+ - apiGroups: ["projectcontour.io"]
+ kinds: ["HTTPProxy"]
+ namespaces:
+ - my-namespace
+ parameters:
+ max: 5
+```
+
+Note that for this `Constraint`, we've added a `spec.match.namespaces` field which defines that this policy should only be applied to `HTTPProxies` created in the `my-namespace` namespace.
+If this `namespaces` matcher is not specified, then the `Constraint` applies to all namespaces.
+You can read more about `Constraint` matchers on the [Gatekeeper website](https://github.com/open-policy-agent/gatekeeper#constraints).
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f retry-count-range-constraint.yml
+```
+
+Now, let's create some `HTTPProxies` to see the policy in action.
+
+Create a namespace called `my-namespace`:
+
+```bash
+$ kubectl create namespace my-namespace
+namespace/my-namespace created
+```
+
+Create a file called `httpproxy-retries.yml` containing the following YAML:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: demo-retries
+ namespace: my-namespace
+spec:
+ virtualhost:
+ fqdn: retries.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /foo
+ services:
+ - name: s1
+ port: 80
+ retryPolicy:
+ count: 6
+```
+
+Apply the YAML:
+
+```bash
+$ kubectl apply -f httpproxy-retries.yml
+```
+
+You should see something like:
+```
+Error from server ([denied by httpproxy-retry-count-range] retry count must be less than or equal to 5): error when creating "proxy-retries.yml": admission webhook "validation.gatekeeper.sh" denied the request: [denied by httpproxy-retry-count-range] retry count must be less than or equal to 5
+```
+
+Now, change the `count` field on the last line of `httpproxy-retries.yml` to have a value of `5`. Save the file, and apply it again:
+
+```bash
+$ kubectl apply -f httpproxy-retries.yml
+```
+
+Now the `HTTPProxy` creates successfully*.
+
+_* Note that the HTTPProxy is still marked invalid by Contour after creation because the service `s1` does not exist, but that's outside the scope of this guide._
+
+Finally, create a file called `httpproxy-retries-default.yml` containing the following YAML:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: demo-retries
+ namespace: default
+spec:
+ virtualhost:
+ fqdn: default.retries.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /foo
+ services:
+ - name: s1
+ port: 80
+ retryPolicy:
+ count: 6
+```
+
+Remember that our `Constraint` was defined to apply only to the `my-namespace` namespace, so it should not block the creation of this proxy, even though it has a retry policy count outside the allowed range.
+
+Apply the YAML:
+
+```bash
+$ kubectl apply -f httpproxy-retries-default.yml
+```
+
+The `HTTPProxy` creates successfully.
+
+## Gatekeeper Audit
+
+We've seen how Gatekeeper constraints can enforce constraints when a user tries to create a new `HTTPProxy`. Now let's look at how constraints can be applied to pre-existing resources in the cluster.
+
+Gatekeeper has an audit functionality, that periodically (every `60s` by default) checks all existing resources against the relevant set of constraints. Any violations are reported in the `Constraint` custom resource's `status.violations` field. This allows an administrator to periodically review & correct any pre-existing misconfigurations, while not having to worry about breaking existing resources when rolling out a new or updated constraint.
+
+To try this out, let's revisit the previous example, and change our constraint to allow a maximum retry count of four.
+
+Edit `retry-count-range-constraint.yml` and change the `max` field to have a value of `4`. Save the file.
+
+Apply it to the cluster:
+
+```bash
+$ kubectl apply -f retry-count-range-constraint.yml
+```
+
+We know that the `demo-retries` proxy has a route with a `retryPolicy.count` of `5`. This should now be invalid according to the updated constraint.
+
+Wait up to `60s` for the next periodic audit to finish, then run:
+
+```bash
+$ kubectl describe httpproxyretrycountrange httpproxy-retry-count-range
+```
+
+You should see something like:
+
+```
+...
+Status:
+ ...
+ Violations:
+ Enforcement Action: deny
+ Kind: HTTPProxy
+ Message: retry policy count must be less than or equal to 4
+ Name: demo-retries
+ Namespace: my-namespace
+```
+
+However, our `HTTPProxy` remains in the cluster and can continue to route requests, and the user can remediate the proxy to bring it inline with the policy on their own timeline.
+
+## Next steps
+
+Contour has a [growing library](https://github.com/projectcontour/contour/tree/main/examples/gatekeeper) of Gatekeeper constraint templates and constraints, for both **validations** and **policies**.
+
+If you're using Gatekeeper, we recommend that you apply all of the **validations** we've defined, since these rules are already being checked internally by Contour and reported as status errors/invalid proxies.
+Using the Gatekeeper constraints will only improve the user experience since users will get earlier feedback if their proxies are invalid.
+The **validations** can be found in `examples/gatekeeper/validations`.
+
+
+You should take more of a pick-and-choose approach to our sample **policies**, since every organization will have different policy needs.
+Feel free to use any/all/none of them, and augment them with your own policies if applicable.
+The sample **policies** can be found in `examples/gatekeeper/policies`.
+
+And of course, if you do develop any new constraints that you think may be useful for the broader Contour community, we welcome contributions!
diff --git a/site/content/docs/1.28/guides/gateway-api.md b/site/content/docs/1.28/guides/gateway-api.md
new file mode 100644
index 00000000000..0759696d3f6
--- /dev/null
+++ b/site/content/docs/1.28/guides/gateway-api.md
@@ -0,0 +1,210 @@
+---
+title: Using Gateway API with Contour
+---
+
+This tutorial walks through an example of using [Gateway API][1] with Contour.
+See the [Contour reference documentation][5] for more information on Contour's Gateway API support.
+
+### Prerequisites
+The following prerequisites must be met before following this guide:
+
+- A working [Kubernetes][2] cluster. Refer to the [compatibility matrix][3] for cluster version requirements.
+- The [kubectl][4] command-line tool, installed and configured to access your cluster.
+
+## Deploy Contour with Gateway API enabled
+
+First, deploy Contour with Gateway API enabled.
+This can be done using either [static or dynamic provisioning][6].
+
+### Option #1: Statically provisioned
+
+Create Gateway API CRDs:
+```shell
+$ kubectl apply -f {{< param github_raw_url>}}/{{< param branch >}}/examples/gateway/00-crds.yaml
+```
+
+Create a GatewayClass:
+```shell
+kubectl apply -f - <}}/quickstart/contour.yaml
+```
+This command creates:
+
+- Namespace `projectcontour` to run Contour
+- Contour CRDs
+- Contour RBAC resources
+- Contour Deployment / Service
+- Envoy DaemonSet / Service
+- Contour ConfigMap
+
+Update the Contour configmap to enable Gateway API processing by specifying a gateway controller name, and restart Contour to pick up the config change:
+
+```shell
+kubectl apply -f - <}}/quickstart/contour-gateway-provisioner.yaml
+```
+
+This command creates:
+
+- Namespace `projectcontour` to run the Gateway provisioner
+- Contour CRDs
+- Gateway API CRDs
+- Gateway provisioner RBAC resources
+- Gateway provisioner Deployment
+
+Create a GatewayClass:
+
+```shell
+kubectl apply -f - <}}/{{< param branch >}}/examples/example-workload/gatewayapi/kuard/kuard.yaml
+```
+This command creates:
+
+- A Deployment named `kuard` in the default namespace to run kuard as the test application.
+- A Service named `kuard` in the default namespace to expose the kuard application on TCP port 80.
+- An HTTPRoute named `kuard` in the default namespace, attached to the `contour` Gateway, to route requests for `local.projectcontour.io` to the kuard service.
+
+Verify the kuard resources are available:
+```shell
+$ kubectl get po,svc,httproute -l app=kuard
+NAME READY STATUS RESTARTS AGE
+pod/kuard-798585497b-78x6x 1/1 Running 0 21s
+pod/kuard-798585497b-7gktg 1/1 Running 0 21s
+pod/kuard-798585497b-zw42m 1/1 Running 0 21s
+
+NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
+service/kuard ClusterIP 172.30.168.168 80/TCP 21s
+
+NAME HOSTNAMES
+httproute.gateway.networking.k8s.io/kuard ["local.projectcontour.io"]
+```
+
+## Test Routing
+
+_Note, for simplicity and compatibility across all platforms we'll use `kubectl port-forward` to get traffic to Envoy, but in a production environment you would typically use the Envoy service's address._
+
+Port-forward from your local machine to the Envoy service:
+```shell
+# If using static provisioning
+$ kubectl -n projectcontour port-forward service/envoy 8888:80
+
+# If using dynamic provisioning
+$ kubectl -n projectcontour port-forward service/envoy-contour 8888:80
+```
+
+In another terminal, make a request to the application via the forwarded port (note, `local.projectcontour.io` is a public DNS record resolving to 127.0.0.1 to make use of the forwarded port):
+```shell
+$ curl -i http://local.projectcontour.io:8888
+```
+You should receive a 200 response code along with the HTML body of the main `kuard` page.
+
+You can also open http://local.projectcontour.io:8888/ in a browser.
+
+### Further reading
+
+This guide only scratches the surface of the Gateway API's capabilities. See the [Gateway API website][1] for more information.
+
+
+[1]: https://gateway-api.sigs.k8s.io/
+[2]: https://kubernetes.io/
+[3]: https://projectcontour.io/resources/compatibility-matrix/
+[4]: https://kubernetes.io/docs/tasks/tools/install-kubectl/
+[5]: /docs/{{< param version >}}/config/gateway-api
+[6]: /docs/{{< param version >}}/config/gateway-api#enabling-gateway-api-in-contour
\ No newline at end of file
diff --git a/site/content/docs/1.28/guides/global-rate-limiting.md b/site/content/docs/1.28/guides/global-rate-limiting.md
new file mode 100644
index 00000000000..c2ff23edd27
--- /dev/null
+++ b/site/content/docs/1.28/guides/global-rate-limiting.md
@@ -0,0 +1,503 @@
+---
+title: Global Rate Limiting
+---
+
+Starting in version 1.13, Contour supports [Envoy global rate limiting][1].
+In global rate limiting, Envoy communicates with an external Rate Limit Service (RLS) over gRPC to make rate limit decisions for each request.
+Envoy is configured to produce 1+ descriptors for incoming requests, containing things like the client IP, header values, and more.
+Envoy sends descriptors to the RLS, and the RLS returns a rate limiting decision to Envoy based on the descriptors and the RLS's configured rate limits.
+
+In this guide, we'll walk through deploying an RLS, configuring it in Contour, and configuring an `HTTPProxy` to use it for rate limiting.
+
+**NOTE: you should not consider the RLS deployment in this guide to be production-ready.**
+The instructions and example YAML below are intended to be a demonstration of functionality only.
+Each user will have their own unique production requirements for their RLS deployment.
+
+## Prerequisites
+
+This guide assumes that you have:
+
+- A local KinD cluster created using [the Contour guide][2].
+- Contour installed and running in the cluster using the [quick start][3].
+
+## Deploy an RLS
+
+For this guide, we'll deploy the [Envoy rate limit service][4] as our RLS.
+Per the project's README:
+
+> The rate limit service is a Go/gRPC service designed to enable generic rate limit scenarios from different types of applications.
+> Applications request a rate limit decision based on a domain and a set of descriptors.
+> The service reads the configuration from disk via [runtime][10], composes a cache key, and talks to the Redis cache.
+> A decision is then returned to the caller.
+
+However, any service that implements the [RateLimitService gRPC interface][5] is supported by Contour/Envoy.
+
+Create a config map with [the ratelimit service configuration][6]:
+
+```yaml
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ name: ratelimit-config
+ namespace: projectcontour
+data:
+ ratelimit-config.yaml: |
+ domain: contour
+ descriptors:
+
+ # requests with a descriptor of ["generic_key": "foo"]
+ # are limited to one per minute.
+ - key: generic_key
+ value: foo
+ rate_limit:
+ unit: minute
+ requests_per_unit: 1
+
+ # each unique remote address (i.e. client IP)
+ # is limited to three requests per minute.
+ - key: remote_address
+ rate_limit:
+ unit: minute
+ requests_per_unit: 3
+```
+
+Create a deployment for the RLS that mounts the config map as a volume.
+**This configuration is for demonstration purposes only and is not a production-ready deployment.**
+```yaml
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ labels:
+ app: ratelimit
+ name: ratelimit
+ namespace: projectcontour
+spec:
+ replicas: 1
+ selector:
+ matchLabels:
+ app: ratelimit
+ template:
+ metadata:
+ labels:
+ app: ratelimit
+ spec:
+ containers:
+ - name: redis
+ image: redis:alpine
+ env:
+ - name: REDIS_SOCKET_TYPE
+ value: tcp
+ - name: REDIS_URL
+ value: redis:6379
+ - name: ratelimit
+ image: docker.io/envoyproxy/ratelimit:6f5de117
+ ports:
+ - containerPort: 8080
+ name: http
+ protocol: TCP
+ - containerPort: 8081
+ name: grpc
+ protocol: TCP
+ volumeMounts:
+ - name: ratelimit-config
+ mountPath: /data/ratelimit/config
+ readOnly: true
+ env:
+ - name: USE_STATSD
+ value: "false"
+ - name: LOG_LEVEL
+ value: debug
+ - name: REDIS_SOCKET_TYPE
+ value: tcp
+ - name: REDIS_URL
+ value: localhost:6379
+ - name: RUNTIME_ROOT
+ value: /data
+ - name: RUNTIME_SUBDIRECTORY
+ value: ratelimit
+ - name: RUNTIME_WATCH_ROOT
+ value: "false"
+ # need to set RUNTIME_IGNOREDOTFILES to true to avoid issues with
+ # how Kubernetes mounts configmaps into pods.
+ - name: RUNTIME_IGNOREDOTFILES
+ value: "true"
+ command: ["/bin/ratelimit"]
+ livenessProbe:
+ httpGet:
+ path: /healthcheck
+ port: 8080
+ initialDelaySeconds: 5
+ periodSeconds: 5
+ volumes:
+ - name: ratelimit-config
+ configMap:
+ name: ratelimit-config
+```
+
+Create a service:
+
+```yaml
+apiVersion: v1
+kind: Service
+metadata:
+ name: ratelimit
+ namespace: projectcontour
+spec:
+ ports:
+ - port: 8081
+ name: grpc
+ protocol: TCP
+ selector:
+ app: ratelimit
+ type: ClusterIP
+```
+
+Check the progress of the deployment:
+
+```bash
+$ kubectl -n projectcontour get pods -l app=ratelimit
+NAME READY STATUS RESTARTS AGE
+ratelimit-658f4b8f6b-2hnrf 2/2 Running 0 12s
+```
+
+Once the pod is `Running` with `2/2` containers ready, move onto the next step.
+
+## Configure the RLS with Contour
+
+Create a Contour extension service for the RLS:
+
+```yaml
+apiVersion: projectcontour.io/v1alpha1
+kind: ExtensionService
+metadata:
+ namespace: projectcontour
+ name: ratelimit
+spec:
+ protocol: h2c
+ # The service name and port correspond to
+ # the service we created in the previous
+ # step.
+ services:
+ - name: ratelimit
+ port: 8081
+ timeoutPolicy:
+ response: 100ms
+```
+
+Update the Contour configmap to have the following RLS configuration:
+
+```yaml
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ name: contour
+ namespace: projectcontour
+data:
+ contour.yaml: |
+ rateLimitService:
+ # extensionService is the /
+ # of the ExtensionService we created in the
+ # previous step.
+ extensionService: projectcontour/ratelimit
+ # domain corresponds to the domain in the
+ # projectcontour/ratelimit-config config map.
+ domain: contour
+ # failOpen is whether to allow requests through
+ # if there's an error connecting to the RLS.
+ failOpen: false
+```
+
+Restart Contour to pick up the new config map:
+
+```bash
+$ kubectl -n projectcontour rollout restart deploy/contour
+deployment.apps/contour restarted
+```
+
+## Deploy a sample app
+
+To demonstrate how to use global rate limiting in a `HTTPProxy` resource, we first need to deploy a simple echo application:
+
+```yaml
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ name: ingress-conformance-echo
+spec:
+ replicas: 1
+ selector:
+ matchLabels:
+ app.kubernetes.io/name: ingress-conformance-echo
+ template:
+ metadata:
+ labels:
+ app.kubernetes.io/name: ingress-conformance-echo
+ spec:
+ containers:
+ - name: conformance-echo
+ image: agervais/ingress-conformance-echo:latest
+ ports:
+ - name: http-api
+ containerPort: 3000
+ readinessProbe:
+ httpGet:
+ path: /health
+ port: 3000
+---
+apiVersion: v1
+kind: Service
+metadata:
+ name: ingress-conformance-echo
+spec:
+ ports:
+ - name: http
+ port: 80
+ targetPort: http-api
+ selector:
+ app.kubernetes.io/name: ingress-conformance-echo
+```
+
+This echo server will respond with a JSON object that reports information about the HTTP request it received, including the request headers.
+
+Once the application is running, we can expose it to Contour with a `HTTPProxy` resource:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: echo
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+ - conditions:
+ - prefix: /foo
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+```
+
+We can verify that the application is working by requesting any path:
+
+```bash
+$ curl -k http://local.projectcontour.io/test/$((RANDOM))
+{"TestId":"","Path":"/test/22808","Host":"local.projectcontour.io","Method":"GET","Proto":"HTTP/1.1","Headers":{"Accept":["*/*"],"Content-Length":["0"],"User-Agent":["curl/7.75.0"],"X-Envoy-Expected-Rq-Timeout-Ms":["15000"],"X-Envoy-Internal":["true"],"X-Forwarded-For":["172.18.0.1"],"X-Forwarded-Proto":["http"],"X-Request-Id":["8ecb85e1-271b-44b4-9cf0-4859cbaed7a7"],"X-Request-Start":["t=1612903866.309"]}}
+```
+
+## Add global rate limit policies
+
+Now that we have a working application exposed by a `HTTPProxy` resource, we can add global rate limiting to it.
+
+Edit the `HTTPProxy` that we created in the previous step to add rate limit policies to both routes:
+
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: echo
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+ rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - remoteAddress: {}
+ - conditions:
+ - prefix: /foo
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+ rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - remoteAddress: {}
+ - entries:
+ - genericKey:
+ value: foo
+```
+
+## Default Global rate limit policy
+
+Contour supports defining a default global rate limit policy in the `rateLimitService` configuration
+which is applied to all virtual hosts unless the host is opted-out by
+explicitly setting `disabled` to `true`. This is useful for a single-tenant
+setup use-case. This means you don't have to edit all HTTPProxy objects with the same rate limit policies, instead you can
+define the policies in the `rateLimitService` configuration like this:
+```yaml
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ name: contour
+ namespace: projectcontour
+data:
+ contour.yaml: |
+ rateLimitService:
+ extensionService: projectcontour/ratelimit
+ domain: contour
+ failOpen: false
+ defaultGlobalRateLimitPolicy:
+ descriptors:
+ - entries:
+ - requestHeader:
+ headerName: X-Custom-Header
+ descriptorKey: CustomHeader
+```
+
+Virtual host can opt out by setting `disabled` to `true`.
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: echo
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ rateLimitPolicy:
+ global:
+ disabled: true
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+```
+
+Also, the default global rate limit policy is not applied in case the virtual host defines its own global rate limit policy.
+```yaml
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: echo
+spec:
+ virtualhost:
+ fqdn: local.projectcontour.io
+ rateLimitPolicy:
+ global:
+ descriptors:
+ - entries:
+ - remoteAddress: {}
+ routes:
+ - conditions:
+ - prefix: /
+ services:
+ - name: ingress-conformance-echo
+ port: 80
+```
+
+## Make requests
+
+Before making requests to our `HTTPProxy`, let's quickly revisit the `ratelimit-config` config map.
+Here's what we defined:
+
+```yaml
+...
+descriptors:
+ # requests with a descriptor of ["generic_key": "foo"]
+ # are limited to one per minute.
+ - key: generic_key
+ value: foo
+ rate_limit:
+ unit: minute
+ requests_per_unit: 1
+
+ # each unique remote address (i.e. client IP)
+ # is limited to three total requests per minute.
+ - key: remote_address
+ rate_limit:
+ unit: minute
+ requests_per_unit: 3
+```
+
+The first entry says that requests with a descriptor of `["generic_key": "foo"]` should be limited to one per minute.
+The second entry says that each unique remote address (client IP) should be allowed three total requests per minute.
+All relevant rate limits are applied for each request, and requests that result in a `429 (Too Many Requests)` count against limits.
+
+So, we should be able to make:
+- a first request to `local.projectcontour.io/foo` that get a `200 (OK)` response
+- a second request to `local.projectcontour.io/foo` that gets a `429 (Too Many Requests)` response (due to the first rate limit)
+- a third request to `local.projectcontour.io/bar`that gets a `200 (OK)` response
+- a fourth request to `local.projectcontour.io/bar`that gets a `429 (Too Many Requests)` response (due to the second rate limit)
+
+Let's try it out (remember, you'll need to make all of these requests within 60 seconds since the rate limits are per minute):
+
+Request #1:
+```
+$ curl -I local.projectcontour.io/foo
+
+HTTP/1.1 200 OK
+content-type: application/json
+date: Mon, 08 Feb 2021 22:25:06 GMT
+content-length: 403
+x-envoy-upstream-service-time: 4
+vary: Accept-Encoding
+server: envoy
+```
+
+Request #2:
+
+```
+$ curl -I local.projectcontour.io/foo
+
+HTTP/1.1 429 Too Many Requests
+x-envoy-ratelimited: true
+date: Mon, 08 Feb 2021 22:59:10 GMT
+server: envoy
+transfer-encoding: chunked
+```
+
+Request #3:
+
+```
+$ curl -I local.projectcontour.io/bar
+
+HTTP/1.1 200 OK
+content-type: application/json
+date: Mon, 08 Feb 2021 22:59:54 GMT
+content-length: 404
+x-envoy-upstream-service-time: 2
+vary: Accept-Encoding
+server: envoy
+```
+
+Request #4:
+
+```
+$ curl -I local.projectcontour.io/bar
+
+HTTP/1.1 429 Too Many Requests
+x-envoy-ratelimited: true
+date: Mon, 08 Feb 2021 23:00:28 GMT
+server: envoy
+transfer-encoding: chunked
+```
+
+## Wrapping up
+
+For more information, see the [Contour rate limiting documentation][7] and the [API reference documentation][8].
+
+The YAML used in this guide is available [in the Contour repository][9].
+
+[1]: https://www.envoyproxy.io/docs/envoy/latest/intro/arch_overview/other_features/global_rate_limiting
+[2]: ../deploy-options/#kind
+[3]: https://projectcontour.io/getting-started/#option-1-quickstart
+[4]: https://github.com/envoyproxy/ratelimit
+[5]: https://www.envoyproxy.io/docs/envoy/latest/api-v3/service/ratelimit/v3/rls.proto
+[6]: https://github.com/envoyproxy/ratelimit#configuration
+[7]: ../config/rate-limiting/
+[8]: ../config/api/
+[9]: {{< param github_url>}}/tree/main/examples/ratelimit
+[10]: https://github.com/lyft/goruntime
diff --git a/site/content/docs/1.28/guides/grpc.md b/site/content/docs/1.28/guides/grpc.md
new file mode 100644
index 00000000000..12f0d9035fb
--- /dev/null
+++ b/site/content/docs/1.28/guides/grpc.md
@@ -0,0 +1,225 @@
+---
+title: Configuring ingress to gRPC services with Contour
+---
+
+## Example gRPC Service
+
+The below examples use the [gRPC server][1] used in Contour end to end tests.
+The server implements a service `yages.Echo` with two methods `Ping` and `Reverse`.
+It also implements the [gRPC health checking service][2] (see [here][3] for more details) and is bundled with the [gRPC health probe][4].
+
+An example base deployment and service for a gRPC server utilizing plaintext HTTP/2 are provided here:
+
+```yaml
+---
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ labels:
+ app.kubernetes.io/name: grpc-echo
+ name: grpc-echo
+spec:
+ replicas: 2
+ selector:
+ matchLabels:
+ app.kubernetes.io/name: grpc-echo
+ template:
+ metadata:
+ labels:
+ app.kubernetes.io/name: grpc-echo
+ spec:
+ containers:
+ - name: grpc-echo
+ image: ghcr.io/projectcontour/yages:v0.1.0
+ ports:
+ - name: grpc
+ containerPort: 9000
+ readinessProbe:
+ exec:
+ command: ["/grpc-health-probe", "-addr=localhost:9000"]
+---
+apiVersion: v1
+kind: Service
+metadata:
+ labels:
+ app.kubernetes.io/name: grpc-echo
+ name: grpc-echo
+spec:
+ selector:
+ app.kubernetes.io/name: grpc-echo
+ ports:
+ - port: 9000
+ protocol: TCP
+ targetPort: grpc
+```
+
+## HTTPProxy Configuration
+
+Configuring proxying to a gRPC service with HTTPProxy is as simple as specifying the protocol Envoy uses with the upstream application via the `spec.routes[].services[].protocol` field.
+For example, in the resource below, for proxying plaintext gRPC to the `yages` sample app, the protocol is set to `h2c` to denote HTTP/2 over cleartext.
+For TLS secured gRPC, the protocol used would be `h2`.
+
+Route path prefix matching can be used to match a specific gRPC message if required.
+
+```yaml
+---
+apiVersion: projectcontour.io/v1
+kind: HTTPProxy
+metadata:
+ name: my-grpc-service
+spec:
+ virtualhost:
+ fqdn: my-grpc-service.foo.com
+ routes:
+ - conditions:
+ - prefix: /yages.Echo/Ping # Matches a specific gRPC method.
+ services:
+ - name: grpc-echo
+ port: 9000
+ protocol: h2c
+ - conditions:
+ - prefix: / # Matches everything else.
+ services:
+ - name: grpc-echo
+ port: 9000
+ protocol: h2c
+```
+
+Using the sample deployment above along with this HTTPProxy example, you can test calling this plaintext gRPC server with the following [grpcurl][5] command:
+
+```
+grpcurl -plaintext -authority=my-grpc-service.foo.com yages.Echo/Ping
+```
+
+If implementing a streaming RPC, it is likely you will need to adjust per-route timeouts to ensure streams are kept alive for the appropriate durations needed.
+Relevant timeout fields to adjust include the HTTPProxy `spec.routes[].timeoutPolicy.response` field which defaults to 15s and should be increased as well as the global timeout policy configurations in the Contour configuration file `timeouts.request-timeout` and `timeouts.max-connection-duration`.
+
+## Ingress v1 Configuration
+
+To configure routing for gRPC requests with Ingress v1, you must add an annotation on the upstream Service resource as below.
+
+```yaml
+---
+apiVersion: v1
+kind: Service
+metadata:
+ labels:
+ app.kubernetes.io/name: grpc-echo
+ annotations:
+ projectcontour.io/upstream-protocol.h2c: "9000"
+ name: grpc-echo
+spec:
+ selector:
+ app.kubernetes.io/name: grpc-echo
+ ports:
+ - port: 9000
+ protocol: TCP
+ targetPort: grpc
+```
+
+The annotation key must follow the form `projectcontour.io/upstream-protocol.{protocol}` where `{protocol}` is `h2c` for plaintext gRPC or `h2` for TLS encrypted gRPC to the upstream application.
+The annotation value contains a comma-separated list of port names and/or numbers that must match with the ones defined in the Service definition.
+
+Using the Service above with the Ingress resource below should achieve the same configuration as with an HTTPProxy.
+
+```yaml
+---
+apiVersion: networking.k8s.io/v1
+kind: Ingress
+metadata:
+ name: my-grpc-service
+spec:
+ rules:
+ - host: my-grpc-service.foo.com
+ http:
+ paths:
+ - path: /
+ backend:
+ service:
+ name: grpc-echo
+ port:
+ number: 9000
+ pathType: Prefix
+```
+
+## Gateway API Configuration
+
+Gateway API now supports a specific resource [GRPCRoute][6] for routing gRPC requests.
+
+Configuring GRPCRoute for routing gRPC requests needs to specify parentRefs, hostnames, and routing rules with specific backendRefs. In the below example, route path matching is conducted via method matching rule for declared services and their methods.
+
+```yaml
+apiVersion: gateway.networking.k8s.io/v1alpha2
+kind: GRPCRoute
+metadata:
+ name: yages
+spec:
+ parentRefs:
+ - namespace: projectcontour
+ name: contour
+ hostnames:
+ - my-grpc-service.foo.com
+ rules:
+ - matches:
+ - method:
+ service: yages.Echo
+ method: Ping
+ - method:
+ service: grpc.reflection.v1alpha.ServerReflection
+ method: ServerReflectionInfo
+ backendRefs:
+ - name: grpc-echo
+ port: 9000
+```
+Using the sample deployment above along with this GRPCRoute example, you can test calling this plaintext gRPC server with the same grpcurl command:
+
+```yaml
+grpcurl -plaintext -authority=my-grpc-service.foo.com yages.Echo/Ping
+```
+Note that the second matching method for service of ServerReflection is required by grpcurl command.
+
+When using GRPCRoute, user should annotate their Service similarly to when using Ingress Configuration, to indicate the protocol to use when connecting to the backend Service, i.e. h2c for HTTP plaintext and h2 for TLS encrypted HTTPS. If it's not specified, Contour will infer the protocol based on the Gateway Listener protocol, h2c for HTTP and h2 for HTTPS.
+
+
+
+
+## gRPC-Web
+
+Contour configures Envoy to automatically convert [gRPC-Web][7] HTTP/1 requests to gRPC over HTTP/2 RPC calls to an upstream service.
+This is a convenience addition to make usage of gRPC web application client libraries and the like easier.
+
+Note that you still must provide configuration of the upstream protocol to have gRPC-Web requests converted to gRPC to the upstream app.
+If your upstream application does not in fact support gRPC, you may get a protocol error.
+In that case, please see [this issue][8].
+
+For example, with the example deployment and routing configuration provided above, an example HTTP/1.1 request and response via `curl` looks like:
+
+```
+curl \
+ -s -v \
+ /yages.Echo/Ping \
+ -XPOST \
+ -H 'Host: my-grpc-service.foo.com' \
+ -H 'Content-Type: application/grpc-web-text' \
+ -H 'Accept: application/grpc-web-text' \
+ -d'AAAAAAA='
+```
+
+This `curl` command sends and receives gRPC messages as base 64 encoded text over HTTP/1.1.
+Piping the output to `base64 -d | od -c` we can see the raw text gRPC response:
+
+```
+0000000 \0 \0 \0 \0 006 \n 004 p o n g 200 \0 \0 \0 036
+0000020 g r p c - s t a t u s : 0 \r \n g
+0000040 r p c - m e s s a g e : \r \n
+0000056
+```
+
+[1]: https://github.com/projectcontour/yages
+[2]: https://pkg.go.dev/google.golang.org/grpc/health/grpc_health_v1
+[3]: https://github.com/grpc/grpc/blob/master/doc/health-checking.md
+[4]: https://github.com/grpc-ecosystem/grpc-health-probe
+[5]: https://github.com/fullstorydev/grpcurl
+[6]: https://gateway-api.sigs.k8s.io/references/spec/#gateway.networking.k8s.io/v1alpha2.GRPCRoute
+[7]: https://github.com/grpc/grpc-web
+[8]: https://github.com/projectcontour/contour/issues/4290
diff --git a/site/content/docs/1.28/guides/health-checking.md b/site/content/docs/1.28/guides/health-checking.md
new file mode 100644
index 00000000000..8e7bcdb5bb5
--- /dev/null
+++ b/site/content/docs/1.28/guides/health-checking.md
@@ -0,0 +1,11 @@
+---
+title: Health Checking
+---
+
+Contour exposes two health endpoints `/health` and `/healthz`. By default these paths are serviced by `0.0.0.0:8000` and are configurable using the `--health-address` and `--health-port` flags.
+
+e.g. `--health-port 9999` would create a health listener of `0.0.0.0:9999`
+
+**Note:** the `Service` deployment manifest when installing Contour must be updated to represent the same port as the above configured flags.
+
+The health endpoints perform a connection to the Kubernetes cluster's API.
diff --git a/site/content/docs/1.28/guides/kind.md b/site/content/docs/1.28/guides/kind.md
new file mode 100644
index 00000000000..dcc374b70af
--- /dev/null
+++ b/site/content/docs/1.28/guides/kind.md
@@ -0,0 +1,63 @@
+---
+title: Creating a Contour-compatible kind cluster
+---
+
+This guide walks through creating a kind (Kubernetes in Docker) cluster on your local machine that can be used for developing and testing Contour.
+
+# Prerequisites
+
+Download & install Docker and kind:
+
+- Docker [installation information](https://docs.docker.com/desktop/#download-and-install)
+- kind [download and install instructions](https://kind.sigs.k8s.io/docs/user/quick-start/)
+
+# Kind configuration file
+
+Create a kind configuration file locally.
+This file will instruct kind to create a cluster with one control plane node and one worker node, and to map ports 80 and 443 on your local machine to ports 80 and 443 on the worker node container.
+This will allow us to easily get traffic to Contour/Envoy running inside the kind cluster from our local machine.
+
+Copy the text below into the local yaml file `kind-config.yaml`:
+
+```yaml
+kind: Cluster
+apiVersion: kind.x-k8s.io/v1alpha4
+nodes:
+- role: control-plane
+- role: worker
+ extraPortMappings:
+ - containerPort: 80
+ hostPort: 80
+ listenAddress: "0.0.0.0"
+ - containerPort: 443
+ hostPort: 443
+ listenAddress: "0.0.0.0"
+```
+
+# Kubernetes cluster using kind
+
+Create a kind cluster using the config file from above:
+
+```yaml
+$ kind create cluster --config kind-config.yaml
+```
+
+Verify the nodes are ready by running:
+
+```yaml
+$ kubectl get nodes
+```
+
+You should see 2 nodes listed with status **Ready**:
+- kind-control-plane
+- kind-worker
+
+Congratulations, you have created your cluster environment. You're ready to install Contour.
+
+_Note:_ When you are done with the cluster, you can delete it by running:
+```yaml
+$ kind delete cluster
+```
+
+# Next Steps
+See https://projectcontour.io/getting-started/ for how to install Contour into your kind cluster.
diff --git a/site/content/docs/1.28/guides/metrics/table.md b/site/content/docs/1.28/guides/metrics/table.md
new file mode 100644
index 00000000000..89405d815c6
--- /dev/null
+++ b/site/content/docs/1.28/guides/metrics/table.md
@@ -0,0 +1,20 @@
+| Name | Type | Labels | Description |
+| ---- | ---- | ------ | ----------- |
+| contour_build_info | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | branch, revision, version | Build information for Contour. Labels include the branch and git SHA that Contour was built from, and the Contour version. |
+| contour_cachehandler_onupdate_duration_seconds | [SUMMARY](https://prometheus.io/docs/concepts/metric_types/#summary) | | Histogram for the runtime of xDS cache regeneration. |
+| contour_dag_cache_object | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | kind | Total number of items that are currently in the DAG cache. |
+| contour_dagrebuild_seconds | [SUMMARY](https://prometheus.io/docs/concepts/metric_types/#summary) | | Duration in seconds of DAG rebuilds |
+| contour_dagrebuild_timestamp | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | | Timestamp of the last DAG rebuild. |
+| contour_dagrebuild_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | | Total number of times DAG has been rebuilt since startup |
+| contour_eventhandler_operation_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind, op | Total number of Kubernetes object changes Contour has received by operation and object kind. |
+| contour_httpproxy | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | namespace | Total number of HTTPProxies that exist regardless of status. |
+| contour_httpproxy_invalid | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | namespace, vhost | Total number of invalid HTTPProxies. |
+| contour_httpproxy_orphaned | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | namespace | Total number of orphaned HTTPProxies which have no root delegating to them. |
+| contour_httpproxy_root | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | namespace | Total number of root HTTPProxies. Note there will only be a single root HTTPProxy per vhost. |
+| contour_httpproxy_valid | [GAUGE](https://prometheus.io/docs/concepts/metric_types/#gauge) | namespace, vhost | Total number of valid HTTPProxies. |
+| contour_status_update_conflict_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind | Number of status update conflicts encountered by object kind. |
+| contour_status_update_duration_seconds | [SUMMARY](https://prometheus.io/docs/concepts/metric_types/#summary) | error, kind | How long a status update takes to finish. |
+| contour_status_update_failed_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind | Number of status updates that failed by object kind. |
+| contour_status_update_noop_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind | Number of status updates that are no-ops by object kind. This is a subset of successful status updates. |
+| contour_status_update_success_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind | Number of status updates that succeeded by object kind. |
+| contour_status_update_total | [COUNTER](https://prometheus.io/docs/concepts/metric_types/#counter) | kind | Total number of status updates by object kind. |
diff --git a/site/content/docs/1.28/guides/prometheus.md b/site/content/docs/1.28/guides/prometheus.md
new file mode 100644
index 00000000000..f0b7364c340
--- /dev/null
+++ b/site/content/docs/1.28/guides/prometheus.md
@@ -0,0 +1,94 @@
+---
+title: Collecting Metrics with Prometheus
+---
+
+
+
+Contour and Envoy expose metrics that can be scraped with Prometheus. By
+default, annotations to gather them are in all the `deployment` yamls and they
+should work out of the box with most configurations.
+
+## Envoy Metrics
+
+Envoy typically [exposes metrics](https://www.envoyproxy.io/docs/envoy/v1.15.0/configuration/http/http_conn_man/stats#config-http-conn-man-stats) through an endpoint on its admin interface. To
+avoid exposing the entire admin interface to Prometheus (and other workloads in
+the cluster), Contour configures a static listener that sends traffic to the
+stats endpoint and nowhere else.
+
+Envoy supports Prometheus-compatible `/stats/prometheus` endpoint for metrics on
+port `8002`.
+
+## Contour Metrics
+
+Contour exposes a Prometheus-compatible `/metrics` endpoint that defaults to listening on port 8000. This can be configured by using the `--http-address` and `--http-port` flags for the `serve` command.
+
+**Note:** the `Service` deployment manifest when installing Contour must be updated to represent the same port as the configured flag.
+
+**The metrics endpoint exposes the following metrics:**
+
+{{% metrics-table %}}
+
+## Sample Deployment
+
+In the `/examples` directory there are example deployment files that can be used to spin up an example environment.
+All deployments there are configured with annotations for prometheus to scrape by default, so it should be possible to utilize any of them with the following quickstart example instructions.
+
+### Deploy Prometheus
+
+A sample deployment of Prometheus and Alertmanager is provided that uses temporary storage. This deployment can be used for testing and development, but might not be suitable for all environments.
+
+#### Stateful Deployment
+
+ A stateful deployment of Prometheus should use persistent storage with [Persistent Volumes and Persistent Volume Claims][1] to maintain a correlation between a data volume and the Prometheus Pod.
+ Persistent volumes can be static or dynamic and depends on the backend storage implementation utilized in environment in which the cluster is deployed. For more information, see the [Kubernetes documentation on types of persistent volumes][2].
+
+#### Quick start
+
+```sh
+# Deploy
+$ kubectl apply -f examples/prometheus
+```
+
+#### Access the Prometheus web UI
+
+```sh
+$ kubectl -n projectcontour-monitoring port-forward $(kubectl -n projectcontour-monitoring get pods -l app=prometheus -l component=server -o jsonpath='{.items[0].metadata.name}') 9090:9090
+```
+
+then go to `http://localhost:9090` in your browser.
+
+#### Access the Alertmanager web UI
+
+```sh
+$ kubectl -n projectcontour-monitoring port-forward $(kubectl -n projectcontour-monitoring get pods -l app=prometheus -l component=alertmanager -o jsonpath='{.items[0].metadata.name}') 9093:9093
+```
+
+then go to `http://localhost:9093` in your browser.
+
+### Deploy Grafana
+
+A sample deployment of Grafana is provided that uses temporary storage.
+
+#### Quick start
+
+```sh
+# Deploy
+$ kubectl apply -f examples/grafana/
+
+# Create secret with grafana credentials
+$ kubectl create secret generic grafana -n projectcontour-monitoring \
+ --from-literal=grafana-admin-password=admin \
+ --from-literal=grafana-admin-user=admin
+```
+
+#### Access the Grafana UI
+
+```sh
+$ kubectl port-forward $(kubectl get pods -l app=grafana -n projectcontour-monitoring -o jsonpath='{.items[0].metadata.name}') 3000 -n projectcontour-monitoring
+```
+
+then go to `http://localhost:3000` in your browser.
+The username and password are from when you defined the Grafana secret in the previous step.
+
+[1]: https://kubernetes.io/docs/concepts/storage/persistent-volumes/
+[2]: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#types-of-persistent-volumes
\ No newline at end of file
diff --git a/site/content/docs/1.28/guides/proxy-proto.md b/site/content/docs/1.28/guides/proxy-proto.md
new file mode 100644
index 00000000000..7753d8c5776
--- /dev/null
+++ b/site/content/docs/1.28/guides/proxy-proto.md
@@ -0,0 +1,53 @@
+---
+title: How to Configure PROXY v1/v2 Support
+---
+
+If you deploy Contour as a Deployment or Daemonset, you will likely use a `type: LoadBalancer` Service to request an [external load balancer][1] from your hosting provider.
+If you use the Elastic Load Balancer (ELB) service from Amazon's EC2, you need to perform a couple of additional steps to enable the [PROXY][0] protocol. Here's why:
+
+External load balancers typically operate in one of two modes: a layer 7 HTTP proxy, or a layer 4 TCP proxy.
+The former cannot be used to load balance TLS traffic, because your cloud provider attempts HTTP negotiation on port 443.
+So the latter must be used when Contour handles HTTP and HTTPS traffic.
+
+However this leads to a situation where the remote IP address of the client is reported as the inside address of your cloud provider's load balancer.
+To rectify the situation, you can add annotations to your service and flags to your Contour Deployment or DaemonSet to enable the [PROXY][0] protocol which forwards the original client IP details to Envoy.
+
+## Enable PROXY protocol on your service in GKE
+
+In GKE clusters a `type: LoadBalancer` Service is provisioned as a Network Load Balancer and will forward traffic to your Envoy instances with their client addresses intact.
+Your services should see the addresses in the `X-Forwarded-For` or `X-Envoy-External-Address` headers without having to enable a PROXY protocol.
+
+## Enable PROXY protocol on your service in AWS
+
+To instruct EC2 to place the ELB into `tcp`+`PROXY` mode, add the following annotations to the `contour` Service:
+
+```
+apiVersion: v1
+kind: Service
+metadata:
+ annotations:
+ service.beta.kubernetes.io/aws-load-balancer-backend-protocol: tcp
+ service.beta.kubernetes.io/aws-load-balancer-proxy-protocol: '*'
+ name: contour
+ namespace: projectcontour
+spec:
+ type: LoadBalancer
+...
+```
+
+## Enable PROXY protocol support for all Envoy listening ports
+
+```
+...
+spec:
+ containers:
+ - image: ghcr.io/projectcontour/contour:
+ imagePullPolicy: Always
+ name: contour
+ command: ["contour"]
+ args: ["serve", "--incluster", "--use-proxy-protocol"]
+...
+```
+
+[0]: http://www.haproxy.org/download/1.8/doc/proxy-protocol.txt
+[1]: https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer
\ No newline at end of file
diff --git a/site/content/docs/1.28/guides/resource-limits.md b/site/content/docs/1.28/guides/resource-limits.md
new file mode 100644
index 00000000000..a531221a88f
--- /dev/null
+++ b/site/content/docs/1.28/guides/resource-limits.md
@@ -0,0 +1,161 @@
+---
+title: Contour / Envoy Resource Limits
+---
+
+## Performance Testing Contour / Envoy
+
+- Cluster Specs
+ - Kubernetes
+ - Version: v1.12.6
+ - Nodes:
+ - 5 Worker Nodes
+ - 2 CPUs Per Node
+ - 8 GB RAM Per Node
+ - 10 GB Network
+ - Contour
+ - Single Instance
+ - 4 Instances of Envoy running in a Daemonset
+ - Each instance of Envoy is running with HostNetwork
+ - Cluster Network Bandwidth
+
+Having a good understanding of the available bandwidth is key when it comes to analyzing performance. It will give you a sense of how many requests per second you can expect to push through the network you are working with.
+
+Use iperf3 to figure out the bandwidth available between two of the kubernetes nodes. The following will deploy an iperf3 server on one node, and an iperf3 client on another node:
+
+```bash
+[ ID] Interval Transfer Bandwidth Retr
+[ 4] 0.00-60.00 sec 34.7 GBytes 4.96 Gbits/sec 479 sender
+[ 4] 0.00-60.00 sec 34.7 GBytes 4.96 Gbits/sec receiver
+```
+
+## Memory / CPU usage
+
+Verify the Memory & CPU usage with varying numbers of services, IngressRoute resources, and traffic load into the cluster.
+
+
+
+ Test Criteria |
+ Contour |
+ Envoy |
+
+
+ #Svc |
+ #Ing |
+ RPS |
+ CC |
+ Memory (MB) |
+ CPU% / Core |
+ Memory (MB) |
+ CPU% / Core |
+
+
+ 0 |
+ 0 |
+ 0 |
+ 0 |
+ 10 |
+ 0 |
+ 15 |
+ 0 |
+
+
+ 5k |
+ 0 |
+ 0 |
+ 0 |
+ 46 |
+ 2% |
+ 15 |
+ 0% |
+
+
+ 10k |
+ 0 |
+ 0 |
+ 0 |
+ 77 |
+ 3% |
+ 205 |
+ 2% |
+
+
+ 0 |
+ 5k |
+ 0 |
+ 0 |
+ 36 |
+ 1% |
+ 230 |
+ 2% |
+
+
+ 0 |
+ 10k |
+ 0 |
+ 0 |
+ 63 |
+ 1% |
+ 10 |
+ 1% |
+
+
+ 5k |
+ 5k |
+ 0 |
+ 0 |
+ 244 |
+ 1% |
+ 221 |
+ 1% |
+
+
+ 10k |
+ 10k |
+ 0 |
+ 0 |
+ 2600 |
+ 6% |
+ 430 |
+ 4% |
+
+
+ 0 |
+ 0 |
+ 30k |
+ 600 |
+ 8 |
+ 1% |
+ 17 |
+ 3% |
+
+
+ 0 |
+ 0 |
+ 100k |
+ 10k |
+ 10 |
+ 1% |
+ 118 |
+ 14% |
+
+
+ 0 |
+ 0 |
+ 200k |
+ 20k |
+ 9 |
+ 1% |
+ 191 |
+ 31% |
+
+
+ 0 |
+ 0 |
+ 300k |
+ 30k |
+ 10 |
+ 1% |
+ 225 |
+ 40% |
+
+
diff --git a/site/content/docs/1.28/img/archoverview.png b/site/content/docs/1.28/img/archoverview.png
new file mode 100644
index 0000000000000000000000000000000000000000..f79bbfe1b4b3a7f0de886d7ce059722308be3ab4
GIT binary patch
literal 78807
zcmeFZWmuJ4*ES3Y5=xgKCDN&+G%6w8i!N!9kd!W^1?g@niG_5BY`R-or5jm<#A3a3
zvG3=(_x=5TzwbD{p`<*kcd1Hq4c9-w^E4D
z)QoLy1bX>2TWZdI(sOY`o<^HY9x3nN
z#LpeH!cu1uh!&2<^NbbG>^norE}KzAMp$$KC}{uw<1Kb^!@-6nI)Nk#>c9VZr1KT)
zKmH}~83q>QEt<2)KZat_Nmjnb{eOtVAt?KWzUKaKQQ)TllWxlY5Rd+ggifa$yKn#B
z^FaR88NvS_bHn(Fwvd0!&Dm>FOW<$7po+nSU-6i;kcO
zrISrr(n0n=Mq@$B{;_Z{TRLUbAgzpm#pwSSO;DzD=RamEnSq90`V6&^`aed~>3qHW
zU&Wz?6Mi>~`TdapWSTj=I5zJDlDDov*xK;iYUOJATajPX0i$5MwKk#9*T}5oS?;o|
z)O2G$U8%e7x&;NxWzdpKZSPrJYPXXf$YdI$n3kq)5LUR*Q~G;$4_Ou3l$%0$YeGN16!`&_FAiX
zTPq;!sika4$~TQ$lpD)-}2r6)g1%fDU12}7=MYk
z?3-+8w_c^|Pu|fUef)DI#`duR8pyHeKDh_}yfA~MC7hNKT}UsyuIZZ{u%*;ROkBrg
zWaAFfp4$~nz04%7kQO&o*Kggg49bJf+tp`!Bh-xbebv3~51LlKXI7!j@DO?ywbgl9
zYnyZ(NA|EQY_clQlz6{A=pCQkB#V}K`K@W?hd3^8w)a6}PsP0a*{DaqM2ml&?e%l1
zRv@a@${UVLD|GXzvS5=n)^69f6p^u))?_PDZ?YIeZ@b!4cr5$HD
zMYVCR@LheiR)?Frq{OSFXjNguw-G}cKdcWsZ{6ecUg=?UJ+vh#4@1vf^QHR?_pMG+rg&2;sc
z*iZ3+dfwp^#|Md9qJ%<*QXyfl41-75T-}*7ObQZOXvyI9XLl_=Oh+_*m_EPH&LXUo
zaIk5RUn5*!Ff3a;0h>iM8L1nKI6BY=CBh2T!!Ioi?WZ-$gR~S_bQW?1J^g<#(oTpy
z;uJx}w=F2-Dk))egl2)sJ*@2$GL5~HGy`c*IY@K4*E38~ATWY+B78WEq2iZa=9Pi6
znbhdqld7%5r)JlvE5){BR~p<;=`Xr3aLBa%V%*CT{;je0cMZR*<~Rfr@54kteOf!c
znpuYh`CaVHjfTbWGJv|&ym*2M4a-%Zj@y6nuvWc3^`T~bt^QJ;)v~*TfMc;{Jcni;
zrgcHIp1KCy7*<_OT?%omX_~ZgFda9{TPI=UDXFNNZxPx*Cf$urOb*g=cs>r__$sdu
z8rt6KWVCem0WlMLh;P*|7uc>s-I6)G=TT=NSEr9NYUcnou@Hy>2yTDnO-9;yORT@C-DttM6zDOAAFz3BmTWaio&`7GByL}(>Q!?B4j*=Z7
z*6vYOl7;V5gTqq$ithDdy|TS0#wtwugn#2JWBD2nwnF*Hjz+b`SxjlW9`S?6ueH+c
zYG-;eYfj+{+}d-^mM>+g1vL2u>{q_C@78uSx#{Z-P_pWs;uIfMJ5G-nzILz?Q5%kV
zkAWi{T2El%OELB{dHGAGe)~-m4?FazJGjdf`^{8JbWd#+t(a4OGlD(
zJ2+L?-$czYVTMB_%Iy;_#piZ%mU}dN%E%K*X&%;1)e$9qVZ2vL=26oTN)k;=hBf%;b^p9mgU$HF
z2iTBR;7Q6;vnt{A(msP2ib<836UuVwLx+;1hY&g|Q*ROLuTeHi*&&-(6(jT7>~<1r
zrqp#+S$dbhCKWQp_eaUu;LL9QFs{245`
zx7kW)@i`=h%HJuT6py;&qt4!ppBtAVvnep`yx&6m{s0YoGemhO{PIKYx
zmpip}XXANEI?Hd2RK>25XI&>N!A%%zOJP{9v;Mo1Txhce#dcya?WXg
zn66jS<&n4)cCpuvK}$Pb9AgL#!#Sjy2~1An^68244PJ5CfF*sg8GcY;o`ScFNL|sE
zu!7Rqu#y&+mSiQ*NGNVop`uT4Nmj~Vk*K1erE-4KSJiw>kC6l+UN+L1-rg*vcPyUq
z?_ui}fp`^!@i@{CuZDAS!|Vc|p4N`G*!n%H&*^Kyj=pRRgV-k2orH46@YbdsUC!HH
zjs9G`ep?@*USt<5EpOfqA!j|+@l~4sI_`LKd>qL&E6=rLa>{b5Guht6_6uTuS$>F}B`sI_hH7T!(Kzkp=rFgibi$$q
z%_QHUe*PhUjH{*9>3kP9#jeM-z=;O^?9?>G_`7=PMT=@_ozz;%v=DBubo=iElcE>v
zq*3j%k>ZWR&vrC5y1s9X{@C;DAWRn!sL-!`ZY+P`@SS0Is3YcMx|@LZFig}TF6_&j
zrDP)tGPed;gSko%1O3L=#-J_tM0KmFY0bIjrV+DL`<@47(DI5W?K4Z_z-L9A6C%s>h7*4b7WjOt(`k`Ht;4E
z3t)C^CFYf_)Ox%d;>yRTO;{THQ_BK>;XAx|M<2SXH{o(kxrJDbmtzTj7
z5V?W`+dO5Slt{Cb70?J`>3VUwL_G=K_xa*Yf-O#aF2|f)q_DmeKM3K5*6u}l!CDKw
z9Qw7*_!kL#^)$$JSNe(vMW
zQ_7<$;L}GjX6Kxg>QB;JBZ%$5R*=LZsp9~qAs7JmJhwMRkND2w_%0q>xj9*
z(_tHeQ{$mzH9mz_g>WIU0_eD2xvG3Nm)=JRea!4Mgmx8uaxsXW7n4|ViQc(%W}c{E
zq+VjPXw7^#REg%`O~o-z?|yl0Rd7vMlXSSBRwUIMJUb^mC7ZL4zKX+2UwWlN$yteO
zYE!i@yyW(aIfv8TC+n?GWks-9Jk`S&OaAiO-%qnzjw~3>x?8$&x#oa9_@ZDk*4!f~n-Iq!36QkB_e?}o@1%1i
zU$zS;5>H*vctKBwvlLn`o-5Si>d9sj8dKLzUH0v=?=pE)+-9e5m{yxkemX_)!+2*t
zbM{v~+Om_v&Tot-U>t7DumB>cdEX2&3CVP`G
zb7o4RtbcZ0pK-E#P0}cYl7^&&-M8l|y(CMPFe?7}m_F8m^ODlk
ztF%@o;b$%MN#fp=WqXxs`o5Cd{?LP9G5-qZi~TDHo7ETF_MX9xu|VBQHBXW;&UGYnRo|Cc`O4zA*2d
z@;mQYzSea+v&rLZUvZI*7J0#O`X0Lx)ndKP5H@tY!OwH!6T2Z%iUlz&$MK5}Kmg)#
zlWisw7Ep?jVL}Qoo~sOc+nYKXQ^P7slr)3M-ddHpE5&~CxvkKvt9LDJrMUa;yAMVs
zev7Uy%qM*QWXyJrbG0~_Xk8h-A-}L8dfP6^Q$wx&6ju&cv{fz|&rA+wre-1f8)7EQ
zexZ1?ft7Hb?1|&q%8AjP%G2?QeN3Olr7p(3$3GKA^Q8+pOUm0CTBUi)a<_=
zT>HP-!i19h*x7Vyy%L7}%yy{WLfvwO2*GxiWQ2U&&GJ4d+N#Uw!XDz1F|7#SC7e4Br8?&P2>}nlPHos*o)k0$KxD%AU-mdQLlpi$SC4(5xOw+RMW0qP9D(E%~
zi@b4vW`oU1e{A4uUpseEZOj?-hD$J}m#d_lX0C3uPdE%G)7|^Ssvoo75(OL|e;Qj8nMm`vtQms
z7fRIUSSN*L*z5j`b!_KzRP&x)Cne*Y)Wf&^&gVVmts-WD7;J+@;6MatXCW#PH>t49
zmm>!W#AQRj|LpHauaiZ$zpBYvUiP@ZPD%n_5M`UOmCYTFp1#cNiY9NKasCg3_C*3`zL-T$Tderz}XX?*w7
zUU^Ir$gqR3o#Y9*2tKuL4b-APuWqFvDP7`5O6}4Q*02;ezat>>y&NC<&Q5KbnXo!(
zG)DTc0$5pw+oX|GWLbRDQte{U@8qy8cBzxZ@fdLd#HxIquj;+dFQr0ARg-W#i
znz@|ck*L8j{IHrOm~HP&bGWzDa9P}2q9>+jbv}9gN5}QZq1lWd1XWdDxO-dPHNU7eVuGeGdncM)qtHt?g4O)i
zPwJISO|w6`wQCJO&%H27+_-QMGAI}}C837@oY1Yb-Q5uV_Fdq|8fldOtH>L-1!}IM
z(bE{TyY771llq9=)gBhLQ5o8I<3^Qwv|BTXe5JK>!@H5@{xCzCBAW0q&=Ang*UH421
z#X?(mLrBRCztk91NK?0duG{&I`y!OMOhB|;374Jaazfl7Z=JBbB1p@~Z``pqu2~;1
z^_lP&?#Y>6rY9j?RVx$8rN&Ps>>ie-Nzju18AC5k+}Z3L&wgsB@4@VA-!V2GJq?n7
zmH8Qxml8{q8k#xVXVF?5(IZyFdH6k*_W9zoNn0kqmLj3hPq^$39)Iq;A0
z;gYY6*XtsP2eBZzh1h*<*U2#HUUB)Kik+k$MxIIouj9!b-PQ`8W|R%7<_r$4*DsP;
zpIqL1vqV#tv6`OQJDL|b>(BMqfS?GAnq$uX>WtE}FyW5p>)=bHY1DSgI0jXf^BM0p
zsM4NFbGX@@5bmhOV9xhCE{JH9RuAVA>*~(69yq&ehmevOLZxoL&y6I(=IXb4@$-8L
zecwsD{uy9@;88*57ToCWru>rDMbS&U({`)xw}@Vn-^wB}+>@(_6Sm>6q}+8j61%%k
zU4JmOaO|t^|Hf}s-x7Dw_(`|A?ber!5V;M%t0(x>e)XOlyDE8abT!)b7IH>!>jHWh
z&!#Oa=pCEZ8#{$?r8j8roKf;x#@E_aS?SZCw;Afg52d0?EGh)CAhGes38K2j8or--
zxxb#wEqCIcIV7H>+qX$J3$eI5_6```qrY<}u#kD#%jTvksVTi(oX$3M(DQm~j{)~!
z>0Q`c-t2r^YlHKU)&)wp<7}9aLb=;^+y|H8%}jxJ($q#@t@A_H^A(>&IH-gh7=VoS
z&_sBb|FT9N_W7-m>A3fXQ~Z|fb;e8mA>R=RytR*%^_dRpv~zZvuBo+n!B>@w^H$8s|
zwd5^if8}A$npSX*p_B!0)mBK1gJ%l9Q^vAC_by{n#JYcF{sGQdiY!Hxp5ZlDwm;L}y%cbKq0c!k
z&|AK+N}?tddl)itTyzP%b9#Cc4usRu2SG2d*QPM&P`Yx-KX%D-a~7mkN{l`ndq5vq
z+ZSdj(W=g2=g<6pV}o4!l(ANif#BlcNkOz*bjP7}MH)T_R=XA6>&gYPT1hF|vFrAN
z3Q#P2N8d9=q$+>~8P8_r-8Hb|pB5UfJpC^05hd)t?5f3_S8q=dEH=A@3yBRNrp}x9
zbgjG)DWOb~UQB3lprNqW)rtLNdsFYC=_~1Kw9DOU)P7%|x_(&{e`7GGx8}O-4R@Bh{lGzm1zuMsDP)Xd3pqw7^sSGa3krSMAd5yMTEnfaK6sFn(Hr0WGDW*%
z_r+m<&}>HkKRT9bj1bb0J?OC>nLs|7s8kJLOUgoP;FN|nl8WV7ARNg5+!LV;LX`EMwyggxceJm5zXOzi*6jvEn
zoqu_$8Q%2dvf(s(V_@oyh~7?IlG!)?8yT$_3ydoN!@c`G&zP=$e=7_LdA>22`Iw6n
zv6Un3*Z#TgGP>_DFzD@UL@_KK!KAGLbW>5cDSEf}(TC7C)MKcyg2h;FhPrlgrCl3K
znq9liGh^f422lU1=7=>LiZ(d$a-#Vid82D&dNsSy{CvLYO=~)J7>%8}Qm2RQhQH}EWt*a7r8!xys>Evl2<((0XM}C`cnpOXKu_3-WHF9vB`i&oKy9#PhqQqvR
z+M^$8fjo9w#pLx^?@xa)Dfrf&bgLQ5_@Bqk-nD2mE?L>>2&gBEzA4;)=8i1qCK)B$
zw{4cJfAlJqEgDKUS}OKxZ??{YhZ$gUCyM+U<8*}<4p)D>+^r`YxoT<+X2}D!iZcGP
zdwXe1rQ3#^#;}FD)&;-SdV5u`qa@#lMvm|vXLzD=<-`Qlm}2v;-$AR}5*|F+^}O%=
ztjD~)rDY~AOKHK$9z#^d8$=BGEMvq^JyNp70$Hj|=k7sFqc
zyB?z9t_VBsER>fF8%P`TB8#FE50+bOOrQn!D$1@EyCW%HRdq`6!P!&FKW0!j3E>#I
zv(GA5)iRKVl^;gCNv<2y@suJ>Rx#II_1L8$9|cEkeCqA9Ee(vz2A0|NOQCrxSGz6#
zFe};p3a{a`!@OXd1@9R|)}o1h$)gXep?dRr#J&VF+GZnzv>s8A9pjC`!d<)4PYc+LQclp6()aew91xSlSmbHYRQX@NEhTtPj
z0+dO{d@8I$pVtR)+fo{K$r{iCv#{@Ti@KV`;~}^&)z-I?(3EVO4-P9%%kCb$KtVkR
z!xklEHN#NQDCkW5E9Ab(+I$Vl^u+cUpx<_KQeSA_9IZJ*Np?x$OX%+di)kw`AopoP<{r
z*^B*+rU6Z>O~>0(=X7@drLSTZsdqr4HXAC>+?v$Gwnd3lN=GojKfW?0BeFNQO~Qgb
zoIy`tAV0;lUG_NK7fQ@Hi&-^&s%`jWyPDzb<>amA#EIy1F06SdaDdf(QVfh6`xwgB
zj}wuyI>Gzv)Kq7iy(n3+jsLOCAl>bpeu3A!yL*ShSWx5ON)va9l>E*UxFd>)aql9u
zjrG?t-HkXdv6{8`^@oBNX^E2O=va^?mtSniRXoNh46)37Kn9;>@Jod<8`SpIC#h>E
zBr7&`5@Ymrnj8PmSpW@&+pz0kq=}hEzaWI0`c?YFR2Gze8ST(8Vm8K^Mdw9wHRv#r%%Dr8?
z966m1V`d5-a)%tT6W+5SDx^k^Ak!t~kVp=VBsXR2fjO0;IxDZ2Na0SUKzKRtOC)wB
zBQORAEYA(&Cr{$S_ELXUh|Oh^Jw-UamUcSk%%Vb(zU{y>Xa<-hswz_=hqtgg{f0g1^rM!11n%_)8=lqdF2I2GA4~a4E&e=?qR6qeYOq
zZwG+C=82EG1Lm?7*ugkyb-4Jz!3&C6@2v0;wP5Y~IPTYOz^xwo6oE9$5s`cuAnh{)
zpLf87Qg&oqn-hZ!UD>zQqT~#X%#;~xj9bts0MKv`lhx7#&|s5bg|H33{rQsUJA+bq
z7>>>rz_CCKq>2)0a`%bpbS%EV=QTiX2NwFvM3~eX0bkss9C3#?6SBh%4SznG*C?mo
z>h=H}Y-n^bD#iidc$Fqu*>cRK<}5!UoMT$;czS)TRQ}e5N%>6xaxs#0yZ|DpPe-GO
zz~3stwk1wV$+1$g5{x;!71z9z$MMR$2VO%#B?l@nc!He(W{UOsMGk}OYOXw*N>Lv7
zr8SwVJff+^{P^*^TjTs=3~~a5kmY+YnE*~zJvAX?JD~^BnmEObfie^x(~R-r3Kr0l
z56GhppS*y!c80eJkyCAytn}H)mM)=SJxsxqj$AY%R?@_moSfWY
zeRR(z9s|mzLCDd@E!H~8ev4Z+u%HsvLJ*Q9={}-$j+HG)pV-)N1Xk1*^R4E|
z83%*@UU9
zk|qhhT{!_*`HKfO=-@4nB7FpM
z?_+|OpP(`3>N2y9pXdLSM|CeZ5aJ|3s`4*pa7<-1rWpxWZ5(uVHd7V`neA%hnQ*K0u{+??_+t%73_4YJOl|X?Hh*d+vrB2eO_!y1T8JFsSd`$jEOHJLU
z?25hM1#;U14*=+z^8W&qHXr~KUA}0TBCos$gPy1F&boTut9)w;N;XbE+WWTrK!j_drM4>eW{G3Tn(Ny`1gqm8lJ3<@^Rn`XhtBHLTp)~G?GyU6W5N|AKB
z9O~MmE8rw9JUh|3NLY}UTq49#z7%7*wWnp^t33VjC;+=L4+G}poWWwqk@v?S+JEDE
zNwxA_n14vuE97*CNP&?aEb=-7p=;kw&f#kBEhBcMPjG(Uyt=T}x;#buJho9=V
zQzi$Y$>14l!u!9qJ2vbc@qSc`-iq#*8EUy7EnB8*XpC&%7*cc;IMrbB7a#8{AUaJ_
z&UzzNeu|H9Zqcq*!|@H8o;gIivf)`>bHm@k$R7BW`eCe7;4NkDU3>QI4SVy46xobh
zS@KCS;TrWv=EnWH!(-DIMT%U1K5(uv)dMNOv&5qnOgdZ^)NxcRbj$f?Wprkdl|uyX
z`prA{NRz`CC&?P{!3aq@I53ElQB+do4ic56xTRVf-j%k*DXy&bznH4F<8e8OLeAtU
z6o{sU^|1iaU1$Zlx*9cGIQaO16RRYuc_Uo-%4yVAynl!HPci^Jl)tx;!&tDj(lihb
zlJu(>;^eX2c~h^;3+hHGcRy5Bg_rUeVW5C1bi70cTqUvs`hb9Tba{5mtlZkm=60EG
zYa)iDMzRvpajg}{G`EF}aN>tVQ(VKhu8BboLcyhdS|^WH{VZB4Q{de2@gz5cxsy%pi0j~zPl_oX~sp`d^IY>qa_V#%WGn(EBCjHJtgvoC)vBqp~q-mqa
zJqFA^n!Owu1NM=ITNIw8{*E9_TQV|hiKLP%^X>R&=&F6>)!!rlV6ofr)ja@A$$GGC
zHvfWJ(fRVoMy$XC?3VI`wi=`(UTV%L{aZLA5T3-bj})$kMs6%*w_7Oru7ZXC{sVL(
zt{hF>V)Kc0Y{%rgx5AqM!`cxbB}cw3M$Uq(1+k5l6LSm{A>4Wq;M7fo3~&={Kq9+D
z=HmcKCM@(ptNomWn{2Y}Z0uEwC>s(wg95_r3PMS`Nx&Xb+t40JfDSD*12YO8=e@{
z;}>*>XfdHd-LgfcE|*BS@F1ZhS3Rr)7FU4Qd7KJhIh5coZ-rGuIF?tajn15i^0#9*
z0Elt+wE<`HfFI)1^lT>+8ZSj}-mCl=#rj>g?b5wbRNNy_J2dvUn_X<7Wc(iQ{;${W
zD!hLDC_wN1Ab2xsB-eq@m;s_QP{uFRuJefgp*QKzqo*I7O>T^;Y=`w4e_bwk@kN=6
zIC5X@r?M-oAasoUS}SDjKf?-9AD+kON8)`t^w(ik8S;LbS@73feUmsz#!1n$hd*ym
zMFuJZzs_+RBC*lPl}bffO!{5qtc5L_|f*bfo57c-Tm+#7LR1=
zHIH_214>HD^$9yiInC;AYpVk5R+VDcWAE*)3~_*!OvQZmrkHmaEPp9%rcNoco}0Uud@9qtTfb
zHa$Cvs@1xx^&i6p2TSO0GQL-@;LWss-)49B?B?65s?e&mmkH+Vx0H{&
zVkG*nlgS40+5IC7>gEdXm@lGU&6&Ny2`#w%ki0L8wo-h)Qy6o@R`%^mkz42-mcU6b
zj3k^quK<#s#}BK5Ep`J0Wyp(=pTkXzsjVnFXu5{gDpFm3rrhq6QEQdHTg#te#^W*n
ztXlv5mPwUQoOkVKb~X=N5MLK>e#D1qF&-qNER#oIa_p+$3u(^n!-9O7S_Qt2BCysu=
z{$+S2==9K+#Y5~m!FQV*f7w_jX0dwOmub=PZP$Kyud-UC-pL$kXMeTE6!j9)rBVO~
zGf=#3SqZotTcC;c#w@+9>X%Ce83{uS_Q}nGZ0&yS6^>-;`UQEkB#XoQhR&yNuD);1Yc6J%Nc|TjFPd3|+0vobH3G-yB=B
ztCx)N%E~=nxk^Z2{?n03(>?g|hKP`z^oqAJz_k;7SGJj{Y$V#O-FseCA4J!zPf_~fXUv&N|SFd@$br(3YJ{I71WRwJhrBMPkfn#PDBxM}b7}w^x+G5iYJNjeb*?O?_
z3VQ@k^JF!E8iRdvR{Od^o#vg7QK@N$=fwC~#dIb3VJy9rVa-A<%^i~;@8~P6cQS7-v@fbv3ZMQlo{xdMo$=VG)BzN@*AJP6?6+o6
zmu`}`ajA&ai}R**G518zYbyqnrntrUmR3hj>#h`#-rz#?*0@PKf3e43rUOz0`f+
zVl94r<~0Xis-mNtv|2{Mru9iHje-6L9&(F^ql+(&JNtyD4G$v{lev{#
z7|C0BKmjSylz7AmxG{dTR3|nQ4{@{FLz>33K!Obm@yA+MJNmH%q;Mg2*q^1Q5Si`x
z$>xjsLZ-&!9uae;sgi(=S)$)8ITv^pC7T&+@XNEWS9;AAoj2m_n^({_!?qU{jq83_
zR)tv#qasT;jxNRBMaDjii5?PhY0?QTMy@P6d)vh9@v<)mk@R2`@C~v0Rb^nZIWGez
zA_ih=lk8g+UALZJ@D!S4CH?%r5uRBUD|#!xyEwQa<=MRHHD6aL-7{WrFYQ$K8F)MK{X2ty
zVuQh0}Z7vX|lwPVb{6oId-U!|{2{Mtoz91&s}SCZ0<3SpO)D
z<;E6R(K;Q$%YEQJlhl5*#9?
z`&5Fo(oD~uk4P)U3b7QY&x=;FW#)^P#K+Lo=}A!Gym7>L+!O$+MhSFBMsQ@_i;UnA
zx`Giin+&-&X;O3!sEHQ1-0equO$`t8fF*!;!@%W=ZUA8{hUauTPhnK!J3S11O7U2K
zagIGu4Yo4i+ur@3cR_A;d%xmUAu=t;%R9mB-wUoI+$xbANrT?Gtz<5b`rc+|4{<8Z
zy1(2qX$FG(T5W9f-pnPQY0^1GXe%Frx`%uDj&I96uTlWrG)>?`CR;URSdgJ6#Tu?I
z3{*jE>(~O|A50y-gzF($i3Cvg^^?dbV7~^e1RU*eql!DKD>$O_cyvvkv*?Hl$D=+#
zUP1YEAJ{gJrVU692&@T#qI?hGYV&SJ5!;wO@*xG9N|AoJ!w$R25YxcLK6JWvK9
za%P|YlFtJotCh-xx}cK{p+J`$;2+2-V|Usa@DQkcss~!Vi9y+oUiVy(Sn^+2Ymh_i)?N#IoM3K
zf^dw!9u!s(KhlC&kY${suOQp)jUNOFdq*ph&t^*X^PEFoS%?$7gFYQ|gi?^h53FdT
zKHL_uD?ldK=oi64fQo+2M!=Enmd4`SY~0)1Fu`u3`;&08Fau`mRhr|0v@I8)w^ZKQ
zC=vik7FD7&&q5;p-VX(8aiJJBeL?`dsWdcb1<3G_1?q&J4g)L?fK(<`;r$-9d7Pv5
zPQUc;d>>`Q)3j+a__&moCOMJ#{R#{{D%0yO*f<9wmP~h-th#S8k$uVzUn2Edm~v=w
zF_>vLYrm9`Bi9xT)`nl*_W-Po99a)_-NG=jl%I5Dn`c
zgigmu=y^gyMBj22HTs957e9QzmMOf@7-^Fr%^b%Bvks8#1zQI}{`f;3l;Tq1GaEVu
zdMBF6Nw0LbNmO~5Ip(wh--b=PCOu!d^$^3z0cgkp&Vds)iTSk_hep%~$Lg%yq-gTX
z6MBYfgc&EJny#|6x34b&6}pXvBNJ$FTr@$sR8)&yNI8Kj^XxSX9W`31)D$P%qO0`9
zluMg}Eb0mH$)NNFwmxDSSuuF0m;;Kw9tGce!|@P$Qu_yAk68s}P&tpA!o282KAm_N3bzCQv&uN_#zz^$i6ZQZ^&%
zP0;bg8#QEI^^++8%l&kxkVxbjb->u4{rRrQo&Mj$e85(&K`rBW)PRQw`M?9HeMO1&
z4$}Mju!9HtQ#?tDC){t-s8=sKeR)t&381@oCz#{GZb99;X?NRfNT9n>j-2i=>O`AHo4W$kcY@K8&8(o
zH6j~DKnw%mQE~SbVYg`dijDq~mzGei*WD}m?{3sX{Xqjj$&p__24C;5F0i;AMuU?8
z8S+xGYjyFiw~I|?59+=Udy}>RBm%LoKv$NHwTutcSi#-nnE&nd?AH?Z;T$?mf7tXP
zQMp<~I&~m7575tWvI4PNx$lwH-Apd|70RWQDs#n)gkK1z&vz(qo1Bt#fgl{9zn&nu
z9rqeT{$-+|O>Dn=sh%+nTc$zMcN*(k9)J?~7x30ZtFPE^S0zV;m889#o4DLc|IAY=
zt|>s43zFZu|JcNw6o8Y
zY}mo3K1cHEulJ7b(z){Audaz^wC`byhj;S2X}
z`HMe;hf-1!&%6#znLig02;Y{T|B-16p^diJo!nup>JsmA|I{Yb8aIhuPCup;Nb9G^
z=-75%5~iLVwQSetzSQz@@NeCM*x4P}TMjlL;PKHnJB8YdLvR8W4F|1)KvYb&aytBq@F^5SjXQ-q#Urgg$CC}C795U2)#ztY-VBGAcY2aVIco8FO{+Cg6f8m|G
z-)E;dyp;TmY6or(F&$df+(#eI?>`m2{EOH5x7KVQ=WOC&@ipO68}}p^gO0^~6fejQ
zUe$vTvcxNi0ETi8nDA`p&QB3`Mvp$OGasmKGh|44MhtPCoZ-Du!n1v{;_6KediJu?
zl{;C)+DJwn4(bYw@^S7U7UpBL5S68UGsV_{6uC@i+I=(SOe$bVqAuMsvA}tU=!#kI
zi@%|}xH@30UwP&moF@Dwztm(p^HRym^1ggH3(Q~GQ^;(8E3UAR2^$^FumW)~Gi~v5
zsRf*5Q)1yl9`zw&0Ifd)2GH#V_xl=zbbMj^&W-RkOp9e0cRzEs@1s>bN=B>eGxRx^
z`o+G_t(QFu&Vz=CPufP_lYUZ;KMK?PxFdV|Fyco(Wn?U$xiOh6`t-KG7S}%g*_%Bd
zXkBmma&5-@X0BWob>VJl!C@2ge}^wTmA)ag6g7*B%RCbeF*i*FHKvGYT2&6{|M_aa6s4+Wv564xy8T-T9NltM4W
zuh%p#4kNi2+8ru{ermVBA&qXkITLAwFJoHV+~cphzM?f>Y4u2kF=ukUbYC8C>wVA4
zd*F%im0K<|wmtuyz8dq@J~O$oeWwxQ?2n>-&tBzM%>DxMO!oBqTiGWoPtZ$06Hg+~
zYL3YRR~Ur!!bEgGWB&MdwB>pSu}7#a=9B!UdYbIaKUvFw
zSp_1-9QsjA_0n-sB@3!+?(etT5?<8**Q9bCv=Q;McRhkaaNe{R{UhTcS9Mp6r`skk
zB$2n`Q*QGfi=LhC@d!`j{xW7adXw&~!6VH5Y{z<-`DwNDc)_|m%AD6==kZ7t{@RB$
z>2mkl`FjwiCp2EyC1-A)FWqAg+l(I}{A?=>cdvU75^@e@+(mpOS6e)aaYcq=-pR}mZt
zzYndj4fMvrq7^!P(aZNs8)Jst;Fvc~7*J!B_MPW;|(>F$%jvOUS%d$VHf3oEDc
zIq7#EEA~m{U&Rr*|B=e0tTkN=-nZnwrCV)TPUW$~;LThp()mTq(&b0@@T>uaC)Ma)
zMEReenBKiZp73LYa{bl2-EONiT8X_MM>y5IRxe^;JdeiSKi_nU+A^vAQG#jlgXzv%
zB(yE*Yx`w+tG5S&b9SL#od3N2c(dJr5#g~H>pYhrG2eb>xuo8F*X*p{Dtj=uH=F4y
zW>I$Ix_q&sf6s~fx4+eQc>0lrk#eQ*?o|9!|}+$u+Y5w+1F|K*#*Qm&h7
zD(|)2@|`Eo4_nVAh^kiwHb<`3+qt?uZJFoVGXdu&EA$2QA*U7;8c30a1_sHF^3EOZ$%@2K
zT7SJP__;67yRnVf*tAgr-^+{S*uxI5wQ#Govn!e^T12GZes!;th)(LLI4QM9#Pwxy
z*IGmgxfIt^`*iW9YP0DxUI^1OYIr*De(xGh>y>5m@a%FKF>_8$X3|JdqUU{R*)Olu
zs}*v=l;H)3vQI(2=h31#?{>m_<@!CXWGk-Bj-0Q=tz`G0+AA!GcWZA&T!piDBOInK
zXv(9LhVQ9})1~7s@)4C~EPX}is7^@(+~}|FZX`Ea1Z1`W+Xnan0)n@8m1$@7@>j25
z%(M+k?u!oTeVRhq{9d3xLm$}n9?RuZSB1zvovyB~^Rcdf#GX!S?;f7;j+D?^2YkIS
zqUx21hs05Fg0Wz!Z07*qYxfaF7B81hjkBMG8!qY_Fm-=UG2;;Yh3OlENjJe2E|t+M
z$08BPh?)O}DqeMD%|(aO`W!~>>$aEdGbZaPVy?0UYq<8}G~$;QFH*40Hd6jP?x9`W
zvDygew69+txZ$_z-RUl(Nl+5Bxw)y(Yy*e?rV6>iE=j&O>1O+fYcm?c8_H(NDu>fX@{b-FT+I
zS%hZ$%KzBjRQ`(Zxz!)7HJqp2O+{pimN`dlYi*q%hJRfEm;UB4+B8$qb@Fz|akR=~
zf?fo|X)G`5a=9KW`&&8PC)TjHbX1&1=)vpx+C=BkU_^wF&eHi&`}zQd;9L=kFs5G0
zsA_8Oqv3NWArbq+5fO(bVdu1*xBQ{zZ|ZS7*-+aHtdU&%C<9Pe7Y&(C5Rqj9UGJ9u
z#*A?|Z%!#Rbk2QWoAg*kxNA9)C-i9wF=A-d+cLS&lh1KoMMBPR{H?n+&vSZqb~?n(
zsKM2Px6C>eE-+m-SSp(f(efiqvUyD5lu_|iZI7Ttq2h0Ijk=cySFfY>1-V^{LoTip
zJ>e_coAdCrfcojA;C*7$%+c3Eodeh*h0)c@VroNwpy_VuLyfz%uf-+!znHwA;x=UR
z5IYK6vMdvUNWfd*nOz#pMNPpQ#@BZarPtyI{U~l+`|r#E1Sx>d3CZCsDvbj^Ef&AP-Uisx{!FJb=x-uQm%2
zdyUCTELG9=R(Z|(Eo=j^x01~vOH%G1HB6;MbG>`Xz0Loqj|%N8TtI)2nipD^ZrtpU
z=}XcC6Jmm;EI-eei4gKUntAv_y2%d+}F2RBmx_Y9s%w>(Ez1o5nr`)XN#8I;pfRT8L&DJij
zyR%TIZ*S_)HLS!Pl$dTA3_53Y#OS{e`pm=$4;~|_#fJd$Akz30UEo@V(i~+#iD4VW
zivoW<sm%X2&riV92?lO<$loyg@VeBY?VGyNY#4W$yU9QK29GXc?q6w
zXWv0O16uIZ=SjFYIM6;|MLggHs>!Fw>aPkR1~^AZ4pKy;l!Bozu_lSSuwiSU(SILw
z48R9Z#Z#oRaRW`(tFbMf0qX;&mlJo3!q5mMJMIB7+(YV?)2}KA$}TKx08;$`QZF|H
zaKJ|}!BbkrywMP_?78Kz4n}ng?zt90dU}_o@?g`H9MS?Aj9x%od@RxQX4{KjDc`S{=M{eN|PJA!7D{Boq-zgvCBn-DBY
zCYLXip}lu~G|A=(YgLhs^zr(!{IdS^w<-lCu(}_V04#QeK`#L22GO`%+D^E}od?|Y
z-!R>|JdH(#UwU~yn>sKJ)UuoDuV;r0^cwpy)9-YSd*cgrc6H_6HgtFaR!0>{KLCSB
z$1r+l-;Yjssv(SQ
zJWsUmvBcun#*s_y_F`*QWc?q`-ZHALCg>Ur7CZz99^BnsgS)%COK^90hu{|6A-H>R
z39i8t++F6}Jn}yC&CL5_{;*j0p3_pbYj<^3bziCoV0-r8GIr3){3kx82!dP(Cod?v
z^Cw6M3RF#kZ0#@5{qxxy@KO+etY_kZQbm8g3(V3C813H-^3P{){AU5SP_&bc{qycI
zjP&!$5$*NznX~2
z>47P>^LWmJZiu`CDhhWChK)hx>OjdWP$>I(28#)J{U=aVos44^oc>4Btw8GPpz6cI
zpF9_d3cwP|y4au;_j~a=K#{}UEU_RJOz{GPxy@__9Ul?s&I07?IsvZ&dI<|CtQ~2l
zH0uUQqZ(2Xr6Sw+dTl!yPVRjnoXDY#H|cl%KRuxkot*%CxUAdt*l(=HsaeJgS+3IJ
zD=eTyJwcp1PzD_;IeV$86^q|IS%pNZMew`lAb24~`n>{DnE1@2BpdoMDD4MBA_okT
zM@thFOJ~v+&`NRcE#XbXsY<|e(RK6w4cuLqblkWo{r2jR7{K@VnY+myFf`ERh-04T
zXlZTw3Oi!k?-u2Nx-g8t)$Kv@{}yV!D#^25PrA~-ezKC{mnY&rVz!3Q0wW+XYg5(y
zymmbr5k2K)DP1Bjjm`iAFwojd0XmxYISrs{6I4M?kZk2Z@J|yD`3oHX)dv9^pd4%6
zzcK+o?E&=;k;h$tEFu*Y8{%U8>xf6(N8m9BZ3|F+|92HxcK
z@(!K_9%yu+a_PMZUSiOT3BX;el0~){HW<2R!4+2O_$AA_vfYx~J5#Ix%2oAG0z?dZ
zlP|HkE%mTmyL~!w=^;Q-pZDTG>X(XjqS6D){$!u=`XQ(5T}n^JTf^4SzY6k~0p_Uv
zQ#k{f08CRd#1h(O5W8fV?Q*g-(0HG=UP+x%2ZZn!AoqW7CyRv)5&|e-onZFKD&G4Q
zbLE|i8tHgud)fzt!?ih+C*YqmD;!$L_A#m(uL1PNHsoS8u1&--mZyKnK;Zv|ZD0XWLCQK!zpdAP
zt;Y9oyA5gL{6vwBTi#fCGoq^Kc$@-T9v=ci%zx^n-ha+{(^CfRnDG5<+bJhXfA`z-
zELz|$M#dShI`0=tTfn`;2jJ<&QHlTyNb57;mQW2!h>Mmmk&-a0i!|YHe{KOu^ehWN
zl&jK+AEhPN2RP2(!JiF_Mx$Hj?en#a(sqVpVje9p$>k=#W_;A4a5
zb6XDk_we1rt}wTY_~+d#p#{Aajc9tEw&W|@2_IPRzd8((1&;=@JxxG|Yeidsox=Jn
z<>T!9StLhS5?#Cgg;~j{$cu0Yza~D@(dkvSqrz%d(e347(h_loeh)2$8dvRS-J;%XIp#AD7&YP
zE_-8X#Hi%5)+#Alc!`bfre^#EnUo`Aj1!5+eExoNVt{YHeXj!kFMf)}Tq
zXSMg_v-UFIm}~dh$Cj5P|KqpR`xxAwQPt($7&>B1c+gIcZuX8gYnsw{?J$l=vYH{u
z(lsMBux8BwcL{)saMpriNZY@Vt-v$jHk_{ru_au_;!auAF!6ismG#uQzs~DjWrVf1
zN=$8<*!mD3?=5dRyB&F$8U44|=?}Xc6j>61kCioOit06Q+8_IJTjhRietg&-
zUO7B`eYYxJB(09$nOTRbpvDdhqxw3z?mi@@@Xh`yb={LZcwNwLrcg4GC?X6zABI94
zwJ=-}^3x|sDBbihn6RsNGoJBO9ib`j1w8~kJo%Y76AqhPTcm93C!Y#-OmouqxtZG@
zSJK>zN^(gYcwwo-S-yp>krd?s4Pi7$=v0pPq9`{xJe*(x2JwN_{5@71fT+DEG`i7N)b$JsZ8fzJ@@L@>fL
zfDzj_;m-kw&cTTEt02YIVa?O_?K#~XC6oLz?=^OU<*gOpt!7r}S2hPAc}^5>uJCWG
zT#Ulg<5X}p3OhN|>FxQE%Srq*Z+(5e
zfl1*kEUWhOn(KzaND`%{jTO9TUSxQ?fsKo`jZJ)Va$zmr?nrW9Unr6@lfDnLLLs$+
zAEHi>ql|yd!@76+XaeD1T?}2ZSwLR`mU#dcw#Zj%6dK*
zRz@Zaob-oqvNx^E+VVz9WD4&I3`iqP^CeM0_z@QK3rhAQ`(Y+(zR5t(QfWI>Sr@w#
zEy_zxVIVJNhHzYl>7gTV4yCbkxcS>K9D*5svU8?5rZolqK#81?h@6;M=oqDLzEW%Z
zdpt4R0GbFxsVC$7KB?Z4oecf5SS+4M2Aj=QZ+|~AYW#>q&cpBYB~(R1qj8qu5h|+7
zd|kT=kHwW0Vp>`m7{L)GIGDH^DyV32Ish2}w&MV#fJGaYm>cY?lNM9HRYO9o?K!AT
zztt-(msNGaOaGqbNjhru%b^<))my$c>Rc9Xo-}0k+1AJ?aBNKalc3PA(WWUc@bpYG
z1F7TXMjPsq7(r|F2A{$AsukG>2R%19$4m8MVK`LoNyk`J=&a#Rgvraz_QW372Zc&s
z>k+OG7ofFF-ENMS0-gYK5lyAj3Hl)7E(zwZgi&Rn^i*
zi4F|ro9Q%J1xttVVSSWz2?26CJs}~Ot!)G{dhmQA?O|ku;+h?~uyzfG>kT-4j#Idz
z#`@*{WPw6~C{!w3OO<3YRkQ19wk+XSiK*3Qdx*TeJdXu6@ii7_3c>~54i7ZvgW2id
z0!9qm|I`AczSj~test>KfYt6o4QW`pK~wJZD0H|x%bM773bB-d410r20B$M-AS9*;
zvy}mT|HK%^mVwYvOha8pE5i@h3_g&Hf`0A?$y88MVd>VO_PdhbP^u;BjKbzH-WKTT
za*56FiM~V-pj247Bz!E`Ba3inogBZ
zE#NnmYGAbggQPxSMxbzD92eYhK}_M)VbG3Js@Rg9P5wzB5?}XngB7BoahSUC
zLXAl}4O6}s%8V)ve>rtIGwE-XFgfYLBT_54H;d|8MypdkpIul^6h6$5N@xG!=dTig
z^k9Qks!#Rwe~{>9@s@97ny-2&VybF}{aSLMLbGG1k0Mk509H~Tz{Lc!rc@xMY=Ocq
zn0kjsnp0F)ERzv3oJekkhWKzsJ%aFskfG7%)Fi-g6nfXHkw{85l`K7S9G)6)w
zcz5^YrQzXW|H493Ur9yvQfe(-by~g7k3t~`$N|DecqpnxsH%^*r$ab)?RkoRRe>QU
z7%*un)Qa0b224eMYA7S1Kh-Ma$v%6y2uD+)jg@69|BbfyNMICAOEAc#
zF|*RkF_+}BN?Ako3^yl?YvO)Cz~RV@K7RSa%^~Hg%&%(Lha9|b{#&ba%CIwB6+Phh>F709ETRMYb&{eTAeOa*ajIS<2n6jxk?_Ynt5
z0s?Oaqp%8+v`Tuat*$q4~^R4S%&I`?$Mu1jei#!ttMjiC@J^K!trRERok7a&&vps9#tU?HPzMnt1|*NDzR*(6;vj1f;I1;z?nVw&foer7os
zu2Q)cv@b5MfFMaI>;?NJ|ucy1&G@43Z;Pd=2^=(XA
z9N5Q^UA5-#811T-DNF3eZ5|z2M`1F8OQp~#guSIGH#0QMA3-2~v>Av%!{qp^@6Qb^
zdV&p>b3cEnW3DMGY1Nqc&VXxJ;|F8`s$FaZVFxSE+yQ}XtOGEd*i5B-*MKH!y<
z9e$*~--sqHJ5a3|h@fweT=u=Y7Sxr$)<7wttgae?(}61eL?edU
zt8Kng<3%hM(;ol@AJm)|S6mYH<-JLOa*Db>D&Y&Ynlgo!%{xsE4gYXa$r%y`=XJa9
z-<2~}8tNN9B|fk!6nl_D-eK?iP|CFKu!8te02_eTZ}a`NeMLt1h65Bv!Zo2pm>JR-
zYa5!+d;7iJm`n!FcYuG*i6%OmapuYMPFogL5iOZ#FuihSe^;lZ*|44v9)oUnh2E%L
zdlYVbYz&fnUtC90Lc$;LFozGwh1D^&)GaK!=5gh0-RK;?C_I7zVul1|k`R8};U>?kZrO7US5#Zmg6TAa?0s4sMziD_A(
zMSAA%r1QpUbsC`v6uuK(k>=3uue3N8N4PuF?_0wz2SOuVl||GBCBHZ53!tj;7J4Ab
zyGB*)l;0D)9DXX%ZecpvppQx@8tt0M;x?MZ8yJ7BvkIEvf!ocmq#I#-dU`tB8i3cC
zqrdas==A55A{J3Pj(Ba3!bCu{TM09*}&yqSYDUpJcS
z?BX+O7QM0ok^C}%6_mx^5IWea%bMQoQ@G$HFMGLy`ty0$biEfASORqL?BD=)5-cs{
zTCQK07j#bZk~yi4jpT<@c|vFB=L(Tum5OEc=_gtf9L$|{?w3vhr&JMUsOkjUB}P$N
zI%-+|=R-|m6YUjO-hI-cPLVX0u&Cv!QHRHM?_hMq{u{~_PLC^!nzz@;waDWaUaC`zwkAP6A%EC4d|eb5s@x-DhOBSG^zvZX{&;4LNP{x+$th?pIfKe
zUjMWMvk6F?yr@4gf<RabQJn4Xv5>+t3m?)}mul)D)M2UEnUiHzL3`ptYOuc^jPegzG
zV0lhN2%V@pus!oE{o=ahd@%71Me#Wwj*HMl9uFjqzdW2PNC(Hu&%5)y+8xgLz$&6j
zc@~I=D_^cD@H
zJ_$AFG0NUaaagP;{`pe~B`Szl{wxOi=#bgMJVRToT4)Db`^Vvg_*Z@ffgKvN@l>B6
zzY6*VsUT}|k+UOJN6+BiJS}Fv;aJf+OSnQdx=_iukB8=B`>w}C(J#u$Z)N>5*aWja
zUHDhKz|{RPVfsl)3?^n8;8ZF#8o(bq=M)g;in&ny6#*As1v$fEp@l_SqtXYpwB3Sj
zTREx3XAHgGen*n4^FhY-3_%lIy+5RI^e-pT4<;l3%<(fnZEl-d2*-Z^@UKt-`=|Nv__VP=o#{AN
znC+unPb>~a$@RJROW}4A2JG}K(%qlT3N$7wfNzW$0
zyO3#L8FR8Fjl&SvJ
zivXlFfaqk2ngj)++(h89tS}0Q|9smI+FH6TM_6jMg5K3RmRJWG1fUG+6@?s_%NDNR
zDm^106-fkMH0gf@@E&X%1SYQ{<9HYtLf|apnaU_J~Tc&V<-qlG=5c&sa
zVSY&fCK4PD6HiHjy_t%{OmW?RW}|Nmyby(uG=>c34~PU7Qe^2rVj_48eA_=IsXR0T
zP_~MNwaP{Kzk-UR0W2fI#>1gK26jSRL4YS>JK^S|HT0H39K
zqbfj2fmoV!v1Z|fUu0{PB^kS!lRHV)Yh-R6Mme
z_*cLDH0R7kqIIVz9Bbc)MM!P!-nqA;OSuoy1qCJ7hIiJ9^^OSZ57Yb$wp@X~Ej@sM
z%C6BS1U>oyW<)A2Ou!!X{o^AuEH&yiX00m%ssZ7D#t92kvDAvheiCBQuv+nn*2!w4
z7qT!>f=K@(xS_8vnSn{BPp+ZJ5pJKb`uG3oN}mYaF{MgGTku7ZYWIWoAV%Lqmr;HK
zFc^k+V6|QRi8V~J{PMLZ=*Z)*s)SQR!lZE;|9~D4!3PvKbhJPyUu)$^<5jcgpPPGv
z&4BL!;GML>ohHbHnP(w<)b(OKDgH*_8}AV!Urh{iIUiM
zsN^Pr0FbFVb7Bq3V7S9yKEt0?GFKeZfOeB_sdB;m=_FV7_1gz$deeJ<8O!U0NQQ`9H++
zhZx2{%Qk95KY+a+A{Q0}TDyFd|43foFA(m5NGDuV@Mx1rCMbC9ui(Ppy}b8V1Vv=R
z)02Te-3k8{Jm@O|D!TB;-TmKGlCJ>BOP0P;RVe;oj?2o5T;=i6WhMpY5fSfKSO`@?
zLo0$))QG8|pr8qJ_UI-{=ivC8PZl>A<^wxe>Bi8&%ZoGsL1Iw=9q(yihfCeC(coR)
z`qQ|D61x1t2HHU&*0-^Ng67srLOcs-t1mDR{Y52#O3@r!Y(~V*EjB~*-JM7t1@g{&
zb^fghMEm9mQO9HR;zo{ydYbucSclQC$l&nOQCGaa_4&8+^B{|zS%3t&Te)G74i1h$
zs5)=K(Eb%9fB4dfA@yBbE0LCOicC2
z8D9`wUJdlPtW*&asNgPE-hkIVx)l=eH*+lg@Lqir;slFKK?RbFggWfuCz^|9i5~!n
z?+nk-oy)&``^Z$4-Mw^(bXONInun%uW+n3lAYBSV@H!1>ww0-`s)ZF8tQW0d2VnbJp1UVg09DxeP3Te)x*u)U@<{C$+WSUjDn+5VK7v7|H
z+oV4m+V9X$nsDan=LDSPiXC9$Re8e(3h#5zDI!dn+9Vab7~9D5;3U5IDGw26R2mHr
zf6DX>l2Dg$o5KF#T)O@yYB&xcZzJT~b90)6nL43oQO#qS^>lVH?AX0EzI02mUdha3
zQz?TpytBUow7
zP)Re%I+ag$mFX&1dbudj{y5qH-hSMH+lRIdpmmm#{&mKy0U}o4>s_JyZ~=Ri+;jEJ
zZ&Gp^vvz;-CR^kLbLSK;T_AiPg@f=dV)=3~8hU~1zeOUjUCUvSnq612k(Av?BG)e`
zP`0gbb8-IOWmQpt1TysD$=+QaK)kQwUb1CFqtZ
z8L3Io29<`EHnLiuO5zWq22R*vsizSbFRMz_p?jZ0W6$p_AhFp?Ak4_gFGH+%mUnk{
ze(Ek7@xaCkI9y1%xi$96Y*TK6@XV9aLaZ@bG!sL667vHyjGeWkDKcafq{3
z-;X|-&B>z{$AgVXw|XO-=9XpxF1B~6)$0P+RLSo@ViJ_(3=EL}?Dbu^^R9LXcF?Sj
zzbDKY9vOkdv#l!8u5DM{fK&-#!pZ^!w4G*k%Q*@#KUS87Jtm2P%Eu&9`4DxxH~h5P
zF&hiE{VDty1)94NDCVVt7)z1e7yJv`N5P|Km*;|}rfdNa&(pxM)%LXpVry&bs0pJ<
zf8x)M&LWqCh*74G5_il_9=F5@IEzSq(7xY~%N{eP;Xe2i*2ah-=hDupeDx_}wjcOn=OA0(lx`I3{ahaBKD)j2HX%V`tUdq!eXM^Z*zrvBc^PgHgXR_0w`wDx
z<4~=4WfVe8HoIrvsQBkfwdl`h*H+jAv%{v7!EZ}l`UiIkznJ=|6OEC~oqtJ4>CqY;?vV|udesjEFH^>PaD@p835ks6v%;&Lk8h4OS;hK18s6ibe;ks{5)o8n+Vg!y)U}ddDB)Nf
z9kM$1%s0KJQg8X(IBi{DJ*=~i6GjNJKJ;snZ_j1!dE6W-qB~A~r~qav<$y}DOTk9>
zChE@dS=IW%D1OikvaRX*-k2m?q@z2m2ckK?y)on8F=B46IOb}r@x_1YuyULXU)w=zl*?j8BULTW8l!Tv_4mx>-~
zs^M5%98Y6GA4%W&@e9kdPNx(srIE*9sW=u90xP#e>m<``&R6F2obNmV7&sbJ7IG0%OGq|zL
zFf-LSU!&OWh&Um%cJgC>GI^>s*cDXQLZkEbxx0g1KVlX)D3Z`KaNZho_u?`|dFWa|
zMHNGuTasag?e~3$+xNw;Q!FDzrq=w1rNM<|?J;;&$XPXt$sFJlh0Lr#zy(;Q2(|b{h|1n>Ea3l6~*CQ<<;0?c?a