com.spotify.docker.client.DockerHost Maven / Gradle / Ivy
/*-
* -\-\-
* docker-client
* --
* Copyright (C) 2016 Spotify AB
* --
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* -/-/-
*/
package com.spotify.docker.client;
import static com.google.common.base.MoreObjects.firstNonNull;
import static com.google.common.base.Strings.isNullOrEmpty;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.MoreObjects;
import com.google.common.net.HostAndPort;
import java.net.URI;
import java.nio.file.Paths;
import java.util.Locale;
/**
* Represents a dockerd endpoint. A codified DOCKER_HOST.
*/
public class DockerHost {
/**
* An interface to be mocked during testing.
*/
@VisibleForTesting
interface SystemDelegate {
String getProperty(String key);
String getenv(String name);
}
private static final SystemDelegate defaultSystemDelegate = new SystemDelegate() {
@Override
public String getProperty(final String key) {
return System.getProperty(key);
}
@Override
public String getenv(final String name) {
return System.getenv(name);
}
};
private static SystemDelegate systemDelegate = defaultSystemDelegate;
private static final String DEFAULT_UNIX_ENDPOINT = "unix:///var/run/docker.sock";
private static final String DEFAULT_WINDOWS_ENDPOINT = "npipe:////./pipe/docker_engine";
private static final String DEFAULT_ADDRESS = "localhost";
private static final int DEFAULT_PORT = 2375;
private final String host;
private final URI uri;
private final URI bindUri;
private final String address;
private final int port;
private final String certPath;
private DockerHost(final String endpoint, final String certPath) {
if (endpoint.startsWith("unix://")) {
this.port = 0;
this.address = DEFAULT_ADDRESS;
this.host = endpoint;
this.uri = URI.create(endpoint);
this.bindUri = URI.create(endpoint);
} else {
final String stripped = endpoint.replaceAll(".*://", "");
final HostAndPort hostAndPort = HostAndPort.fromString(stripped);
final String hostText = hostAndPort.getHost();
final String scheme = isNullOrEmpty(certPath) ? "http" : "https";
this.port = hostAndPort.getPortOrDefault(defaultPort());
this.address = isNullOrEmpty(hostText) ? DEFAULT_ADDRESS : hostText;
this.host = address + ":" + port;
this.uri = URI.create(scheme + "://" + address + ":" + port);
this.bindUri = URI.create("tcp://" + address + ":" + port);
}
this.certPath = certPath;
}
/**
* Get a Docker endpoint usable for instantiating a new DockerHost with
* DockerHost.from(endpoint).
*
* @return A unix socket path or, in the case of a TCP socket, the hostname and port which
* represents a Docker endpoint.
*/
public String host() {
return host;
}
/**
* Get the Docker rest uri.
*
* @return The uri of the Docker endpoint.
*/
public URI uri() {
return uri;
}
/**
* Get the Docker rest bind uri.
*
* @return The uri of the host for binding ports (or setting $DOCKER_HOST).
*/
public URI bindUri() {
return bindUri;
}
/**
* Get the Docker endpoint port.
*
* @return The port.
*/
public int port() {
return port;
}
/**
* Get the Docker ip address or hostname.
*
* @return The ip address or hostname.
*/
public String address() {
return address;
}
/**
* Get the path to certificate and key for connecting to Docker via HTTPS.
*
* @return The path to the certificate.
*/
public String dockerCertPath() {
return certPath;
}
@VisibleForTesting
static void setSystemDelegate(final SystemDelegate delegate) {
systemDelegate = delegate;
}
@VisibleForTesting
static void restoreSystemDelegate() {
systemDelegate = defaultSystemDelegate;
}
/**
* Create a {@link DockerHost} from DOCKER_HOST and DOCKER_PORT env vars.
*
* @return The DockerHost object.
*/
public static DockerHost fromEnv() {
final String host = endpointFromEnv();
final String certPath = certPathFromEnv();
return new DockerHost(host, certPath);
}
/**
* Create a {@link DockerHost} from an explicit address or uri.
*
* @param endpoint The Docker endpoint.
* @param certPath The certificate path.
* @return The DockerHost object.
*/
public static DockerHost from(final String endpoint, final String certPath) {
return new DockerHost(endpoint, certPath);
}
static String defaultDockerEndpoint() {
final String osName = systemDelegate.getProperty("os.name");
final String os = osName.toLowerCase(Locale.ENGLISH);
if (os.equalsIgnoreCase("linux") || os.contains("mac")) {
return DEFAULT_UNIX_ENDPOINT;
} else if (System.getProperty("os.name").equalsIgnoreCase("Windows 10")) {
//from Docker doc: Windows 10 64bit: Pro, Enterprise or Education
return DEFAULT_WINDOWS_ENDPOINT;
} else {
return DEFAULT_ADDRESS + ":" + defaultPort();
}
}
static String endpointFromEnv() {
return firstNonNull(systemDelegate.getenv("DOCKER_HOST"), defaultDockerEndpoint());
}
public static String defaultUnixEndpoint() {
return DEFAULT_UNIX_ENDPOINT;
}
public static String defaultWindowsEndpoint() {
return DEFAULT_WINDOWS_ENDPOINT;
}
public static String defaultAddress() {
return DEFAULT_ADDRESS;
}
public static int defaultPort() {
return DEFAULT_PORT;
}
static int portFromEnv() {
final String port = systemDelegate.getenv("DOCKER_PORT");
if (port == null) {
return defaultPort();
}
try {
return Integer.parseInt(port);
} catch (NumberFormatException e) {
return defaultPort();
}
}
static String defaultCertPath() {
final String userHome = systemDelegate.getProperty("user.home");
return Paths.get(userHome, ".docker").toString();
}
static String certPathFromEnv() {
return systemDelegate.getenv("DOCKER_CERT_PATH");
}
@Override
public String toString() {
return MoreObjects.toStringHelper(this)
.add("host", host)
.add("uri", uri)
.add("bindUri", bindUri)
.add("address", address)
.add("port", port)
.add("certPath", certPath)
.toString();
}
@Override
public boolean equals(final Object obj) {
if (this == obj) {
return true;
}
if (obj == null || getClass() != obj.getClass()) {
return false;
}
final DockerHost that = (DockerHost) obj;
if (port != that.port) {
return false;
}
if (host != null ? !host.equals(that.host) : that.host != null) {
return false;
}
if (uri != null ? !uri.equals(that.uri) : that.uri != null) {
return false;
}
if (bindUri != null ? !bindUri.equals(that.bindUri) : that.bindUri != null) {
return false;
}
if (address != null ? !address.equals(that.address) : that.address != null) {
return false;
}
return certPath != null ? certPath.equals(that.certPath) : that.certPath == null;
}
@Override
public int hashCode() {
int result = host != null ? host.hashCode() : 0;
result = 31 * result + (uri != null ? uri.hashCode() : 0);
result = 31 * result + (bindUri != null ? bindUri.hashCode() : 0);
result = 31 * result + (address != null ? address.hashCode() : 0);
result = 31 * result + port;
result = 31 * result + (certPath != null ? certPath.hashCode() : 0);
return result;
}
}