This repository implements a small launcher that prepares a mesh environment and starts the user application.
In K8S, the mesh implementation relies on a mutating webhook that patches the Pod, injecting for required environment. Docker and CloudRun images do not have an injector - this application is playing the same role, using the K8S and GCP APIs to setup iptables and the sidecar process or the proxyless bootstrap.
This supports running an app:
- in an Istio-like environment with a Sidecar and iptables interception
- if iptables is not available (gVisor, regular docker, dev machine), configure 'whitebox' mode ( HTTP_PROXY and localhost port forwarding for TCP)
- in proxyless gRPC mode for applications that natively support XDS and mesh, without iptables or sidecar.
The launcher is responsible for:
- discovering a GKE/K8S cluster based on environment (metadata server, env variables), and getting credentials and config
- discovering the XDS address and config (root certificates, metadata)
- setting up iptables ( equivalent to the init container in K8S ), using pilot-agent
- launching pilot-agent and envoy
- configuring pilot-agent to intercept DNS
- launching the application - after the setup is ready
- creating any necessary tunnels to allow use of mTLS, based on the HBONE (tunneling over HTTP/2) proposal in Istio.
The repository also includes a specialised SNI-routing gateway that allows any mesh node using mTLS to route back to CloudRun with the proper authentication and tunneling for mTLS.
The user application should be able to communicate with other mesh workloads - in Pods, VMs or other CloudRun services using mTLS and the mesh launcher.
The code is based on the Istio VM startup script and injection template and will need to be kept in sync with future changes in the mesh startup.
Common environment variables used in this document:
export PROJECT_ID=wlhe-cr
export CLUSTER_LOCATION=us-central1-c
export CLUSTER_NAME=asm-cr
# CloudRun region
export REGION=us-central1
export WORKLOAD_NAMESPACE=fortio # Namespace where the CloudRun service will 'attach'
export WORKLOAD_NAME=cloudrun
# Name of the service account running the CloudRun service. It is recommended to use a dedicated SA for each K8S namespace
# and keep permissions as small as possible.
# By default the namespace is extracted from the GSA name - if using a different SA or naming, WORKLOAD_NAMESPACE env
# is required when deploying the docker image.
# (This may change as we polish the UX)
export CLOUDRUN_SERVICE_ACCOUNT=k8s-${WORKLOAD_NAMESPACE}@${PROJECT_ID}.iam.gserviceaccount.com
# Name for the cloudrun service - will use the same as the workload.
# Note that the service must be unique for region, if you want the same name in multiple namespace you must
# use explicit config for WORKLOAD_NAME when deploying and unique cloudrun service name
export CLOUDRUN_SERVICE=${WORKLOAD_NAME}
Requirements:
- For each region, you need a Serverless connector, using the same network as the GKE cluster(s) and VMs. CloudRun will use it to communicate with the Pods/VMs.
- 'gen2' VM required for iptables. 'gen1' works in 'whitebox mode', using HTTP_PROXY.
- The project should be allowed by policy to use 'allow-unauthenticated'. WIP to eliminate this limitation.
You need to have gcloud and kubectl, and admin permissions for the project and cluster.
After installation, new services can be configured for namespaces using only namespace-level permissions in K8S.
-
If you don't already have a cluster with managed ASM, follow Install docs. Also supported is [in-cluster ASM] (https://cloud.google.com/service-mesh/docs/scripted-install/gke-install)
-
Configure the in-cluster 'mesh connector' gateway and permissions. (this step is temporary, WIP to upstream it to Istiod and the East-West gateway. Versions of Istio after upstreaming will not need this step.)
kubectl apply -k github.com/costinm/krun/manifests/
For each region where GKE and CloudRun will be used, install CloudRun connector Using the UI is usually easier - it does require a /28 range to be specified. You can call the connector 'serverlesscon'
- the name will be used when deploying the CloudRun service.
If you already have a connector, you can continue to use it, and adjust the '--vpc-connector' parameter on the deploy command.
The connector MUST be on the same VPC network with the GKE cluster and configured with a not-in-use CIDR range.
The Google Service Account running the CloudRun service will be mapped to a K8S namespace.
The service account used by CloudRun must be granted access to the GKE APIserver with minimal permissions, and must be allowed to get K8S tokens.
This steps can be run by a user or service account with namespace permissions in K8S - does not require k8s cluster admin. It does require IAM permissions on the project running the CloudRun service.
-
Create a google service account for the CloudRun app (recommended - one per namespace, to reduce permission scope).
-
Grant '--role="roles/container.clusterViewer"' to the service account.
-
Grant RBAC permissions to the google service account, allowing it to access in-namespace config map and use TokenReview for the default KSA. (this step is also temporary, WIP to make it optional). This is used to get the MeshCA certificate and communicate with the managed control plane - Istio injector is mounting the equivalent tokens.
gcloud --project ${PROJECT_ID} iam service-accounts create k8s-${WORKLOAD_NAMESPACE} \
--display-name "Service account with access to ${WORKLOAD_NAMESPACE} k8s namespace"
# Allow the GSA to access GKE clusters, as viewer
gcloud --project ${PROJECT_ID} projects add-iam-policy-binding \
${PROJECT_ID} \
--member="serviceAccount:${CLOUDRUN_SERVICE_ACCOUNT}" \
--role="roles/container.clusterViewer"
# Make sure we use the current config cluster
gcloud container clusters get-credentials ${CLUSTER_NAME} --zone ${CLUSTER_LOCATION} --project ${PROJECT_ID}
# Make sure the namespace is created
kubectl create ns ${WORKLOAD_NAMESPACE}
# Uses WORKLOAD_NAMESPACE and PROJECT_ID to associate the Google Service Account with the K8S Namespace.
curl https://raw.githubusercontent.com/costinm/krun/main/manifests/google-service-account-template.yaml | envsubst | kubectl apply -f -
Mesh-enabled images include the sidecar (envoy), launcher and the actual application. The easiest way to create one is using a 'golden image', that is used as base.
samples/fortio/Dockerfile contains an example Dockerfile - you can also use the pre-build image
gcr.io/wlhe-cr/fortio-mesh:main
To build your own image:
# Base image. You can create a custom 'golden' base, starting with ASM proxy image and adding the
# startup helper (krun) and other files or configs you need. This doc uses the image
# used for the CI/CD testing.
export GOLDEN_IMAGE=gcr.io/wlhe-cr/krun:main
# Target image
export IMAGE=gcr.io/${PROJECT_ID}/fortio-mesh:main
(cd samples/fortio && docker build . -t ${IMAGE} --build-arg=BASE=${GOLDEN_IMAGE} )
docker push ${IMAGE}
Deploy the service, with explicit configuration:
# If you didn't build a custom docker image, you can use the prebuild image:
# IMAGE=gcr.io/wlhe-cr/fortio-mesh:main
gcloud alpha run deploy ${CLOUDRUN_SERVICE} \
--platform managed \
--project ${PROJECT_ID} \
--region ${REGION} \
--execution-environment=gen2 \
--service-account=k8s-${WORKLOAD_NAMESPACE}@${PROJECT_ID}.iam.gserviceaccount.com \
--allow-unauthenticated \
--use-http2 \
--port 15009 \
--image ${IMAGE} \
--vpc-connector projects/${PROJECT_ID}/locations/${REGION}/connectors/serverlesscon \
--set-env-vars="CLUSTER_NAME=${CLUSTER_NAME}" \
--set-env-vars="CLUSTER_LOCATION=${CLUSTER_LOCATION}"
For versions of 'gcloud' older than 353.0, replace --execution-environment=gen2
with --sandbox=minivm
CLUSTER_NAME and CLUSTER_LOCATION will be optional - krun will pick a config cluster in the same region that is setup with MCP, and fallback to other config cluster if the local cluster is unavailable. Cluster names starting with 'istio' will be used first in a region. (Will likely change to use a dedicated label on the project - WIP)
gcloud run deploy SERVICE --platform=managed --project --region
is common required parameters--execution-environment=gen2
is currently required to have iptables enabled. Without it the 'whitebox' mode will be used (still WIP)--service-account
is recommended for 'minimal priviledge'. The service account will act as a K8S SA, and have its RBAC permissions--allow-unauthenticated
is only needed temporarily if you want to ssh into the instance for debug. WIP to fix this.--use-http2
and--port 15009
are required
For workloads in k8s to communicate with the CloudRun service we need to create few Istio configurations.
This step will be replaced by auto-registration (WIP):
export SNI_GATE_IP=$(kubectl -n istio-system get service internal-hgate -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
export K_SERVICE=$(gcloud run services describe ${CLOUDRUN_SERVICE} --region ${REGION} --project ${PROJECT_ID} --format="value(status.address.url)" | sed s,https://,, | sed s/.a.run.app// )
curl https://raw.githubusercontent.com/costinm/krun/main/manifests/sni-service-template.yaml | SNI_GATE_IP=${SNI_GATE_IP} K_SERVICE=${K_SERVICE} envsubst | kubectl apply -f -
# Or: cat ../../manifests/sni-service-template.yaml | SNI_GATE_IP=${SNI_GATE_IP} K_SERVICE=${K_SERVICE} envsubst | kubectl apply -f -
This will create a K8S Service with the same base name as the cloudrun service (for example: fortio-cr-icq63pqnqq-uc). You can use VirtualService or K8S Gateway API to aggregate routing and use custom names.
- Deploy an in-cluster application. The CloudRun service will connect to it:
gcloud container clusters get-credentials ${CLUSTER_NAME} --zone ${CLUSTER_LOCATION} --project ${PROJECT_ID}
kubectl create ns fortio
kubectl label namespace fortio istio.io/rev=asm-managed
kubectl -n fortio apply -f https://raw.githubusercontent.com/costinm/krun/main/samples/fortio/in-cluster.yaml
- Use the CloudRun service to connect to the in-cluster workload. Use the CR service URL with /fortio/ path to access the UI of the app.
In the UI, use "http://fortio.fortio.svc:8080" and you should see the results for testing the connection to the in-cluster app.
In general, the CloudRun applications can use any K8S service name - including shorter version for same-namespace services. So fortio, fortio.fortio, fortio.fortio.svc.cluster.local also work.
In this example the in-cluster application is using ASM - it is also possible to access regular K8S applications without a sidecar.
- To verify calls from K8S to CloudRun, you can use 'kubectl exec' to the fortio pod, with a command like
curl http://${KSERVICE}.fortio.svc:8080/fortio/
or
fortio load http://${KSERVICE}.fortio.svc:8080/echo
The cloudrun service is mapped to a K8S service with the same name - this can be used as a destination in Gateway or VirtualService using a different name or load balancing multiple CloudRun regions.
apiVersion: v1
kind: Service
metadata:
name: myservice
namespace: fortio
spec:
ports:
- port: 8080
name: http
targetPort: 15443
---
apiVersion: networking.istio.io/v1alpha3
kind: VirtualService
metadata:
name: myservice
namespace: fortio
spec:
hosts:
- myservice
- myservice.fortio.svc
http:
- route:
# Use the actual CR service.
- destination:
host: cloudrun-6c3hzwsnla-uc
weight: 25
- destination:
host: cloudrun2-6c3hzwsnla-uc
weight: 75
When running in CloudRun, default automatic configuration is based on environment variables, metadata server and calls to GKE APIs. For debugging (as a regular process), when running with a regular docker or to override the defaults, the settings must be explicit.
- WORKLOAD_NAMESPACE - default value extracted from the service account running the CloudRun service
- WORKLOAD_NAME - default value is the CloudRun service name. Also used as 'canonical service'.
- PROJECT_ID - default is same project as the CloudRun service.
- CLUSTER_LOCATION - default is same region as the CloudRun service. If CLUSTER_NAME is not specified, a cluster with ASM in the region or zone will be picked.
- CLUSTER_NAME - if not set, clusters in same region or a zone in the region will be picked. Cluster names starting with 'istio' are currently picked first. (WIP to define a labeling or other API for cluster selection)
Also for local development:
- GOOGLE_APPLICATION_CREDENTIALS must be set to a file that is mounted, containing GSA credentials.
- Alternatively, a KUBECONFIG file must be set and configured for the intended cluster.
The setup is similar with Istio on VM support.
The 'mesh connector' is an extended version of the "East-West" Gateway, provide access to the in-cluster Istiod and handles forwarding the requests from pods to CloudRun.
The 'golden image' includes the istio sidecar components and a launcher, similar with Istio on VMs. The launcher will load the istio-system/mesh-env config map, created by the connector (until it is moved to Istiod proper), set iptables interception, generate the mesh certificates and start envoy - using pilot-agent, and finally run the application.
The application works the same as pods or VMs with Istio support - outgoing requests are intercepted, sent to Envoy where mTLS and policies are handled.
For requests from Pods to CloudRun, the WorkloadGroup and other configs created by
For install, few useful commands:
kubectl -n istio-system get svc hgate -o yaml
kubectl -n istio-system get svc internal-hgate -o yaml
# Status should show the external and internal IPs of the mesh connector.
kubectl -n istio-system get cm mesh-env -o yaml
# Should include 'MESH_TENANT' if using managed ASM
# Should include root cert, external and internal IP address of the 'mesh connector'
Since CloudRun and docker doesn't support kubectl exec or port-forward, we include a minimal sshd server that is enabled using a K8S Secret or environment variables. There are 2 ways to enable ssh for debugging:
- create a Secret name sshdebug with the SSH private key and authorized_keys. See samples/ssh for setup example.
- add the SSH_AUTH env variable containing the authorized public
key.
--set-env-vars="SSH_AUTH=$(cat ~/.ssh/id_ecdsa.pub)"
You can ssh into the service and forward ports using a regular ssh client and a ProxyCommand that implements tunneling over HTTP/2:
# Compile the proxy command
go install ./cmd/hbone
(cd samples/ssh; WORKLOAD_NAMESPACE=fortio make secret)
# Re-deploy the cloudrun service - or wait for it to scale to zero. The ssh is enabled on startup.
# Set with your own service URL
export SERVICE_URL=$(gcloud run services describe ${CLOUDRUN_SERVICE} --project ${PROJECT_ID} --region ${REGION} --format="value(status.address.url)")
ssh -o ProxyCommand='hbone ${SERVICE_URL}:443/_hbone/22' root@${CLOUDRUN_SERVICE}
# or use "-F /dev/null" to disable any configured settings that may interfere, and
# turn off SSH host checking since the tunnel is checking the TLS cert of the service:
ssh -F /dev/null -o StrictHostKeyChecking=no -o "UserKnownHostsFile /dev/null" \
-o ProxyCommand='hbone ${SERVICE_URL}:443/_hbone/22' root@${CLOUDRUN_SERVICE}