Connect to External Kafka Cluster
Pivotal Cloud Foundry does not have Apache Kafka as a managed service in the Marketplace. However, it is common for developers to develop and deploy applications that interact with an external Kafka cluster to Cloud Foundry. This recipe specifically walks through the developer expectations from Spring Cloud Stream, Spring Cloud Data Flow, and the Spring Cloud Data Flow for PCF Tile.
We review the required Spring Cloud Stream properties and how they are translated over to the applications for the following deployment options in Cloud Foundry.
- Applications run as standalone app instances.
- Applications deployed as part of a streaming data pipeline through open-source SCDF.
- Applications deployed as part of a streaming data pipeline through SCDF for PCF tile.
Prerequisite
We start with preparing the external Kafka cluster credentials.
Typically, a series of Kafka brokers is collectively referred to as a Kafka cluster. Each brokers can be reached individually through its external IP address or through a well-defined DNS route (if one has been made available).
For this walk-through, we stick to a simpler setup of a three-broker cluster with their DNS addresses being foo0.broker.foo
, foo1.broker.foo
, and foo2.broker.foo
. The default port of each broker is 9092
.
If the cluster is secured, depending on the security option in use at the broker, different properties are expected to be supplied when applications attempt to connect to the external cluster.
Again, for simplicity, we use Kafka's JAAS set up of PlainLoginModule
with a username of test
and a password of bestest
.
User-provided Services versus Spring Boot Properties
The next question Cloud Foundry developers stumble upon is whether or not to set up Kafka connection as a Cloud Foundry custom user-provided service (CUPS) or simply pass connection credentials as Spring Boot properties.
Cloud Foundry has no Spring Cloud Connector or CF-JavaEnv support for Kafka, so, by service-binding the Kafka CUPS with the application, you can not automatically parse VCAP_SERVICES
and pass the connection credentials to the applications at runtime.
Even with CUPS in place, it is your responsibility to parse the VCAP_SERVICES
JSON and pass them as Boot properties, so no automation is in place for Kafka.
For the curious, you can see an example of CUPS in action in the Spring Cloud Data Flow's Reference Guide.
For this walk-through, we stick to the Spring Boot properties.
Standalone Streaming Apps
The typical Cloud Foundry deployment of an application includes a manifest.yml
file.
We use the source-sample
source application to highlight the configuration needed to connect to an external Kafka cluster:
---
applications:
- name: source-sample
host: source-sample
memory: 1G
disk_quota: 1G
instances: 1
path: source-sample.jar
env:
... # other application properties
... # other application properties
SPRING_APPLICATION_JSON: |-
{
"spring.cloud.stream.kafka.binder": {
"brokers": "foo0.broker.foo,foo1.broker.foo,foo2.broker.foo",
"jaas.options": {
"username": "test",
"password":"bestest"
},
"jaas.loginModule":"org.apache.kafka.common.security.plain.PlainLoginModule"
},
"spring.cloud.stream.bindings.output.destination":"fooTopic"
}
With these settings, when the source-sample
source is deployed to Cloud Foundry, it should be able to connect to the external cluster.
You can verify the connection credentials by accessing the /configprops
of the source-sample
actuator endpoint.
Likewise, you can also see the connection credentials printed in the app logs.
The Kafka connection credentials are supplied through the Spring Cloud Stream Kafka binder properties, which, in
this case are all the properties with the spring.spring.cloud.stream.kafka.binder.*
prefix.
Alternatively, instead of supplying the properties through SPRING_APPLICATION_JSON
, these properties can be supplied as plain environment variables as well.
Streaming Data Pipeline in SCDF (Open Source)
Deploying a streaming data pipeline in SCDF requires at least two applications. We use the out-of-the-box time
application as the source and the log
application as the sink here.
Global Kafka Connection Configurations
Before we jump to the demo walk-through, we review how you can configure global properties centrally in SCDF. With that flexibility, every stream application deployed through SCDF also automatically inherits all the globally defined properties, and it can be convenient for cases like Kafka connection credentials. The following listing shows the global properties:
---
applications:
- name: scdf-server
host: scdf-server
memory: 2G
disk_quota: 2G
timeout: 180
instances: 1
path: spring-cloud-dataflow-server-2.11.4.jar
env:
SPRING_PROFILES_ACTIVE: cloud
JBP_CONFIG_SPRING_AUTO_RECONFIGURATION: '{enabled: false}'
SPRING_CLOUD_SKIPPER_CLIENT_SERVER_URI: http://your-skipper-server-uri/api
SPRING_APPLICATION_JSON: |-
{
"spring.cloud": {
"dataflow.task.platform.cloudfoundry": {
"accounts": {
"foo": {
"connection": {
"url": <api-url>,
"org": <org>,
"space": <space>,
"domain": <app-domain>,
"username": <email>,
"password": <password>,
"skipSslValidation": true
},
"deployment": {
"services": <comma delimited list of service>"
}
}
}
},
"stream": {
"kafka.binder": {
"brokers": "foo0.broker.foo,foo1.broker.foo,foo2.broker.foo",
"jaas": {
"options": {
"username": "test",
"password":"bestest"
},
"loginModule":"org.apache.kafka.common.security.plain.PlainLoginModule"
}
},
"bindings.output.destination":"fooTopic"
}
}
}
services:
- mysql
With the preceding manifest.yml
file, SCDF should now automatically propagate the Kafka connection credentials to all the stream application deployments.
Now you can create the streams:
dataflow:>stream create fooz --definition "time | log"
Created new stream 'fooz'
dataflow:>stream deploy --name fooz
Deployment request has been sent for stream 'fooz'
When the time
and log
applications are successfully deployed and started in Cloud Foundry, they should automatically connect to the configured external Kafka cluster.
You can verify the connection credentials by accessing the /configprops
of the time
or log
actuator endpoints.
Likewise, you can also see the connection credentials printed in the app logs.
Explicit Stream-level Kafka Connection Configuration
Alternatively, if you intend to deploy only a particular stream with external Kafka connection credentials, you can do so when deploying a stream with explicit overrides:
dataflow:>stream create fooz --definition "time | log"
Created new stream 'fooz'
dataflow:>stream deploy --name fooz --properties "app.*.spring.cloud.stream.kafka.binder.brokers=foo0.broker.foo,foo1.broker.foo,foo2.broker.foo,app.*.spring.spring.cloud.stream.kafka.binder.jaas.options.username=test,app.*.spring.spring.cloud.stream.kafka.binder.jaas.options.password=besttest,app.*.spring.spring.cloud.stream.kafka.binder.jaas.loginModule=org.apache.kafka.common.security.plain.PlainLoginModule"
Deployment request has been sent for stream 'fooz'
When the time
and log
applications are successfully deployed and started in Cloud Foundry, they should automatically connect to the external Kafka cluster.
You can verify the connection credentials by accessing the /configprops
of the time
or log
actuator endpoints.
Likewise, you can also see the connection credentials printed in the app logs.
Streaming Data Pipeline in SCDF for PCF Tile
The option discussed in the Explicit Stream Configuration section should still work when you deploy a stream from Spring Cloud Data Flow running as a managed service in Pivotal Cloud Foundry.
Alternatively, you could supply Kafka connection credentials as CUPS properties when creating the service instance of SCDF for PCF Tile:
cf create-service p-dataflow standard data-flow -c '{"messaging-data-service": { "user-provided": {"brokers":"foo0.broker.foo,foo1.broker.foo,foo2.broker.foo","username":"test","password":"bestest"}}}'
With that, when deploying the stream, you would supply the CUPS properties as values from VCAP_SERVICES
.
dataflow:>stream create fooz --definition "time | log"
Created new stream 'fooz'
dataflow:>stream deploy --name fooz --properties "app.*.spring.cloud.stream.kafka.binder.brokers=${vcap.services.messaging-<GENERATED_GUID>.credentials.brokers},app.*.spring.spring.cloud.stream.kafka.binder.jaas.options.username=${vcap.services.messaging-<GENERATED_GUID>.credentials.username},app.*.spring.spring.cloud.stream.kafka.binder.jaas.options.password=${vcap.services.messaging-<GENERATED_GUID>.credentials.password},app.*.spring.spring.cloud.stream.kafka.binder.jaas.loginModule=org.apache.kafka.common.security.plain.PlainLoginModule"
Deployment request has been sent for stream 'fooz'
Replace <GENERATED_GUID>
with the GUID of the generated messaging service-instance name, which you can find from cf services
command (for example: messaging-b3e76c87-c5ae-47e4-a83c-5fabf2fc4f11
).
As another alternative, you can also provide global configuration properties through the SCDF service app instance. Once the SCDF service instance is ready, you can do the following.
- On the SCDF server app instance UI on the apps manager, go to the settings.
- Find “User Provided Environment Variables”, then find SPRINGCLOUDDATAFLOWTILECONFIGURATION.
- Provide the following as the value: {"spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.brokers": <foo0.broker.foo>, "spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.jaas.loginModule": "org.apache.kafka.common.security.plain.PlainLoginModule", "spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.jaas.options.username": "test", "spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.jaas.options.password": "password", "spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.configuration.security.protocol": "SASL_PLAINTEXT", "spring.cloud.dataflow.applicationProperties.stream.spring.cloud.stream.kafka.binder.configuration.sasl.mechanism": "PLAIN" }
- Once these changes are applied, click the update button and make sure to restart the dataflow server application on CF.
Values in the configuration above (in step #3) are provided for illustrative purposes only. Please update them accordingly. These configurations are for a Kafka cluster that is secured with SASL PLAINTEXT security.