All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.spotify.helios.client.HeliosClient Maven / Gradle / Ivy

There is a newer version: 0.9.283
Show newest version
/*-
 * -\-\-
 * 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.auth.oauth2.AccessToken;
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.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 boolean googleCredentialsEnabled = true; private AccessToken googleAccessToken; private List googleAccessTokenScopes = GoogleCredentialsAccessTokenSupplier.DEFAULT_SCOPES; 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 setGoogleCredentialsEnabled(final boolean enabled) { this.googleCredentialsEnabled = enabled; return this; } public Builder setGoogleAccessToken(final AccessToken accessToken) { this.googleAccessToken = accessToken; return this; } public Builder setGoogleAccessTokenScopes(final List scopes) { this.googleAccessTokenScopes = scopes; 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); Supplier> accessTokenSupplier = new GoogleCredentialsAccessTokenSupplier(googleCredentialsEnabled, googleAccessToken, googleAccessTokenScopes); 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, accessTokenSupplier, 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(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy