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.hubspot.singularity.executor.task.SingularityExecutorTaskProcessBuilder Maven / Gradle / Ivy
package com.hubspot.singularity.executor.task;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.Callable;
import org.apache.mesos.Protos.TaskInfo;
import org.apache.mesos.Protos.TaskState;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.base.Strings;
import com.google.common.collect.Lists;
import com.hubspot.deploy.ArtifactList;
import com.hubspot.deploy.EmbeddedArtifact;
import com.hubspot.deploy.ExecutorData;
import com.hubspot.deploy.ExternalArtifact;
import com.hubspot.deploy.RemoteArtifact;
import com.hubspot.deploy.S3Artifact;
import com.hubspot.deploy.S3ArtifactSignature;
import com.hubspot.singularity.SingularityTaskExecutorData;
import com.hubspot.singularity.executor.TemplateManager;
import com.hubspot.singularity.executor.config.SingularityExecutorConfiguration;
import com.hubspot.singularity.executor.models.DockerContext;
import com.hubspot.singularity.executor.models.EnvironmentContext;
import com.hubspot.singularity.executor.models.RunnerContext;
import com.hubspot.singularity.executor.task.SingularityExecutorArtifactFetcher.SingularityExecutorTaskArtifactFetcher;
import com.hubspot.singularity.executor.utils.DockerUtils;
import com.hubspot.singularity.executor.utils.ExecutorUtils;
import com.hubspot.singularity.runner.base.shared.ProcessFailedException;
import com.spotify.docker.client.exceptions.DockerException;
public class SingularityExecutorTaskProcessBuilder implements Callable {
private final SingularityExecutorTask task;
private final TemplateManager templateManager;
private final SingularityExecutorConfiguration configuration;
private final String executorPid;
private final ExecutorUtils executorUtils;
private final SingularityTaskExecutorData executorData;
private final SingularityExecutorArtifactFetcher artifactFetcher;
private Optional taskArtifactFetcher;
private DockerUtils dockerUtils;
private final ObjectMapper objectMapper;
public SingularityExecutorTaskProcessBuilder(SingularityExecutorTask task,
ExecutorUtils executorUtils,
SingularityExecutorArtifactFetcher artifactFetcher,
TemplateManager templateManager,
SingularityExecutorConfiguration configuration,
SingularityTaskExecutorData executorData, String executorPid,
DockerUtils dockerUtils, ObjectMapper objectMapper) {
this.executorData = executorData;
this.objectMapper = objectMapper;
this.task = task;
this.executorUtils = executorUtils;
this.artifactFetcher = artifactFetcher;
this.templateManager = templateManager;
this.configuration = configuration;
this.executorPid = executorPid;
this.taskArtifactFetcher = Optional.empty();
this.dockerUtils = dockerUtils;
}
@Override
public ProcessBuilder call() throws Exception {
if (task.getTaskInfo().hasContainer() && task.getTaskInfo().getContainer().hasDocker()) {
executorUtils.sendStatusUpdate(task.getDriver(), task.getTaskInfo().getTaskId(), TaskState.TASK_STARTING, String.format("Pulling image... (executor pid: %s)", executorPid), task.getLog());
try {
dockerUtils.pull(task.getTaskInfo().getContainer().getDocker().getImage());
} catch (DockerException e) {
throw new ProcessFailedException("Could not pull docker image", e);
}
}
executorUtils.sendStatusUpdate(task.getDriver(), task.getTaskInfo().getTaskId(), TaskState.TASK_STARTING, String.format("Staging files... (executor pid: %s)", executorPid), task.getLog());
taskArtifactFetcher = Optional.of(artifactFetcher.buildTaskFetcher(executorData, task));
taskArtifactFetcher.get().fetchFiles(executorData.getEmbeddedArtifacts(), executorData.getS3Artifacts(),
executorData.getS3ArtifactSignaturesOrEmpty(), executorData.getExternalArtifacts());
task.getArtifactVerifier().checkSignatures(executorData.getS3ArtifactSignaturesOrEmpty());
List artifactLists = new ArrayList<>();
artifactLists.addAll(checkArtifactsForArtifactLists(executorData.getS3Artifacts()));
artifactLists.addAll(checkArtifactsForArtifactLists(executorData.getS3ArtifactSignaturesOrEmpty()));
artifactLists.addAll(checkArtifactsForArtifactLists(executorData.getExternalArtifacts()));
if (!artifactLists.isEmpty()) {
List embeddedArtifacts = new ArrayList<>();
List s3Artifacts = new ArrayList<>();
List s3ArtifactSignatures = new ArrayList<>();
List externalArtifacts = new ArrayList<>();
for (ArtifactList artifactList : artifactLists) {
embeddedArtifacts.addAll(artifactList.getEmbeddedArtifacts());
s3Artifacts.addAll(artifactList.getS3Artifacts());
s3ArtifactSignatures.addAll(artifactList.getS3ArtifactSignatures());
externalArtifacts.addAll(artifactList.getExternalArtifacts());
}
task.getLog().info("Found {} artifact lists with {} embedded, {} s3, {} external, fetching...", artifactLists.size(), embeddedArtifacts.size(), s3Artifacts.size() + s3ArtifactSignatures.size(),
externalArtifacts.size());
taskArtifactFetcher.get().fetchFiles(embeddedArtifacts, s3Artifacts, s3ArtifactSignatures, externalArtifacts);
task.getArtifactVerifier().checkSignatures(s3ArtifactSignatures);
}
ProcessBuilder processBuilder = buildProcessBuilder(task.getTaskInfo(), executorData, task.getTaskDefinition().getServiceLogFileName());
task.getTaskLogManager().setup();
return processBuilder;
}
private List checkArtifactsForArtifactLists(List extends RemoteArtifact> remoteArtifacts) {
List artifactLists = new ArrayList<>();
for (RemoteArtifact remoteArtifact : remoteArtifacts) {
if (remoteArtifact.isArtifactList()) {
Path pathToArtifact = task.getArtifactPath(remoteArtifact, task.getTaskDefinition().getTaskDirectoryPath()).resolve(remoteArtifact.getFilename());
if (!Files.exists(pathToArtifact)) {
throw new IllegalStateException(String.format("Couldn't find artifact at %s - %s", pathToArtifact, remoteArtifact));
}
try {
artifactLists.add(objectMapper.readValue(pathToArtifact.toFile(), ArtifactList.class));
} catch (IOException e) {
throw new RuntimeException("Couldn't read artifacts from " + pathToArtifact, e);
}
}
}
return artifactLists;
}
public void cancel() {
if (taskArtifactFetcher.isPresent()) {
taskArtifactFetcher.get().cancel();
}
}
private Path getPath(String filename) {
return task.getTaskDefinition().getTaskDirectoryPath().resolve(filename);
}
private String getCommand(ExecutorData executorData) {
final StringBuilder bldr = new StringBuilder(Strings.isNullOrEmpty(executorData.getCmd()) ? "" : executorData.getCmd());
for (String extraCmdLineArg : executorData.getExtraCmdLineArgs()) {
bldr.append(" ");
bldr.append(extraCmdLineArg);
}
return bldr.toString();
}
private String getExecutorUser() {
return System.getProperty("user.name"); // TODO: better way to do this?
}
private ProcessBuilder buildProcessBuilder(TaskInfo taskInfo, SingularityTaskExecutorData executorData, String serviceLog) {
final String cmd = getCommand(executorData);
boolean isDocker = taskInfo.hasContainer() && taskInfo.getContainer().hasDocker();
RunnerContext runnerContext = new RunnerContext(
cmd,
configuration.getTaskAppDirectory(),
configuration.getLogrotateToDirectory(),
executorData.getUser().orElse(configuration.getDefaultRunAsUser()),
serviceLog,
serviceLogOutPath(serviceLog),
task.getTaskId(),
executorData.getMaxTaskThreads().isPresent() ? executorData.getMaxTaskThreads() : configuration.getMaxTaskThreads(),
!getExecutorUser().equals(executorData.getUser().orElse(configuration.getDefaultRunAsUser())),
executorData.getMaxOpenFiles().orElse(null),
String.format(configuration.getSwitchUserCommandFormat(), executorData.getUser().orElse(configuration.getDefaultRunAsUser())),
configuration.isUseFileAttributes(),
getCfsQuota(executorData),
configuration.getDefaultCfsPeriod(),
isDocker ? configuration.getExtraDockerScriptContent() : configuration.getExtraScriptContent());
EnvironmentContext environmentContext = new EnvironmentContext(taskInfo);
if (isDocker) {
task.getLog().info("Writing a runner script to execute {} in docker container", cmd);
templateManager.writeDockerScript(getPath("runner.sh"),
new DockerContext(environmentContext, runnerContext, configuration.getDockerPrefix(), configuration.getDockerStopTimeout(), taskInfo.getContainer().getDocker().getPrivileged()));
} else {
templateManager.writeEnvironmentScript(getPath("deploy.env"), environmentContext);
task.getLog().info("Writing a runner script to execute {} with {}", cmd, runnerContext);
templateManager.writeRunnerScript(getPath("runner.sh"), runnerContext);
}
List command = Lists.newArrayList();
command.add("bash");
command.add("runner.sh");
ProcessBuilder processBuilder = new ProcessBuilder(command);
processBuilder.directory(task.getTaskDefinition().getTaskDirectoryPath().toFile());
processBuilder.redirectError(task.getTaskDefinition().getExecutorBashOutPath().toFile());
processBuilder.redirectOutput(task.getTaskDefinition().getExecutorBashOutPath().toFile());
return processBuilder;
}
private Integer getCfsQuota(SingularityTaskExecutorData executorData) {
if (!executorData.getCpuHardLimit().isPresent()) {
return null;
} else {
return executorData.getCpuHardLimit().get() * configuration.getDefaultCfsPeriod();
}
}
private String serviceLogOutPath(String serviceLog) {
Path basePath = task.getTaskDefinition().getTaskDirectoryPath();
Path app = basePath.resolve(configuration.getTaskAppDirectory()).normalize();
return app.relativize(basePath).resolve(serviceLog).toString();
}
@Override
public String toString() {
return "SingularityExecutorTaskProcessBuilder [task=" + task.getTaskId() + "]";
}
}