Page last updated:
This topic describes how the Gorouter, the main component in the Cloud Foundry routing tier, routes HTTP traffic within Cloud Foundry (CF).
For more information, see the Routing section of the CF Components topic.
The Gorouter has a limit of 1 MB for HTTP Headers.
X-Forwarded-Proto header gives the scheme of the HTTP request from the client.
If an incoming request includes the
X-Forwarded-Proto header, the Gorouter:
- Appends it to the existing header
- Sets the scheme to HTTP if the client made an insecure request, meaning a request on port 80
- Sets the scheme to HTTPS if the client made a secure request, meaning a request on port 443
Developers can configure their apps to reject insecure requests by inspecting the
X-Forwarded-Proto HTTP header on incoming traffic. The header may have multiple values represented as a comma-separated list, so developers must ensure the app rejects traffic that includes any
X-Forwarded-Proto values that are not HTTPS.
X-Forwarded-For is present, the Gorouter appends the load balancer’s IP address to it and forwards the list. If
X-Forwarded-For is not present, then the Gorouter sets it to the IP address of the load balancer in the forwarded request (some load balancers masquerade the client IP). If a load balancer sends the client IP using the PROXY protocol, then the Gorouter uses the client IP address to set
If your load balancer terminates TLS on the client side of the Gorouter, it must append these headers to requests forwarded to the Gorouter. For more information, see Securing Traffic into Cloud Foundry.
Zipkin is a tracing system that enables app developers to troubleshoot failures or latency issues. Zipkin provides the ability to trace requests and responses across distributed systems. For more information, see Zipkin.io.
When Zipkin tracing is enabled in Cloud Foundry, the Gorouter examines the HTTP request headers and performs:
- If the
X-B3-SpanIdHTTP headers are not present in the request, the Gorouter generates values for these and inserts the headers into the request forwarded to an app. These values are also found in the Gorouter access log message for the request:
- If the
X-B3-SpanIdHTTP headers are present in the request, the Gorouter forwards them unmodified. In addition to these trace and span IDs, the Gorouter access log message for the request includes
Developers can then add Zipkin trace IDs to their app logging in order to trace app requests and responses in CF.
After adding Zipkin HTTP headers to app logs, developers can use
cf logs myapp to correlate the trace and span IDs logged by the Gorouter with the trace ids logged by their app. To correlate trace IDs for a request through multiple apps, each app must forward appropriate values for the headers with requests to other apps.
For more information about Zipkin tracing, see Enabling Zipkin Tracing.
Developers who want to obtain debug data for a specific instance of an app can use the HTTP header
X-Cf-App-Instance to make a request to an app instance.
To make an HTTP request to a specific app instance:
Obtain the GUID of your app:
$ cf app YOUR-APP --guid
List your app instances and retrieve the index number of the instance you want to debug:
$ cf app YOUR-APP
Make a request to the app route using the HTTP header
X-Cf-App-Instanceset to the concatenated values of the app GUID and the instance index:
$ curl app.example.com -H "X-Cf-App-Instance":"YOUR-APP-GUID:YOUR-INSTANCE-INDEX"
Use of theIf this header is set to an invalid value, Gorouter resturns a
X-Cf-App-Instanceheader is only available for users on the Diego architecture.
400status code and the response from Gorouter contains a
X-Cf-Routererrorheader with more information about the error. The following table describes the possible error responses:
X-Cf-Routererror Value Reason for Error Response Body
The value provided for
X-Cf-App-Instancewas not a properly formatted GUID.
The value provided for
X-Cf-App-Instanceis a correctly formatted GUID, but there is no instance found with that guid for the route requested.
400 Bad Request: Requested instance ('1') with guid ('aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa') does not exist for route ('myroute.cf.com')
Apps that require mutual TLS (mTLS) need metadata from client certificates to authorize requests. CF supports this use case without bypassing layer-7 load balancers and the Gorouter.
The HTTP header
X-Forwarded-Client-Cert (XFCC) may be used to pass the originating client certificate along the data path to the application. Each component in the data path must trust that the back end component has not allowed the header to be tampered with.
If you configure the load balancer to terminate TLS and set the XFCC header from the received client certificate, you must also configure the load balancer to strip this header if it is present in client requests. This configuration is required to prevent spoofing of the client certificate.
The following sections describe supported deployment configurations.
By default, Cloud Foundry forwards arbitrary headers that are not otherwise mentioned in the documentation. You can configure a load balancer to put the certificate of the originating client, received during the mutual TLS handshake, into an HTTP header that the load balancer forwards upstream. Cloud Foundry recommends the header XFCC for this use case, since this header is used in other configuration modes described below. The value of the header should be the base64-encoded bytes of the certificate, which is equivalent to a PEM file with newlines, headers, and footers removed.
This mode is enabled when
router.forwarded\_client\_cert is set to
Alternatively, the operator can configure the Gorouter to forward the XFCC header set by the load balancer only when the connection with the load balancer is mutual TLS. The client certificate received by the Gorouter in the mutual TLS handshake is not forwarded in the header.
This mode is enabled when
router.forwarded\_client\_cert is set to
If the Gorouter is the first component to terminate TLS, such that it receives the certificate of the originating client in the mutual TLS handshake, the operator should configure the Gorouter to strip any instances of the XFCC header from client requests, set the value of the header to the base64-encoded bytes of the client certificate received in the mutual handshake, and forward the header upstream to the application.
This mode is enabled when
If TLS is terminated for the first time at Gorouter, the Gorouter must be configured to trust the root certificate authority used to sign the Diego intermediate certificate authority, which in turn is used to sign certificates generated for each Diego container. This trust enables mutual authentication between applications that are running on Cloud Foundry. Operators should configure the
router.ca_certs property for the Gorouter job with the root certificate authority in their BOSH deployment manifest.
jobs: properties: router: ca_certs: -----BEGIN CERTIFICATE----- (contents of certificate) -----END CERTIFICATE-----
router.ca_certs property is a string of concatenated certificate authorities in PEM format.
Depending on your needs, you can configure your deployment to terminate TLS at the Gorouter, at the Gorouter and the load balancer, or at the load balancer only. For more information, see Securing Traffic into Cloud Foundry.
The Gorouter supports TLS and mutual authentication to back end destinations, including app instances, platform services, and any other routable endpoints.
This has the following benefits:
- Improved availability for apps by keeping routes in the Gorouter’s routing table when TTL expires
- Increased guarantees against misrouting by validating the identity of back ends before forwarding requests
- Increased security by encrypting data in flight from the Gorouter to back ends
This feature is enabled by default in
cf-deployment, which configures the following properties:
router.ca_certs: must include the CA certificate used to sign any back end to which the Gorouter initiates a TLS handshake.
The Gorouter does not automatically initiate TLS handshakes with back end destinations. To register that the Gorouter must initiate a TLS handshake before forwarding HTTP requests to a component, the component must include the following optional fields in its
route.register message to NATS:
"tls_port": The port to which the Gorouter should open a connection and initiate a TLS handshake.
"server_cert_domain_san": An identifier for the back end which the Gorouter expects to find in the Domain Subject Alternative Name of the certificate presented by the back end. This is used to prevent misrouting for back ends whose IPs or ports are expected to change. For more information, see Consistency.
Additional configuration included on the
rep job on the diego-cell in
cf-deployment enables TLS from the Gorouter to app instances.
Authors of routable components can configure Route Registrar to send the necessary NATS message automatically. For more information, see the route-registrar repository on GitHub.
To enable mutual authentication between the Gorouter and back ends, operators configure the Gorouter with a certificate and private key using the following manifest properties:
The Gorouter presents the certificate if requested by the back end during the TLS handshake.
As CF manages and balances apps, the internal IP address and ports for app instances change. To keep the Gorouter’s routing tables current, a Route Emitter on each Diego cell sends periodic messages for each app instance running on that cell to all Gorouters through NATS. Each message includes the location and a unique identifier for the app instance to verify its identity when using TLS to communicate with the instance.
Network partitions or NATS failures can cause the Gorouter’s routing table to fall out of sync, as CF continues to re-create containers across hosts to keep apps running. This can lead to routing of requests to incorrect destinations.
Before forwarding traffic to an app instance, the Gorouter initiates a TLS handshake with an Envoy proxy running in each app container. In the TLS handshake, the Envoy proxy presents a certificate generated by Diego for each container which uniquely identifies the container using the same app instance identifier sent by the Route-Emitter, configured in the certificate as a domain Subject Alternative Name (SAN). For more information, see Envoyproxy.io.
If the Gorouter confirms that the app instance identifier in the certificate matches the one received in the route registration message, the Gorouter forwards the HTTP request over the TLS session, and the Envoy proxy then forwards it to the app process. If the instance identifiers do not match, the Gorouter removes the app instance from its routing table and transparently retries another instance of the app.
While it is possible to disable TLS application identity verification, this consistency mode is no longer recommended.
In this consistency mode, the Diego Route-Emitters on each cell send route registration messages that include instructions for the Gorouter to send unencrypted requests to the app instance. If the Gorouter does not receive an update for the route within the time-to-live (TTL) interval, the route is pruned from the Gorouter’s routing table. For more information, see TLS to Apps and Other Back End Services.
The Gorouter can validate app instance identity using TLS only when Diego Cells are configured appropriately. Because Diego Cells are configured for TLS through the instance group that they belong to, the Gorouter can run in different consistency modes with Diego Cells in different instance groups. For example, the Gorouter can communicate over TLS and validate the Diego Cells in one Isolation Segment, while communicating with Diego Cells in another Isolation Segment over plain text and without validating instance identity.
Currently, only Linux cells support the Gorouter validating app instance identities using TLS by default. With Windows cells, the Gorouter connects to backends without TLS, forwarding requests to Windows apps over plain text and pruning based on route TTL.
This experimental opsfile can be used to enable the feature on windows:
The Gorouter can be configured to use different load balancing algorithms for routing incoming requests to app instances. The Gorouter maintains a dynamically updated list of app instances for each route. Depending on which algorithm is selected, it forwards to one of the app instances.
To configure the behavior, you can change the value of
router.balancing_algorithm manifest property. The available options are:
By default, the Gorouter uses the round-robin algorithm.
Incoming requests for a given route are forwarded to all app instances one after another, looping back to the first one after they have each received a request. This algorithm is suitable for most use cases and evenly distributes the load between app instances.
Each request for a given route is forwarded to the app instance with the least number of open connections. This algorithm can be more suitable for some cases. For example, if app instances have long-lived connections and are scaled up, then new instances receive fewer connections, causing a disproportionate load. In this case, choosing a least-connection algorithm sends new connections to new instances to equalize the load.
WebSockets is a protocol providing bi-directional communication over a single, long-lived TCP connection, commonly implemented by web clients and servers. WebSockets are initiated through HTTP as an upgrade request. The Gorouter supports this upgrade handshake, and holds the TCP connection open with the selected app instance. To support WebSockets, the operator must configure the load balancer correctly. Depending on the configuration, clients may have to use a different port for WebSocket connections, such as port 4443, or a different domain name. For more information, see Supporting WebSockets.
The Gorouter supports session affinity, or sticky sessions, for incoming HTTP requests to compatible apps.
With sticky sessions, when multiple instances of an app are running on CF, requests from a particular client always reach the same app instance. This allows apps to store session data specific to a user session.
To support sticky sessions, configure your app to return a sticky session cookie in responses.
The default value for this field is
You can configure the cookie names that the routing tier uses for sticky sessions.
To configure the names of the cookies, edit the
router.sticky_session_cookie_names config key in your manifest.
If an app returns a sticky session cookie to a client request, the CF routing tier generates a unique
VCAP_ID for the
app instance based on its GUID with expiry same as that of
JSESSIONID in the following format:
On subsequent requests, the client must provide both the sticky session and
The CF routing tier uses the
VCAP_ID cookie to forward client requests to the same app instance every time. The sticky session cookie is forwarded to the app instance to enable session continuity. If the app instance identified by the
VCAP_ID crashes, the Gorouter attempts to route the request to a different instance of the app. If the Gorouter finds a healthy instance of the app, it initiates a new sticky session.
For more information, see Session Affinity in GitHub.
Note: CF does not persist or replicate HTTP session data across app instances. If an app instance crashes or is stopped, session data for that instance is lost. If you require session data to persist across crashed or stopped instances, or to be shared by all instances of an app, store session data in a CF marketplace service that offers data persistence.
The Gorouter supports keep-alive connections from clients and does not close the TCP connection with clients immediately after returning an HTTP response. Clients are responsible for closing these connections.
If keep-alive connections are disabled, the Gorouter closes the TCP connection with an app instance or system component after receiving an HTTP response.
If keep-alive connections are enabled, the Gorouter maintains established TCP connections to back ends. The Gorouter supports up to 10 idle connections to each back end:
- If an idle connection exists for a given back end, the Gorouter reuses it to route subsequent requests.
- If no idle connection exists for this back end, the Gorouter creates a new connection.
For more information, see Gorouter Back End Keep-Alive Connections.
If the Gorouter cannot establish a TCP connection with a selected app instance, the Gorouter considers the instance ineligible for requests for 30 seconds and transparently attempts to connect to another app instance. Once the Gorouter has established a TCP connection with an app instance, the Gorouter forwards the HTTP request.
When you deploy an app that requires Diego Cells to restart or recreate, the app may not respond to a Gorouter request before the keep-alive connection breaks. The following table describes how the Gorouter behaves if it cannot establish a TCP connection to an app:
|If the Gorouter…||and the back end…||then the Gorouter…|
|cannot establish a TCP connection to a back end||N/A||retries another back end, no more than 3 times|
|establishes a TCP connection to a back end and forwards the request||does not respond||waits 15 minutes for a response, and if it errors, does not retry another back end|
|establishes a TCP connection to a back end and forwards the request||returns a TCP connection error||returns an error to the client, marks backend ineligible, and does not retry another back end|
In all cases, if the app still does not respond to the request, the Gorouter returns a
502 error. For more information, see Troubleshooting Router Error Responses.