io.vertx.servicediscovery.docker.DockerServiceImporter Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2016 The original author or authors
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* and Apache License v2.0 which accompanies this distribution.
*
* The Eclipse Public License is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* The Apache License v2.0 is available at
* http://www.opensource.org/licenses/apache2.0.php
*
* You may elect to redistribute this code under either of these licenses.
*/
package io.vertx.servicediscovery.docker;
import com.github.dockerjava.api.DockerClient;
import com.github.dockerjava.api.model.Container;
import com.github.dockerjava.core.DockerClientBuilder;
import com.github.dockerjava.core.DockerClientConfig;
import io.vertx.core.Future;
import io.vertx.core.Handler;
import io.vertx.core.Vertx;
import io.vertx.core.json.JsonObject;
import io.vertx.core.logging.Logger;
import io.vertx.core.logging.LoggerFactory;
import io.vertx.servicediscovery.Record;
import io.vertx.servicediscovery.spi.ServiceImporter;
import io.vertx.servicediscovery.spi.ServicePublisher;
import java.io.IOException;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.List;
/**
* A discovery bridge collecting services from Docker, and importing them in the Vert.x
* discovery infrastructure.
*
* @author Clement Escoffier
*/
public class DockerServiceImporter implements ServiceImporter {
private final static Logger LOGGER = LoggerFactory.getLogger(DockerServiceImporter.class);
private long timer;
private DockerClient client;
private List services = new ArrayList<>();
private ServicePublisher publisher;
private Vertx vertx;
private String host;
volatile boolean started;
/**
* Starts the bridge.
*
* @param vertx the vert.x instance
* @param publisher the service discovery instance
* @param configuration the bridge configuration if any
* @param completion future to assign with completion status
*/
@Override
public void start(Vertx vertx, ServicePublisher publisher, JsonObject configuration, Future completion) {
this.publisher = publisher;
this.vertx = vertx;
DockerClientConfig.DockerClientConfigBuilder builder =
DockerClientConfig.createDefaultConfigBuilder();
String dockerCertPath = configuration.getString("docker-cert-path");
String dockerCfgPath = configuration.getString("docker-cfg-path");
String email = configuration.getString("docker-registry-email");
String password = configuration.getString("docker-registry-password");
String username = configuration.getString("docker-registry-username");
String host = configuration.getString("docker-host");
boolean tlsVerify = configuration.getBoolean("docker-tls-verify", true);
String registry
= configuration.getString("docker-registry-url", "https://index.docker.io/v1/");
String version = configuration.getString("version");
if (dockerCertPath != null) {
builder.withDockerCertPath(dockerCertPath);
}
if (dockerCfgPath != null) {
builder.withDockerConfig(dockerCfgPath);
}
if (email != null) {
builder.withRegistryEmail(email);
}
if (password != null) {
builder.withRegistryPassword(password);
}
if (username != null) {
builder.withRegistryUsername(username);
}
if (host != null) {
builder.withDockerHost(host);
}
if (registry != null) {
builder.withRegistryUrl(registry);
}
if (version != null) {
builder.withApiVersion(version);
}
builder.withDockerTlsVerify(tlsVerify);
DockerClientConfig config = builder.build();
if (config.getDockerHost().getScheme().equalsIgnoreCase("unix")) {
try {
this.host = InetAddress.getLocalHost().getHostAddress();
} catch (UnknownHostException e) {
completion.fail(e);
}
} else {
this.host = config.getDockerHost().getHost();
}
client = DockerClientBuilder.getInstance(config).build();
long period = configuration.getLong("scan-period", 3000L);
if (period > 0) {
timer = vertx.setPeriodic(period, l -> {
scan(null);
});
}
scan(completion);
}
synchronized void scan(Future completion) {
vertx.>executeBlocking(
future -> {
try {
future.complete(client.listContainersCmd().withStatusFilter("running").exec());
} catch (Exception e) {
future.fail(e);
}
},
ar -> {
if (ar.failed()) {
if (completion != null) {
completion.fail(ar.cause());
} else {
LOGGER.error("Fail to import services from docker", ar.cause());
}
return;
}
started = true;
List running = ar.result();
List toRemove = new ArrayList<>();
// Detect lost containers
services.stream()
.filter(service -> isNotRunning(service.id(), running))
.forEach(service -> {
unpublish(service);
toRemove.add(service);
});
services.removeAll(toRemove);
if (running != null) {
// Detect new containers
running.stream()
.filter(container -> !isKnown(container))
.forEach(container -> {
DockerService service = new DockerService(container, host);
if (!service.records().isEmpty()) {
services.add(service);
publish(service);
}
});
}
if (completion != null) {
completion.complete();
}
}
);
}
private void publish(DockerService service) {
for (Record record : service.records()) {
publisher.publish(record, ar -> {
if (ar.succeeded()) {
record.setRegistration(ar.result().getRegistration());
LOGGER.info("Service from container " + service.id() + " on location "
+ record.getLocation() + " has been published");
} else {
LOGGER.error("Service from container " + service.id() + " on location "
+ record.getLocation() + " could not have been published", ar.cause());
}
});
}
}
private void unpublish(DockerService service) {
for (Record record : service.records()) {
publisher.unpublish(record.getRegistration(), ar -> {
LOGGER.info("Service from container " + service.id()
+ " on location " + record.getLocation() + " has been unpublished");
});
}
}
private boolean isKnown(Container container) {
for (DockerService service : services) {
if (service.id().equalsIgnoreCase(container.getId())) {
return true;
}
}
return false;
}
private boolean isNotRunning(String containerId, List running) {
if (running == null) {
// No running container
return true;
}
for (Container container : running) {
if (container.getId().equalsIgnoreCase(containerId)) {
// Found in the running list
return false;
}
}
// Not found in the running list
return true;
}
@Override
public void close(Handler completionHandler) {
vertx.cancelTimer(timer);
try {
started = false;
client.close();
LOGGER.info("Successfully closed the service importer " + this);
} catch (IOException e) {
LOGGER.error("A failure has been caught while stopping " + this, e);
}
if (completionHandler != null) {
completionHandler.handle(null);
}
}
synchronized List getServices() {
return new ArrayList<>(services);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy