Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.spotify.helios.client.HeliosClient Maven / Gradle / Ivy
/*-
* -\-\-
* Helios 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.helios.client;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Strings.isNullOrEmpty;
import static com.google.common.util.concurrent.Futures.catching;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import static com.google.common.util.concurrent.Futures.transform;
import static com.google.common.util.concurrent.Futures.transformAsync;
import static com.spotify.helios.common.VersionCompatibility.HELIOS_SERVER_VERSION_HEADER;
import static com.spotify.helios.common.VersionCompatibility.HELIOS_VERSION_STATUS_HEADER;
import static java.lang.String.format;
import static java.net.HttpURLConnection.HTTP_BAD_METHOD;
import static java.net.HttpURLConnection.HTTP_BAD_REQUEST;
import static java.net.HttpURLConnection.HTTP_FORBIDDEN;
import static java.net.HttpURLConnection.HTTP_NOT_FOUND;
import static java.net.HttpURLConnection.HTTP_OK;
import static java.util.Arrays.asList;
import static java.util.Collections.singletonList;
import static java.util.concurrent.TimeUnit.SECONDS;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.type.TypeFactory;
import com.google.common.base.Function;
import com.google.common.base.Optional;
import com.google.common.base.Strings;
import com.google.common.base.Supplier;
import com.google.common.base.Suppliers;
import com.google.common.base.Throwables;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.collect.Multimap;
import com.google.common.collect.Multimaps;
import com.google.common.escape.Escaper;
import com.google.common.net.UrlEscapers;
import com.google.common.util.concurrent.AsyncFunction;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningScheduledExecutorService;
import com.google.common.util.concurrent.MoreExecutors;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import com.spotify.helios.common.HeliosException;
import com.spotify.helios.common.Json;
import com.spotify.helios.common.Resolver;
import com.spotify.helios.common.Version;
import com.spotify.helios.common.VersionCompatibility;
import com.spotify.helios.common.descriptors.Deployment;
import com.spotify.helios.common.descriptors.DeploymentGroup;
import com.spotify.helios.common.descriptors.HostStatus;
import com.spotify.helios.common.descriptors.Job;
import com.spotify.helios.common.descriptors.JobId;
import com.spotify.helios.common.descriptors.JobStatus;
import com.spotify.helios.common.descriptors.RolloutOptions;
import com.spotify.helios.common.protocol.CreateDeploymentGroupResponse;
import com.spotify.helios.common.protocol.CreateJobResponse;
import com.spotify.helios.common.protocol.DeploymentGroupStatusResponse;
import com.spotify.helios.common.protocol.HostDeregisterResponse;
import com.spotify.helios.common.protocol.JobDeleteResponse;
import com.spotify.helios.common.protocol.JobDeployResponse;
import com.spotify.helios.common.protocol.JobUndeployResponse;
import com.spotify.helios.common.protocol.RemoveDeploymentGroupResponse;
import com.spotify.helios.common.protocol.RollingUpdateRequest;
import com.spotify.helios.common.protocol.RollingUpdateResponse;
import com.spotify.helios.common.protocol.SetGoalResponse;
import com.spotify.helios.common.protocol.TaskStatusEvents;
import com.spotify.helios.common.protocol.VersionResponse;
import com.spotify.sshagentproxy.AgentProxies;
import com.spotify.sshagentproxy.AgentProxy;
import com.spotify.sshagenttls.CertKeyPaths;
import java.io.Closeable;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import javax.annotation.Nullable;
import org.apache.http.client.utils.URIBuilder;
import org.jetbrains.annotations.NotNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class HeliosClient implements Closeable {
private static final Logger log = LoggerFactory.getLogger(HeliosClient.class);
private final String user;
private final RequestDispatcher dispatcher;
private final AtomicBoolean versionWarningLogged = new AtomicBoolean();
private final TypeReference> jobIdMap = new TypeReference>() {
};
public HeliosClient(final String user, final RequestDispatcher dispatcher) {
this.user = checkNotNull(user);
this.dispatcher = checkNotNull(dispatcher);
}
@Override
public void close() throws IOException {
dispatcher.close();
}
private URI uri(final String path) {
return uri(path, Collections.emptyMap());
}
private URI uri(final String path, final Map query) {
return uri(path, Multimaps.forMap(query));
}
private URI uri(final String path, final Multimap query) {
checkArgument(path.startsWith("/"));
final URIBuilder builder = new URIBuilder()
.setScheme("http")
.setHost("helios")
.setPath(path);
for (final Map.Entry q : query.entries()) {
builder.addParameter(q.getKey(), q.getValue());
}
builder.addParameter("user", user);
try {
return builder.build();
} catch (URISyntaxException e) {
throw new RuntimeException(e);
}
}
private String path(final String resource, final Object... params) {
final String path;
final Escaper escaper = UrlEscapers.urlPathSegmentEscaper();
if (params.length == 0) {
path = resource;
} else {
final List encodedParams = Lists.newArrayList();
for (final Object param : params) {
encodedParams.add(escaper.escape(param.toString()));
}
path = format(resource, encodedParams.toArray());
}
return path;
}
private ListenableFuture request(final URI uri, final String method) {
return request(uri, method, null);
}
private ListenableFuture request(final URI uri, final String method,
final Object entity) {
final Map> headers = Maps.newHashMap();
final byte[] entityBytes;
headers.put(VersionCompatibility.HELIOS_VERSION_HEADER,
Collections.singletonList(Version.POM_VERSION));
if (entity != null) {
headers.put("Content-Type", singletonList("application/json"));
headers.put("Charset", singletonList("utf-8"));
entityBytes = Json.asBytesUnchecked(entity);
} else {
entityBytes = new byte[]{};
}
final ListenableFuture f = dispatcher.request(uri, method, entityBytes, headers);
return transform(f, new Function() {
@Override
public Response apply(final Response response) {
checkProtocolVersionStatus(response);
return response;
}
});
}
private void checkProtocolVersionStatus(final Response response) {
final VersionCompatibility.Status versionStatus = getVersionStatus(response);
if (versionStatus == null) {
log.debug("Server didn't return a version header!");
return; // shouldn't happen really
}
final String serverVersion = response.header(HELIOS_SERVER_VERSION_HEADER);
if ((versionStatus == VersionCompatibility.Status.MAYBE)
&& (versionWarningLogged.compareAndSet(false, true))) {
log.warn("Your Helios client version [{}] is ahead of the server [{}]. This will"
+ " probably work ok but there is the potential for weird things. If in doubt,"
+ " contact the Helios team if you think the cluster you're connecting to is out"
+ " of date and should be upgraded.", Version.POM_VERSION, serverVersion);
}
}
private VersionCompatibility.Status getVersionStatus(final Response response) {
final String status = response.header(HELIOS_VERSION_STATUS_HEADER);
if (status != null) {
return VersionCompatibility.Status.valueOf(status);
}
return null;
}
private ListenableFuture get(final URI uri, final TypeReference typeReference) {
return get(uri, Json.type(typeReference));
}
private ListenableFuture get(final URI uri, final Class clazz) {
return get(uri, Json.type(clazz));
}
private ListenableFuture get(final URI uri, final JavaType javaType) {
return transformAsync(request(uri, "GET"), new ConvertResponseToPojo(javaType));
}
private ListenableFuture put(final URI uri) {
return status(request(uri, "PUT"));
}
public ListenableFuture deploy(final Deployment job, final String host) {
return deploy(job, host, "");
}
public ListenableFuture deploy(final Deployment job, final String host,
final String token) {
final Set deserializeReturnCodes = ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND,
HTTP_BAD_METHOD,
HTTP_BAD_REQUEST,
HTTP_FORBIDDEN);
return transformAsync(request(uri(path("/hosts/%s/jobs/%s", host, job.getJobId()),
ImmutableMap.of("token", token)),
"PUT", job),
ConvertResponseToPojo.create(JobDeployResponse.class, deserializeReturnCodes));
}
public ListenableFuture setGoal(final Deployment job, final String host) {
return setGoal(job, host, "");
}
public ListenableFuture setGoal(final Deployment job, final String host,
final String token) {
return transformAsync(request(uri(path("/hosts/%s/jobs/%s", host, job.getJobId()),
ImmutableMap.of("token", token)),
"PATCH", job),
ConvertResponseToPojo.create(SetGoalResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND,
HTTP_FORBIDDEN)));
}
private ListenableFuture status(final ListenableFuture req) {
return transform(req,
new Function() {
@Override
public Integer apply(final Response reply) {
return reply.status();
}
});
}
public ListenableFuture deployment(final String host, final JobId job) {
return get(uri(path("/hosts/%s/jobs/%s", host, job)), Deployment.class);
}
public ListenableFuture hostStatus(final String host) {
return hostStatus(host, Collections.emptyMap());
}
public ListenableFuture hostStatus(
final String host,
final Map queryParams) {
return get(uri(path("/hosts/%s/status", host), queryParams), HostStatus.class);
}
public ListenableFuture> hostStatuses(final List hosts) {
return hostStatuses(hosts, Collections.emptyMap());
}
public ListenableFuture> hostStatuses(
final List hosts,
final Map queryParams) {
final ConvertResponseToPojo> converter = ConvertResponseToPojo.create(
TypeFactory.defaultInstance().constructMapType(Map.class, String.class, HostStatus.class),
ImmutableSet.of(HTTP_OK));
return transformAsync(request(uri("/hosts/statuses", queryParams), "POST", hosts), converter);
}
public ListenableFuture registerHost(final String host, final String id) {
return put(uri(path("/hosts/%s", host), ImmutableMap.of("id", id)));
}
public ListenableFuture deleteJob(final JobId id) {
return deleteJob(id, "");
}
public ListenableFuture deleteJob(final JobId id, final String token) {
return transformAsync(request(uri(path("/jobs/%s", id),
ImmutableMap.of("token", token)),
"DELETE"),
ConvertResponseToPojo.create(JobDeleteResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND,
HTTP_BAD_REQUEST,
HTTP_FORBIDDEN)));
}
public ListenableFuture undeploy(final JobId jobId, final String host) {
return undeploy(jobId, host, "");
}
public ListenableFuture undeploy(final JobId jobId, final String host,
final String token) {
return transformAsync(request(uri(path("/hosts/%s/jobs/%s", host, jobId),
ImmutableMap.of("token", token)),
"DELETE"),
ConvertResponseToPojo.create(JobUndeployResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND,
HTTP_BAD_REQUEST,
HTTP_FORBIDDEN)));
}
public ListenableFuture deregisterHost(final String host) {
return transformAsync(request(uri(path("/hosts/%s", host)), "DELETE"),
ConvertResponseToPojo.create(HostDeregisterResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND)));
}
/**
* Returns a list of all hosts registered in the Helios cluster.
*/
public ListenableFuture> listHosts() {
return listHosts(ImmutableMultimap.of());
}
/**
* Returns a list of all hosts registered in the Helios cluster whose name matches the given
* pattern.
*/
public ListenableFuture> listHosts(final String namePattern) {
return listHosts(ImmutableMultimap.of("namePattern", namePattern));
}
/**
* Returns a list of all hosts registered in the Helios cluster which match the given list of
* host
* selectors.
*
* For example, {@code listHosts(Arrays.asList("site=foo"))} will return all agents in the
* cluster whose labels match the expression {@code site=foo}.
*/
public ListenableFuture> listHosts(final Set unparsedHostSelectors) {
final Multimap query = HashMultimap.create();
query.putAll("selector", unparsedHostSelectors);
return listHosts(query);
}
/**
* Returns a list of all hosts registered in the Helios cluster that match both the given hostname
* pattern and set of host selectors.
*
* @see #listHosts(Set)
*/
public ListenableFuture> listHosts(final String namePattern,
final Set unparsedHostSelectors) {
final Multimap query = HashMultimap.create();
query.put("namePattern", namePattern);
query.putAll("selector", unparsedHostSelectors);
return listHosts(query);
}
private ListenableFuture> listHosts(final Multimap query) {
return get(uri("/hosts/", query), new TypeReference>() {
});
}
public ListenableFuture> listMasters() {
return get(uri("/masters/"), new TypeReference>() {
});
}
public ListenableFuture version() {
// Create a fallback in case we fail to connect to the master. Return null if this happens.
// The transform below will handle this and return an appropriate error message to the caller.
final ListenableFuture futureWithFallback = catching(
request(uri("/version/"), "GET"),
Exception.class,
new Function() {
@Override
public Response apply(final Exception ex) {
return null;
}
}
);
return transformAsync(
futureWithFallback,
new AsyncFunction() {
@Override
public ListenableFuture apply(@NotNull Response reply) throws Exception {
final String masterVersion =
reply == null ? "Unable to connect to master" :
reply.status() == HTTP_OK ? Json.read(reply.payload(), String.class) :
"Master replied with error code " + reply.status();
return immediateFuture(new VersionResponse(Version.POM_VERSION, masterVersion));
}
});
}
public ListenableFuture createJob(final Job descriptor) {
return transformAsync(request(uri("/jobs/"), "POST", descriptor),
ConvertResponseToPojo.create(CreateJobResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_BAD_REQUEST)));
}
/**
* Lists all jobs that exist in the Helios cluster.
*/
public ListenableFuture> jobs() {
return jobs(null, null);
}
/**
* Lists all jobs in the cluster whose name starts with the given string.
*
* @param jobQuery job name filter
*/
public ListenableFuture> jobs(@Nullable final String jobQuery) {
return jobs(jobQuery, null);
}
/**
* Lists all jobs in the cluster whose name starts with the given string, and that are deployed to
* hosts that match the given name pattern.
*
* @param jobQuery job name filter
* @param hostNamePattern hostname filter
*/
public ListenableFuture> jobs(@Nullable final String jobQuery,
@Nullable final String hostNamePattern) {
final Map params = new HashMap<>();
if (!Strings.isNullOrEmpty(jobQuery)) {
params.put("q", jobQuery);
}
if (!Strings.isNullOrEmpty(hostNamePattern)) {
params.put("hostPattern", hostNamePattern);
}
return get(uri("/jobs", params), jobIdMap);
}
public ListenableFuture jobHistory(final JobId jobId) {
return transformAsync(
request(uri(path("/history/jobs/%s", jobId.toString())), "GET"),
ConvertResponseToPojo.create(TaskStatusEvents.class,
ImmutableSet.of(HTTP_OK, HTTP_NOT_FOUND)));
}
public ListenableFuture jobStatus(final JobId jobId) {
return get(uri(path("/jobs/%s/status", jobId)), JobStatus.class);
}
public ListenableFuture> jobStatuses(final Set jobs) {
final ConvertResponseToPojo> converter = ConvertResponseToPojo.create(
TypeFactory.defaultInstance().constructMapType(Map.class, JobId.class, JobStatus.class),
ImmutableSet.of(HTTP_OK));
return transformAsync(request(uri("/jobs/statuses"), "POST", jobs), converter);
}
public ListenableFuture deploymentGroup(final String name) {
return get(uri("/deployment-group/" + name), new TypeReference() {
});
}
public ListenableFuture> listDeploymentGroups() {
return get(uri("/deployment-group/"), new TypeReference>() {
});
}
public ListenableFuture deploymentGroupStatus(final String name) {
return get(uri(path("/deployment-group/%s/status", name)),
new TypeReference() {});
}
public ListenableFuture createDeploymentGroup(
final DeploymentGroup descriptor) {
return transformAsync(request(uri("/deployment-group/"), "POST", descriptor),
ConvertResponseToPojo.create(CreateDeploymentGroupResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_BAD_REQUEST)));
}
public ListenableFuture removeDeploymentGroup(final String name) {
return transformAsync(request(uri("/deployment-group/" + name), "DELETE"),
ConvertResponseToPojo.create(RemoveDeploymentGroupResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_BAD_REQUEST)));
}
public ListenableFuture rollingUpdate(
final String deploymentGroupName, final JobId job, final RolloutOptions options) {
return transformAsync(
request(uri(path("/deployment-group/%s/rolling-update", deploymentGroupName)),
"POST", new RollingUpdateRequest(job, options)),
ConvertResponseToPojo.create(RollingUpdateResponse.class,
ImmutableSet.of(HTTP_OK, HTTP_BAD_REQUEST)));
}
public ListenableFuture stopDeploymentGroup(final String deploymentGroupName) {
return status(request(
uri(path("/deployment-group/%s/stop", deploymentGroupName)), "POST"));
}
private static final class ConvertResponseToPojo implements AsyncFunction {
private final JavaType javaType;
private final Set decodeableStatusCodes;
private ConvertResponseToPojo(final JavaType javaType) {
this(javaType, ImmutableSet.of(HTTP_OK));
}
public ConvertResponseToPojo(final JavaType type, final Set decodeableStatusCodes) {
this.javaType = type;
this.decodeableStatusCodes = decodeableStatusCodes;
}
public static ConvertResponseToPojo create(final JavaType type,
final Set decodeableStatusCodes) {
return new ConvertResponseToPojo<>(type, decodeableStatusCodes);
}
public static ConvertResponseToPojo create(final Class clazz,
final Set decodeableStatusCodes) {
return new ConvertResponseToPojo<>(Json.type(clazz), decodeableStatusCodes);
}
@Override
public ListenableFuture apply(@NotNull final Response reply)
throws HeliosException {
if (reply.status() == HTTP_NOT_FOUND && !decodeableStatusCodes.contains(HTTP_NOT_FOUND)) {
return immediateFuture(null);
}
if (!decodeableStatusCodes.contains(reply.status())) {
throw new HeliosException("request failed: " + reply);
}
if (reply.payload().length == 0) {
throw new HeliosException("bad reply: " + reply);
}
final T result;
try {
result = Json.read(reply.payload(), javaType);
} catch (IOException e) {
throw new HeliosException("bad reply: " + reply, e);
}
return immediateFuture(result);
}
}
public static Builder newBuilder() {
return new Builder();
}
public static class Builder {
private static final String HELIOS_CERT_PATH = "HELIOS_CERT_PATH";
// used in the name format of the Client's executor's ThreadFactory to differentiate between
// different instances of HeliosClient. this way we avoid having multiple threads named
// "helios-client-1" etc.
private static final AtomicInteger clientCounter = new AtomicInteger(0);
private String user;
private CertKeyPaths certKeyPaths;
private Supplier> endpointSupplier;
private boolean sslHostnameVerification = true;
private ListeningScheduledExecutorService executorService;
private boolean shutDownExecutorOnClose = true;
private int httpTimeout = 10000;
private long requestRetryTimeout = 60000;
private Builder() {
}
public Builder setUser(final String user) {
this.user = user;
return this;
}
public Builder setDomain(final String domain) {
return setEndpointSupplier(Endpoints.of(Resolver.supplier("helios", domain)));
}
public Builder setEndpoints(final List endpoints) {
return setEndpointSupplier(Suppliers.ofInstance(Endpoints.of(endpoints)));
}
public Builder setEndpoints(final URI... endpoints) {
return setEndpointSupplier(Suppliers.ofInstance(Endpoints.of(asList(endpoints))));
}
public Builder setEndpoints(final String... endpoints) {
return setEndpointStrings(asList(endpoints));
}
public Builder setEndpointStrings(final List endpoints) {
final List uris = Lists.newArrayList();
for (final String endpoint : endpoints) {
uris.add(URI.create(endpoint));
}
return setEndpoints(uris);
}
public Builder setEndpointSupplier(final Supplier> endpointSupplier) {
this.endpointSupplier = endpointSupplier;
return this;
}
/**
* Can be used to disable hostname verification for HTTPS connections to the Helios master.
* Defaults to being enabled.
*/
public Builder setSslHostnameVerification(final boolean enabled) {
this.sslHostnameVerification = enabled;
return this;
}
public Builder setCertKeyPaths(final CertKeyPaths certKeyPaths) {
this.certKeyPaths = certKeyPaths;
return this;
}
public Builder setExecutorService(final ScheduledExecutorService executorService) {
this.executorService = MoreExecutors.listeningDecorator(executorService);
return this;
}
public Builder setShutDownExecutorOnClose(final boolean shutDownExecutorOnClose) {
this.shutDownExecutorOnClose = shutDownExecutorOnClose;
return this;
}
/**
* Set the per-request HTTP connect/read timeout used when communicating with master. Default is
* 10 seconds.
*/
public Builder setHttpTimeout(final int timeout, TimeUnit unit) {
this.httpTimeout = (int) unit.toMillis(timeout);
return this;
}
/**
* Set the total amount of time for which the HeliosClient will retrying failed requests to the
* Helios masters.
*/
public Builder setRetryTimeout(final int timeout, TimeUnit unit) {
this.requestRetryTimeout = (int) unit.toMillis(timeout);
return this;
}
public HeliosClient build() {
return new HeliosClient(user, createDispatcher());
}
private static ListeningScheduledExecutorService defaultExecutorService() {
final int clientCount = clientCounter.incrementAndGet();
final ThreadFactory threadFactory = new ThreadFactoryBuilder()
.setNameFormat("helios-client-" + clientCount + "-thread-%d")
.build();
final ScheduledThreadPoolExecutor stpe = new ScheduledThreadPoolExecutor(4, threadFactory);
final ScheduledExecutorService exitingExecutor =
MoreExecutors.getExitingScheduledExecutorService(stpe, 0, SECONDS);
return MoreExecutors.listeningDecorator(exitingExecutor);
}
private RequestDispatcher createDispatcher() {
if (executorService == null) {
executorService = defaultExecutorService();
}
final RequestDispatcher dispatcher = new DefaultRequestDispatcher(
createHttpConnector(sslHostnameVerification), executorService, shutDownExecutorOnClose);
return RetryingRequestDispatcher.forDispatcher(dispatcher)
.setExecutor(executorService)
.setRetryTimeout(requestRetryTimeout, TimeUnit.MILLISECONDS)
.build();
}
private HttpConnector createHttpConnector(final boolean sslHostnameVerification) {
final EndpointIterator endpointIterator = EndpointIterator.of(endpointSupplier.get());
if (!endpointIterator.hasNext()) {
throw new IllegalStateException(
"no endpoints found to connect to, check your configuration");
}
final DefaultHttpConnector connector =
new DefaultHttpConnector(endpointIterator, httpTimeout, sslHostnameVerification);
Optional agentProxyOpt = Optional.absent();
try {
agentProxyOpt = Optional.of(AgentProxies.newInstance());
} catch (RuntimeException e) {
// the user likely doesn't have ssh-agent setup. This may not matter at all if the masters
// do not require authentication, so we delay reporting any sort of error to the user until
// the servers return 401 Unauthorized.
log.debug("Exception (possibly benign) while loading AgentProxy", e);
}
// set up the CertKeyPaths, giving precedence to any values set with setCertKeyPaths()
if (certKeyPaths == null) {
final String heliosCertPath = System.getenv(HELIOS_CERT_PATH);
if (!isNullOrEmpty(heliosCertPath)) {
final Path certPath = Paths.get(heliosCertPath, "cert.pem");
final Path keyPath = Paths.get(heliosCertPath, "key.pem");
if (certPath.toFile().canRead() && keyPath.toFile().canRead()) {
this.certKeyPaths = CertKeyPaths.create(certPath, keyPath);
} else {
log.warn("{} is set to {}, but {} and/or {} do not exist or cannot be read. "
+ "Will not send client certificate in HeliosClient requests.",
HELIOS_CERT_PATH, heliosCertPath, certPath, keyPath);
}
}
}
return new AuthenticatingHttpConnector(user,
agentProxyOpt,
Optional.fromNullable(certKeyPaths),
endpointIterator,
connector);
}
}
/**
* Create a new helios client as a specific user, connecting to a helios master cluster in a
* specific domain.
*
* @param domain The target domain.
* @param user The user to identify as.
* @return A helios client.
*/
public static HeliosClient create(final String domain, final String user) {
return HeliosClient.newBuilder()
.setDomain(domain)
.setUser(user)
.build();
}
}