Introduction
One of the changes in the latest Kubernetes version, 1.30 (Uwubnetes) is the transition from SPDY to websocket for some of the kubectl
commands. Most of the updates get most of the attention, but this is no less important.
Understanding SPDY:
Google created SPDY to improve HTTP by speeding up webpage load times and enhancing security with a session layer. It allowed multiple data streams to be sent simultaneously, prioritized request values, and optimized communication by compressing headers. However, SPDY was deprecated in 2016 and replaced by HTTP/2, which is more advanced. Therefore, for technologies like Kubernetes that require better and standards communication methods, SPDY is no longer relevant.
Why WebSockets?
WebSockets use a bidirectional communication protocol over a single TCP connection, widely supported in web servers, proxies, and network gateways. This increases compatibility and communication performance for kubectl
commands. Here are some points about WebSockets:
- WebSockets allow for full-duplex communication, so it's very efficient to send and receive data. It requires no multiple connections, that would require handshakes.
- This technology reduces overhead and allows for high-speed data exchanges, which is especially interesting for Kubernetes-related tasks.
- WebSockets are protocol-independent so they may be used for communication over any full-duplex single-connection protocol, which will reduce latency and help in real-time command execution.
- WebSockets keep a persistent connection and thus use resources more efficiently and cut the number of TCP connections used, in turn reducing the amount of network overhead.
- The use of WebSockets fix issues that older protocols like SPDY had, and it's more compatible with modern web technologies.
The migration from SPDY to WebSockets in Kubernetes makes it much easier to communicate control messages, which will allow for better cluster management and stability.
Kep 4006
The SPDY protocol was deprecated in 2016. In 2023, a proposal was opened in the Kubernetes community to change the protocol for commands like port-forward, attach, exec, and portforward that use upgraded connections.
Implementation Overview:
WebSocket support in Kubernetes and kubectl starts with an HTTP handshake that upgrades the connection to a WebSocket, like as the SPDY. The WebSocket framing method provides improved technique for data framing for streams such as stdin, stdout, and stderr through framing with a channel identifier. This is an extension of the technique once used by SPDY through binary data framing.
As a fallback mechanism in case WebSocket support is not available on the server, a fallback to SPDY is utilized. The initial performance assessments demonstrated that WebSocket is as efficient as, or even a little better than, SPDY for standard kubectl operations, and continuing performance testing is intended to better performance across multiple use cases. There are many implications from this transition, including greater network compatibility in settings where strong proxy and firewall policies are in place, and better integration with external tools.
API server changes:
The API server now supports the WebSocket protocol by replacing SPDY with explicit headers to enable bidirectional streams. Additionally, the StreamTranslatorProxy
has been added to help create WebSocket connections from clients to internal SPDY connections in Kubernetes.
The management of WebSocket frames is critical, especially for the occasional use of the ping/pong frames to keep the WebSocket connection alive. This feature was first introduced in an alpha version that one could use through the feature gate for port forwarding Websockets within the API server and a corresponding flag in kubectl
.
Client (kubectl
) Changes:
On the client side, it added an upgrade to WebSocket for port forwarding, along with signing, response processing, and data serialization and deserialization over the WebSocket. This gives the user the ability to enable WebSocket port forwarding through environment variables. To make this transition easy, kubectl has also introduced mechanisms to handle errors and reliability, with error handling during its alpha and beta phases, along with a fallback to SPDY in case the WebSocket connection fails to ensure that service continuity is not lost.
Websockets in kubectl port-forward
with Kind
Setup Env
Before creating my environment, Docker, Kind, and Go have to be installed on the machine. All of these are needed for developing and deploying our Kubernetes cluster on our local machines. Let's set up the environment with Kubernetes 1.30.
Let's start by setting up the necessary environment variables and dependencies:
# Set Go environment variables
export GOPATH=$HOME/go
export GOROOT=/usr/local/go
export PATH=$PATH:$GOROOT/bin:$GOPATH/bin
# Create a directory for Kubernetes source
mkdir -p $GOPATH/src/k8s.io
cd $GOPATH/src/k8s.io
# Clone the Kubernetes repository
git clone https://github.com/kubernetes/kubernetes.git
cd kubernetes
Install Dependencies on Mac M1
For those uses Mac with an M1 chip:
brew install bash
brew install gnu-tar
Building the kubernetes and kubectl 1.30 image
To build the images and kubectl, we need to have Go 1.22 or later. Then, execute some commands:
# Checkout to the Kubernetes 1.30 release
git checkout release-1.30
# Build a node image using Kind
kind build node-image
# Build the kubectl in this version
make kubectl KUBE_BUILD_PLATFORMS=darwin/arm64
# kubectl artifact output path : $GOPATH/kubernetes/_output/dockerized/bin/darwin/arm64/kubectl
chmod +x $GOPATH/kubernetes/_output/dockerized/bin/darwin/arm64/kubectl
Kind Cluster Configs
First, set up the kind cluster with 1.30 images by creating a config.yaml
file with featureGates PortForwardWebsockets
enabled:
kind: Cluster
apiVersion: kind.x-k8s.io/v1alpha4
featureGates:
"PortForwardWebsockets": true
nodes:
- role: control-plane
image: kindest/node:latest
- role: worker
image: kindest/node:latest
- role: worker
image: kindest/node:latest
Create the cluster
When you have your configuration file set up you can add the cluster with the following command:
# Change default kubeconfig path to have only with kind test cluster
export KUBECONFIG=~/.kube/kind-kind
kind create cluster --name kind-1-30 --config config.yaml
Deploy a sample app
You can test your new WebSocket capability by deploying a simple application and using the cluster to connect by port forwarding:
git clone https://github.com/digitalocean/kubernetes-sample-apps.git
cd kubernetes-sample-apps
kubectl apply -k bookinfo-example/kustomize
Forward the Port Using WebSockets
Enable the WebSocket port forwarding :
export KUBECTL_PORT_FORWARD_WEBSOCKETS="true"
$GOPATH/kubernetes/_output/dockerized/bin/darwin/arm64/kubectl -v6 port-forward svc/productpage -n bookinfo 9080:9080
Now, open http://localhost:9080/
in your browser. You should get responses from the echoserver.
Verifying WebSocket Port Forwarding
To confirm if WebSocket port forwarding is active, review the kubectl logs:
This is the log with SDPY (default):
I0422 17:32:50.963059 11299 round_trippers.go:466] curl -v -XPOST -H "X-Stream-Protocol-Version: portforward.k8s.io" -H "User-Agent: kubectl/v1.31.0 (darwin/arm64) kubernetes/cae35db" 'https://192.168.68.109:6443/api/v1/namespaces/bookinfo/pods/productpage-v1-64c658687d-5bhsg/portforward'
I0422 17:32:50.965663 11299 round_trippers.go:510] HTTP Trace: Dial to tcp:192.168.68.109:6443 succeed
I0422 17:32:50.978422 11299 round_trippers.go:553] POST https://192.168.68.109:6443/api/v1/namespaces/bookinfo/pods/productpage-v1-64c658687d-5bhsg/portforward 101 Switching Protocols in 15 milliseconds
I0422 17:32:50.978440 11299 round_trippers.go:570] HTTP Statistics: DNSLookup 0 ms Dial 2 ms TLSHandshake 0 ms Duration 15 ms
I0422 17:32:50.978444 11299 round_trippers.go:577] Response Headers:
I0422 17:32:50.978448 11299 round_trippers.go:580] Upgrade: SPDY/3.1
I0422 17:32:50.978450 11299 round_trippers.go:580] X-Stream-Protocol-Version: portforward.k8s.io
I0422 17:32:50.978453 11299 round_trippers.go:580] Connection: Upgrade
This is the logs with Websocket enabled:
17:29:21.504179 7561 tunneling_dialer.go:75] Before WebSocket Upgrade Connection...
I0422 17:29:21.504194 7561 round_trippers.go:466] curl -v -XGET -H "Sec-Websocket-Protocol: SPDY/3.1+portforward.k8s.io" -H "User-Agent: kubectl/v1.31.0 (darwin/arm64) kubernetes/cae35db" 'https://192.168.68.109:6443/api/v1/namespaces/bookinfo/pods/productpage-v1-64c658687d-5bhsg/portforward'
I0422 17:29:21.506351 7561 round_trippers.go:510] HTTP Trace: Dial to tcp:192.168.68.109:6443 succeed
I0422 17:29:21.519936 7561 round_trippers.go:553] GET https://192.168.68.109:6443/api/v1/namespaces/bookinfo/pods/productpage-v1-64c658687d-5bhsg/portforward 101 Switching Protocols in 15 milliseconds
I0422 17:29:21.519954 7561 round_trippers.go:570] HTTP Statistics: DNSLookup 0 ms Dial 2 ms TLSHandshake 4 ms ServerProcessing 8 ms Duration 15 ms
I0422 17:29:21.519958 7561 round_trippers.go:577] Response Headers:
I0422 17:29:21.519962 7561 round_trippers.go:580] Sec-Websocket-Accept: +AqmlgtoGPP/Rlfw6oAZMCN34SY=
I0422 17:29:21.519964 7561 round_trippers.go:580] Sec-Websocket-Protocol: SPDY/3.1+portforward.k8s.io
I0422 17:29:21.519966 7561 round_trippers.go:580] Upgrade: websocket
I0422 17:29:21.519968 7561 round_trippers.go:580] Connection: Upgrade
I0422 17:29:21.519973 7561 tunneling_dialer.go:85] negotiated protocol: portforward.k8s.io
The logs show that the protocol upgrade to SPDY via WebSocket was successful in Kubernetes 1.30
KEP Roadmap Port Forward over Websockets
Alpha Release (Kubernetes 1.30):
WebSocket support added. Feature enabled via the KUBECTL_PORT_FORWARD_WEBSOCKETS=true
env var on the client side and PortForwardWebsockets
alpha feature flag should be enabled on the API Server,.
Focus in the alpha phase would be testing from all perspectives, user feedback, and updating developer and user documentation for this alpha feature.
Beta Release(Target 1.31):
More extensive testing for performance fine-tuning, user feedback, and updated documentation, the documentation will be updated regularly with the changes during the beta phase improvements.
User community feedback will be the basis in making the feature stable and usable.
Stable Release (Target 1.32):
Ensure WebSocket support transitions from beta to stable with alpha feature flags removed and publish a deprecation notice to officially mark the transition away from SPDY-based port forwarding.
KEP Roadmap RemoteCommand over websockets (exec, cp, attach)
Beta release (Kubernetes 1.30):
WebSocket for RemoteCommand
(exec, cp, attach) will be on by default. The feature gates to the client and server are expected to ensure that the SPDY replacement has as little impact on the user as possible.
The biggest emphasis here is on making the changes transparent to the user and, for those who are wary of this new implementation, will always have the choice to disable it.
Path to Stability:
Post 1.30 and leading up to the 1.32 release, the focus will be on refinement in WebSocket functionality, all aimed at reaching zero known critical bugs.
Conclusion
And thats all, folks. This hopefully explains the important protocol communication change in Kubernetes. I've had too many issues due to SPDY and lack of compatibility with new load balancers, proxies, etc., which was so complicated that I found myself at work. Now I see this migration from SPDY to WebSockets, so I'm a bit excited.
Looking forward, I'll let you know if there are any updates about this KEP
Also, maybe there'll be errors or misinformation in this post; I'm just human after all. Feel free to let me know if you find anything off on social media or post a comment here at the blog. Thanks!