GEP-1748: Gateway API Interaction with Multi-Cluster Services
- Issue: #1748
- Status: Experimental
Prolonged
This GEP will be in the “Experimental” channel for a prolonged period of time. This explores the interaction of Gateway API with the Multi-Cluster Services API. Until the MCS API is also GA, it will be impossible for this GEP to graduate beyond “Experimental”.
This GEP is also exempt from the Probationary Period rules as it predated them.
TLDR
The Kubernetes Multi-Cluster Services API enables Services to span multiple clusters. Gateway API enables advanced traffic routing, serving as the next generation Ingress, Load Balancing, and Mesh API. This document describes how these APIs can be used together.
Goals
- Define the interaction between Gateway API and Multi-Cluster Services
- Define any situations where Gateway API may span multiple clusters without the Multi-Cluster Services API
Non-Goals
- Make significant changes to either API
Introduction
Gateway API and the Multi-Cluster Services API represent two of the newest Kubernetes networking APIs. As they’ve been developed in parallel, there’s been some cross-project discussion about how they can interact, but that has never formally been written down. This GEP aims to change that.
Overview
Multi-Cluster Services can be used within Gateway API wherever Services can be used. The difference is that Services refer only to cluster-local endpoints while Multi-Cluster Services can refer to endpoints throughout an entire ClusterSet.
ServiceImport as a Backend
A Route can forward traffic to the endpoints attached to an imported Service. This behaves identically to how forwarding to Services work in Kubernetes, with the exception that the endpoints attached to a ServiceImport may span multiple clusters. For example, the following HTTPRoute would forward traffic to endpoints attached to the “store” ServiceImport:
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: store
spec:
parentRefs:
- name: external-http
rules:
- backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store
port: 8080
Routing to Specific Clusters
In some cases, it may be helpful to route certain paths to a specific cluster (or region). Similar to single-cluster Services, this can be accomplished by creating multiple Multi-Cluster Services, one for each subset of endpoints you would like to route to. For example, the following configuration will send requests with paths prefixed with “/west” to the store-west ServiceImport, and “/east” to the store-east ServiceImport. Requests that don’t match either of these paths will be routed to the “store” ServiceImport which represents a superset of the “store-west” and “store-east” ServiceImports.
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: store
spec:
parentRefs:
- name: external-http
rules:
- matches:
- path:
type: PathPrefix
value: /west
backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store-west
port: 8080
- matches:
- path:
type: PathPrefix
value: /east
backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store-east
port: 8080
- backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store
port: 8080
Advanced Routing With ServiceImports
All Routing capabilities in Gateway API should apply equally whether the backend is a Service or ServiceImport. For example, when routing to a system with multiple read replicas, it could be beneficial to route requests based on HTTP Method. In the following example, requests with POST, PUT, and DELETE methods are routed to api-primary
while the rest are routed to api-replicas
:
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: api
spec:
parentRefs:
- name: api-gw
rules:
- matches:
- method: POST
- method: PUT
- method: DELETE
backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: api-primary
port: 8080
- backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: api-replicas
port: 8080
Routing to Both Services and ServiceImports
There are some situations where it will be useful to split traffic between a Service and ServiceImport. In the following example, 90% of traffic would go to endpoints attached to the cluster-local “store” Service, and the remaining 10% would go to endpoints attached to the Multi-Cluster “store-global” Service:
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: store
spec:
parentRefs:
- name: external-http
rules:
- backendRefs:
- kind: Service
name: store
port: 8080
weight: 90
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store-global
port: 8080
weight: 10
Cross-Namespace References with ReferenceGrant
It is possible to use ReferenceGrant to enable cross-namespace references to ServiceImports. For example, the following HTTPRoute would forward traffic to endpoints attached to the “bar” Multi-Cluster Service in the “bar” namespace:
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: foo
namespace: foo
spec:
rules:
- matches:
- path:
type: PathPrefix
value: /bar
backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: bar
namespace: bar
---
kind: ReferenceGrant
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: bar
namespace: bar
spec:
from:
- group: gateway.networking.k8s.io
kind: HTTPRoute
namespace: foo
to:
- group: multicluster.x-k8s.io
kind: ServiceImport
Mesh: ServiceImport as Parent
In some cases, you may want to override traffic destined for a Multi-Cluster Service with a mesh. As part of the broader GAMMA initiative, ServiceImport can be used in the same way that Service is used as a ParentRef. When a Service is specified as a parent, meshes will intercept traffic destined for the ClusterIP of the Service and apply any policies or routing decisions defined by the Route. Similarly, when a ServiceImport is specified as a parent, meshes will intercept traffic destined for the ClusterSetIP and apply any policies or routing decisions defined by the Route. In the following example, the mesh would intercept traffic destined for the store ClusterSetIP matching the /cart
path and redirect it to the cart
Multi-Cluster Service.
kind: HTTPRoute
apiVersion: gateway.networking.k8s.io/v1beta1
metadata:
name: store
spec:
parentRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: store
rules:
- matches:
- path:
value: "/cart"
backendRefs:
- group: multicluster.x-k8s.io
kind: ServiceImport
name: cart
Services vs ServiceImports
It is important that all implementations provide a consistent experience. That means that references to Services SHOULD always be interpreted as references to endpoints within the same cluster for that Service. References to ServiceImports MUST be interpreted as routing to Multi-Cluster endpoints across the ClusterSet for the given ServiceImport. In practice, that means that users should use “Service” when they want to reference cluster-local endpoints, and “ServiceImport” when they want to route to Multi-Cluster endpoints across the ClusterSet for the given ServiceImport. This behavior should be analogous to using .cluster.local
versus .clusterset.local
DNS for a given Service.
API Changes
- ServiceImport is recognized as backend with “Extended” conformance
- ServiceImport is included in GAMMA GEP(s) with “Extended” conformance
- Clarification that Services refer to endpoints within the same cluster
Alternatives
Develop Custom Multi-Cluster Concepts Independently from Multi-Cluster Services
We could theoretically develop an entirely new way to handle multi-cluster routing. We’re choosing not to do that because the existing APIs are sound and can work well together.