com.spotify.helios.cli.command.JobInspectCommand Maven / Gradle / Ivy
/*-
* -\-\-
* Helios Tools
* --
* 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.cli.command;
import static com.spotify.helios.cli.Utils.printMap;
import com.google.common.base.CharMatcher;
import com.google.common.base.Function;
import com.google.common.base.Joiner;
import com.google.common.base.Strings;
import com.google.common.collect.Lists;
import com.google.common.collect.Ordering;
import com.spotify.helios.client.HeliosClient;
import com.spotify.helios.common.Json;
import com.spotify.helios.common.descriptors.ExecHealthCheck;
import com.spotify.helios.common.descriptors.HealthCheck;
import com.spotify.helios.common.descriptors.HttpHealthCheck;
import com.spotify.helios.common.descriptors.Job;
import com.spotify.helios.common.descriptors.PortMapping;
import com.spotify.helios.common.descriptors.RolloutOptions;
import com.spotify.helios.common.descriptors.ServicePorts;
import com.spotify.helios.common.descriptors.TcpHealthCheck;
import java.io.BufferedReader;
import java.io.PrintStream;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Map;
import java.util.TimeZone;
import java.util.concurrent.ExecutionException;
import net.sourceforge.argparse4j.inf.Namespace;
import net.sourceforge.argparse4j.inf.Subparser;
public class JobInspectCommand extends WildcardJobCommand {
private static final DateFormat DATE_FORMATTER =
new SimpleDateFormat("EEE, d MMM yyyy HH:mm:ss Z");
private static final Function QUOTE = new Function() {
@Override
public String apply(final String input) {
return quote(input);
}
};
private static final Function FORMAT_PORTMAPPING =
new Function() {
@Override
public String apply(final PortMapping input) {
final StringBuilder builder = new StringBuilder();
if (input.getIp() != null) {
builder.append(input.getIp());
}
builder.append(":").append(input.getInternalPort());
if (input.getExternalPort() != null) {
builder.append(":").append(input.getExternalPort());
}
if (input.getProtocol() != null) {
builder.append("/").append(input.getProtocol());
}
return builder.toString();
}
};
private static final Function FORMAT_SERVICE_PORTS =
new Function() {
@Override
public String apply(final ServicePorts input) {
return Joiner.on(", ").join(Ordering.natural().sortedCopy(input.getPorts().keySet()));
}
};
private static String formatHealthCheck(final HealthCheck healthCheck) {
if (healthCheck == null) {
return "";
}
String str = String.format("type: %s", String.valueOf(healthCheck.getType()));
if (healthCheck instanceof HttpHealthCheck) {
final HttpHealthCheck httpHealthCheck = (HttpHealthCheck) healthCheck;
str += String.format(", port: %s, path: %s", httpHealthCheck.getPort(),
httpHealthCheck.getPath());
} else if (healthCheck instanceof TcpHealthCheck) {
final TcpHealthCheck tcpHealthCheck = (TcpHealthCheck) healthCheck;
str += String.format(", port: %s", tcpHealthCheck.getPort());
} else if (healthCheck instanceof ExecHealthCheck) {
final ExecHealthCheck execHealthCheck = (ExecHealthCheck) healthCheck;
str += String.format(", command: %s", Joiner.on(" ").join(execHealthCheck.getCommand()));
}
return str;
}
private static String formatRolloutOptions(final RolloutOptions options) {
if (options == null) {
return "";
}
final List output = new ArrayList<>();
output.add(String.format("timeout: %d", options.getTimeout()));
output.add(String.format("parallelism: %d", options.getParallelism()));
output.add(String.format("migrate: %s", options.getMigrate()));
output.add(String.format("overlap: %s", options.getOverlap()));
output.add(String.format("token: %s", options.getToken()));
output.add(String.format("ignoreFailures: %s", options.getIgnoreFailures()));
return Joiner.on(", ").join(output);
}
public JobInspectCommand(final Subparser parser) {
super(parser);
parser.help("print the configuration of a job");
}
public JobInspectCommand(final Subparser parser, final TimeZone timeZone) {
super(parser);
DATE_FORMATTER.setTimeZone(timeZone);
parser.help("print the configuration of a job");
}
@Override
protected int runWithJob(final Namespace options, final HeliosClient client,
final PrintStream out, final boolean json, final Job job,
final BufferedReader stdin)
throws ExecutionException, InterruptedException {
if (json) {
out.println(Json.asPrettyStringUnchecked(job));
} else {
out.printf("Id: %s%n", job.getId());
out.printf("Image: %s%n", job.getImage());
out.printf("Created: %s%n",
job.getCreated() == null ? "" : DATE_FORMATTER.format(new Date(job.getCreated())));
out.printf("Expires: %s%n",
job.getExpires() == null ? "never" : DATE_FORMATTER.format(job.getExpires()));
out.printf("Hostname: %s%n", job.getHostname());
out.printf("Command: %s%n", quote(job.getCommand()));
printMap(out, "Env: ", QUOTE, job.getEnv());
out.printf("Health check: %s%n", formatHealthCheck(job.getHealthCheck()));
out.printf("Grace period (seconds): %s%n", job.getGracePeriod());
out.printf("Time to wait before kill (seconds): %s%n", job.getSecondsToWaitBeforeKill());
printMap(out, "Metadata: ", QUOTE, job.getMetadata());
printMap(out, "Ports: ", FORMAT_PORTMAPPING, job.getPorts());
printMap(out, "Reg: ", FORMAT_SERVICE_PORTS, job.getRegistration());
out.printf("Security options: %s%n", job.getSecurityOpt());
out.printf("Network mode: %s%n", job.getNetworkMode());
out.printf("Token: %s%n", job.getToken());
printVolumes(out, job.getVolumes());
out.printf("Add capabilities: %s%n", Joiner.on(", ").join(job.getAddCapabilities()));
out.printf("Drop capabilities: %s%n", Joiner.on(", ").join(job.getDropCapabilities()));
out.printf("Runtime: %s%n", job.getRuntime());
out.printf("Rollout options (null options will fallback to defaults at "
+ "rolling-update time): %s%n", formatRolloutOptions(job.getRolloutOptions()));
}
return 0;
}
private void printVolumes(final PrintStream out, final Map volumes) {
final String prefix = "Volumes: ";
out.print(prefix);
boolean first = true;
for (final Map.Entry entry : volumes.entrySet()) {
if (!first) {
out.print(Strings.repeat(" ", prefix.length()));
}
final String path = entry.getValue();
final String source = entry.getKey();
if (source == null) {
out.printf("%s%n", path);
} else {
// Note that we're printing this in value:key order as that's the host:container:[rw|ro]
// order used by docker and the helios create command.
out.printf("%s:%s%n", path, source);
}
first = false;
}
if (first) {
out.println();
}
}
private static String quote(final String str) {
if (str == null) {
return "";
}
return CharMatcher.whitespace().matchesAnyOf(str)
? '"' + str + '"'
: str;
}
private static List quote(final List ss) {
final List output = Lists.newArrayList();
for (final String s : ss) {
output.add(quote(s));
}
return output;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy