Provides a Docker Swarm mode based discovery strategy for Hazlecast 3.6+ enabled applications. This is an easy to configure plug-and-play Hazlecast DiscoveryStrategy that will enable Hazelcast applications to dynamically discover one another when deployed as Docker Swarm services.
- Status
- Releases
- Requirements
- Maven/Gradle install
- How it works
- Features
- Usage
- Build from source
- Unit tests
- Related Info
- Todo
- Troubleshooting
This is release candidate code, tested against Hazelcast 3.6-EA+ through 3.9.x Stable releases. See Releases below for compatible jars. For use only on Docker 1.12+ "swarm mode" environments.
-
MASTER - in progress, this README refers to what is in the master tag. Switch to relevant RELEASE tag above to see that version's README
-
1.0-RC11 For Hazelcast 3.9+ only. Added
java.util.Properties
based constructor forSwarmMemberAddressProvider
-
1.0-RC10 For Hazelcast 3.9+ only. Better logging in:
DockerDNSRRMemberAddressProvider
for bitsofinfo#25 -
1.0-RC9 For Hazelcast 3.9+ only. Better NPE handling for invalid/null Tasks returned from service spec or no network attachments
-
1.0-RC8: For Hazelcast 3.9+ only. Incorporated PRs #18 (adjust depedencies declaration), #17
-
1.0-RC7: For Hazelcast 3.9+ only. Incorporated PRs #14 (initial scan self check), #15 (docker service names optional)
-
1.0-RC6: For Hazelcast 3.9+ only. Added support for swarm dnsrr based discovery, thanks Cardds, bitsofinfo#10
-
1.0-RC5: For Hazelcast 3.9+ only. Added support for SSL swarm manager URIs and skip verify for SSL.
-
1.0-RC4: For Hazelcast 3.9+ only. Changed gradle dependencies for HZ
3.9.+
& spotify docker-client for8.+
. Implemented newMemberAddressProvider
SPI, as alternative option to usingSwarmAddressPicker
-
1.0-RC3: Use with Hazelcast 3.8.x and below. Will not work with Hazelcast 3.9+. Improved SwarmAddressPicker constructor PR #6
-
1.0-RC1: Initial release
- Java 7+
- Hazelcast 3.6+
- Docker 1.12+ Swarm Mode with one or more swarm manager nodes listening on a
tcp://
socket
To use this discovery strategy in your Maven or Gradle project use the dependency samples below.
repositories {
jcenter()
}
dependencies {
// <!-- Use 1.0.RC3 for Hazelcast < 3.8.x -->
compile 'org.bitsofinfo:hazelcast-docker-swarm-discovery-spi:1.0-RC11'
}
<dependencies>
<dependency>
<groupId>org.bitsofinfo</groupId>
<artifactId>hazelcast-docker-swarm-discovery-spi</artifactId>
<version>1.0-RC11</version> <!-- Use 1.0.RC3 for Hazelcast < 3.8.x -->
</dependency>
</dependencies>
<repositories>
<repository>
<snapshots>
<enabled>false</enabled>
</snapshots>
<id>central</id>
<name>bintray</name>
<url>http://jcenter.bintray.com</url>
</repository>
</repositories>
- Will permit Hazelcast instances deployed on a Docker 1.12+ Swarm to automatically discover and connect with one another
Hazelcast applications that use this discovery SPI will discover one another when deployed as Docker services in the following way.
-
Launch your docker service with its target overlay network name and either its service name or service labels. In addition, specify additional ENVIRONMENT variables via
-e
nameddockerNetworkNames
and optionallydockerServiceNames
anddockerServiceLabels
. You should have at least one ofdockerServiceNames
ordockerServiceLabels
defined. These variables will be consumed by the discovery SPI. TheDOCKER_HOST
environment variable for the container should also be set to a name that resolves to one or more swarm manager nodes, via the formattcp://
,http://
orhttps://
-
The Docker Swarm Discovery SPI consumes from the
DOCKER_HOST
,dockerNetworkNames
, optionallydockerServiceNames
,dockerServiceLabels
andhazelcastPeerPort
and begins the following process.-
Leverages the custom
MemberAddressProvider
SPI implementation (SwarmMemberAddressProvider
) orSwarmAddressPicker
hack (read below!) to talk to the $DOCKER_HOST/networks
,/services
and/tasks
APIs to determine the current node's IP address on the docker network, and bind hazelcast onhazelcastPeerPort
to that address. -
Next hazelcast invokes the SPI
discoverMembers()
to determine all peer docker service tasks (containers) ip addresses and attempts to connect to them to form the cluster connecting to the configuredhazelcastPeerPort
(default 5701)
-
Alternatively, discovery can be limited to the bounds of the docker cluster using docker endpoint_mode dnsrr.
-
Launch your docker service with its service name and target overlay network name with
endpoint_mode
set todnsrr
. In addition, specify additional ENVIRONMENT variables via-e
namedserviceName
,servicePort
, andpeerServicesCsv
-
Leverages the custom
MemberAddressProvider
SPI implementation (DockerDNSRRMemberAddressProvider
) to resolve the DNS entries for its own service,serviceName
, and port,servicePort
, to IP addresses within the docker network. -
Next, hazelcast invokes the SPI DiscoveryStrategy to resolve the internal docker DNS entries for all services that the hazelcast instance should consider its peers, and connects to them over the ports specified in
peerServicesCsv
(default 5701)
Note:
peerServicesCsv
must contain a reference to theserviceName
andservicePort
, if the service is meant to cluster with itself -
Ensure your project has the hazelcast-docker-swarm-discovery-spi
artifact dependency declared in your maven pom or gradle build file as described above. Or build the jar yourself and ensure the jar is in your project's classpath.
Note this is only available in RC4+ and only for Hazelcast 3.9+. apps
Configure your hazelcast.xml configuration file to use the DockerSwarmDiscoveryStrategy
and SwarmMemberAddressProvider
(similar to the below): See hazelcast-docker-swarm-discovery-spi-example-member-address-provider.xml for an example with documentation of options.
Config conf =new ClasspathXmlConfig("yourHzConfig.xml");
HazelcastInstance hazelcastInstance = HazelcastInstanceFactory
.newHazelcastInstance(conf,"myAppName",new DefaultNodeContext());
Note this is only available in RC6+ and only for Hazelcast 3.9+. apps
Configure your hazelcast.xml configuration file to use the DockerDNSRRDiscoveryStrategy
and DockerDNSRRMemberAddressProvider
(similar to the below): See hazelcast-docker-swarm-dnsrr-discovery-spi-example.xml for an example with documentation of options.
Config conf =new ClasspathXmlConfig("yourHzConfig.xml");
HazelcastInstance hazelcastInstance = HazelcastInstanceFactory
.newHazelcastInstance(conf,"myAppName",new DefaultNodeContext());
Note this the preferred method for Hazecast <= 3.8.x apps
Configure your hazelcast.xml configuration file to use the DockerSwarmDiscoveryStrategy
(similar to the below): See hazelcast-docker-swarm-discovery-spi-example-address-picker.xml for an example with documentation of options.
import org.bitsofinfo.hazelcast.discovery.docker.swarm.SwarmAddressPicker;
...
Config conf =new ClasspathXmlConfig("yourHzConfig.xml");
NodeContext nodeContext = new DefaultNodeContext() {
@Override
public AddressPicker createAddressPicker(Node node) {
return new SwarmAddressPicker(new ILogger() {
// you provide the impl... or use provided "SystemPrintLogger"
});
}
};
HazelcastInstance hazelcastInstance = HazelcastInstanceFactory
.newHazelcastInstance(conf,"myAppName",nodeContext);
-
Create a Docker image for your application that uses Hazelcast
-
Create an overlay network for your service,
docker network create -d overlay [name]
-
Launch your services via
docker service create
against your Docker Swarm cluster:
Note this example command assumes an entrypoint script exists that execs the java
command. Your DOCKER_HOST must be accessible over http (i.e. daemons listening on the tcp:// socket
NOTE! All -D
java System properties below can be omitted and alternatively defined within the <member-address-provider>
Hazelcast XML configuration stanza itself. You can mix/match combination of -D defined properties and those defined in Hazelcast XML. Properties defined in Hazelcast XMl take priority.
DOCKER_HOST non-tls
docker service create \
--network [mynet] \
--name myHzService1 \
-l myLabel1=value1 \
-l myLabel2=value2 \
-e "DOCKER_HOST=http://[swarmmgr]:[port]" \
[yourappimage] \
java \
-DdockerNetworkNames=[mynet] \
-DdockerServiceNames=myHzService1 \
-DdockerServiceLabels="myLabel1=value1,myLabel2=value2" \
-DhazelcastPeerPort=5701 \
-jar /test.jar
1.0-RC5+ ONLY: DOCKER_HOST SSL w/ optional skip verify
docker service create \
--network [mynet] \
--name myHzService1 \
-l myLabel1=value1 \
-l myLabel2=value2 \
[yourappimage] \
java \
-DdockerNetworkNames=[mynet] \
-DdockerServiceNames=myHzService1 \
-DdockerServiceLabels="myLabel1=value1,myLabel2=value2" \
-DhazelcastPeerPort=5701 \
-DswarmMgrUri=http(s)://[swarmmgr]:[port] \
-DskipVerifySsl=[true|false] \
-jar /test.jar
NOTE! All -D
java System properties above can be omitted and alternatively defined within the <member-address-provider>
Hazelcast XML configuration stanza itself. You can mix/match combination of -D defined properties and those defined in Hazelcast XML. Properties defined in Hazelcast XMl take priority.
Example configuration (using MemberAddressProvider for Hazelcast 3.9+): see the example: (hazelcast-docker-swarm-discovery-spi-example-member-address-provider.xml)[src/main/resources/META-INF/hazelcast-docker-swarm-discovery-spi-example-member-address-provider.xml]
For Hazelcast <= 3.8.x apps: see the example: (hazelcast-docker-swarm-discovery-spi-example-address-picker.xml)[src/main/resources/META-INF/hazelcast-docker-swarm-discovery-spi-example-address-picker.xml]
<network>
<port auto-increment="true">5701</port>
<interfaces enabled="false">
</interfaces>
<join>
<multicast enabled="false"/>
<aws enabled="false"/>
<tcp-ip enabled="false" />
<!-- for Hazelcast 3.9+ apps only, comment out for <= 3.8.x apps)
<member-address-provider enabled="true">
<class-name>org.bitsofinfo.hazelcast.discovery.docker.swarm.SwarmMemberAddressProvider</class-name>
<!-- OPTIONAL:
The following will be passed as a java.util.Properties to
the SwarmMemberAddressProvider(java.util.Properties) constructor.
If you do not defined these here in XML, by default they will be fetched via
-D java System.properties by the same names via the default no-arg constructor
of SwarmMemberAddressProvider
<property name="dockerNetworkNames">...</property>
<property name="dockerServiceLabels">...</property>
<property name="dockerServiceNames">...</property>
<property name="hazelcastPeerPort">...</property>
<property name="swarmMgrUri">...</property>
<property name="skipVerifySsl">...</property>
-->
</member-address-provider>
<!-- Enable a Docker Swarm based discovery strategy -->
<discovery-strategies>
<discovery-strategy enabled="true"
class="org.bitsofinfo.hazelcast.discovery.docker.swarm.DockerSwarmDiscoveryStrategy">
<properties>
<!-- Comma delimited list of Docker network names to discover matching services on -->
<property name="docker-network-names">${dockerNetworkNames}</property>
<!-- Comma delimited list of relevant Docker service names
to find tasks/containers on the above networks -->
<property name="docker-service-names">${dockerServiceNames}</property>
<!-- Comma delimited list of relevant Docker service label=values
to find tasks/containers on the above networks -->
<property name="docker-service-labels">${dockerServiceLabels}</property>
<!-- 1.0-RC5+ ONLY: Swarm Manager URI (overrides DOCKER_HOST) -->
<property name="swarm-mgr-uri">${swarmMgrUri}</property>
<!-- 1.0-RC5+ ONLY: If Swarm Mgr URI is SSL, to enable skip-verify for it -->
<property name="skip-verify-ssl">${skipVerifySsl}</property>
<!-- The raw port that hazelcast is listening on
IMPORTANT: this is NOT a docker "published" port, nor is it necessarily
a EXPOSEd port... it is simply the hazelcast port that the service
is configured with, this must be the same for all matched containers
in order to work, and just using the default of 5701 is the simplist
way to go.
-->
<property name="hazelcast-peer-port">${hazelcastPeerPort}</property>
</properties>
</discovery-strategy>
</discovery-strategies>
</join>
</network>
-
Create a Docker image for your application that uses Hazelcast
-
Create an overlay network for your service,
docker network create -d overlay [mynetname]
-
Launch your services via
docker service create
against your Docker Swarm cluster:
Note this example command assumes an entrypoint script exists that execs the java
command.
docker service create \
--network [mynetname] \
--name myHzService1 \
--endpoint-mode dnsrr
[yourappimage] \
java
-DserviceName=myHzService1
-DservicePort=5701
-DpeerServicesCsv=myHzService1:5701
-jar /test.jar
Example configuration, full text at hazelcast-docker-swarm-dnsrr-discovery-spi-example.xml
<properties>
<!-- Explicitly enable hazelcast discovery join methods -->
<property name="hazelcast.discovery.enabled">true</property>
</properties>
<network>
<!--
Auto-increment is turned off for the port; docker containers will
always be available at the available in-network ports.
-->
<port auto-increment="false">${servicePort}</port>
<member-address-provider enabled="true">
<class-name>org.bitsofinfo.hazelcast.spi.docker.swarm.dnsrr.DockerDNSRRMemberAddressProvider</class-name>
<properties>
<!-- Name of the docker service that this instance is running in -->
<property name="serviceName">${serviceName}</property>
<!-- Internal port that hazelcast is listening on -->
<property name="servicePort">${servicePort}</property>
</properties>
</member-address-provider>
<join>
<!-- Explicitly disable other cluster join methods -->
<multicast enabled="false"/>
<aws enabled="false"/>
<tcp-ip enabled="false" />
<discovery-strategies>
<discovery-strategy
enabled="true"
class="org.bitsofinfo.hazelcast.spi.docker.swarm.dnsrr.discovery.DockerDNSRRDiscoveryStrategy"
>
<properties>
<!--
Comma separated list of docker services and associated ports
to be considered peers of this service.
Note, this must include itself (the definition of
serviceName and servicePort) if the service is to
cluster with other instances of this service.
-->
<property name="peerServicesCsv">${peerServicesCsv}</property>
</properties>
</discovery-strategy>
</discovery-strategies>
</join>
</network>
-
From the root of this project, build a Jar :
./gradlew jar
-
Include the built jar artifact located at
build/libs/hazelcast-docker-swarm-discovery-spi-[VERSION].jar
in your hazelcast project -
If not already present in your hazelcast application's Maven (pom.xml) or Gradle (build.gradle) dependencies section; ensure that these dependencies are present (versions may vary as appropriate):
compile group: 'com.spotify', name: 'docker-client', version: '8.7.3'
There are really no traditional Java "unit tests" for this SPI due to its reliance on Docker.
There is however a Travis CI test that properly validates the SPI functionality in a real Docker swarm environment that brings up a single instance, scales it to 10 hazelcast nodes and then back down to 2 nodes. Demonstrating the proper discovery of peer hazelcast members.
See the .travis.yml file for the full details.
- https://docs.docker.com/engine/swarm/
- http://docs.hazelcast.org/docs/3.8/manual/html-single/index.html#discovery-spi
- hazelcast/hazelcast#10801
- hazelcast/hazelcast#10802
- hazelcast/hazelcast#11997
- None at this time
If you get an exception (e.g. AbstractMethodError
), this may have been caused by having jersey-common
library twice.
(One from the plugin itself -as a transitive dependency- and the other from the shaded docker-client
library. In such
a case you may add an exclusion to your project's build file.
For maven:
<dependency>
<groupId>org.bitsofinfo</groupId>
<artifactId>hazelcast-docker-swarm-discovery-spi</artifactId>
<version>1.0-RC5</version>
<exclusions>
<exclusion>
<groupId>org.glassfish.jersey.core</groupId>
<artifactId>jersey-common</artifactId>
</exclusion>
</exclusions>
</dependency>
For gradle:
compile('org.bitsofinfo:hazelcast-docker-swarm-discovery-spi:1.0-RC5') {
exclude module: 'jersey-common'
}
For details please see Pull Request #6.