blob: 24088a761647c38478dd38efc91c5553b11ac98c [file] [log] [blame]
/*
* Copyright 2012-present Facebook, Inc.
*
* 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.facebook.buck.cli;
import com.facebook.buck.command.Build;
import com.facebook.buck.json.BuildFileParseException;
import com.facebook.buck.model.BuildTarget;
import com.facebook.buck.model.BuildTargetException;
import com.facebook.buck.parser.NoSuchBuildTargetException;
import com.facebook.buck.rules.ArtifactCache;
import com.facebook.buck.rules.BuildEvent;
import com.facebook.buck.rules.BuildRule;
import com.facebook.buck.rules.DependencyGraph;
import com.facebook.buck.step.StepFailedException;
import com.facebook.buck.step.TargetDevice;
import com.facebook.buck.util.Console;
import com.facebook.buck.util.ExceptionWithHumanReadableMessage;
import com.facebook.buck.util.HumanReadableException;
import com.facebook.buck.util.Verbosity;
import com.google.common.base.Joiner;
import com.google.common.base.Optional;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterators;
import java.io.IOException;
import java.util.Set;
import java.util.concurrent.ExecutionException;
public class BuildCommand extends AbstractCommandRunner<BuildCommandOptions> {
private Build build;
private ImmutableList<BuildTarget> buildTargets = ImmutableList.of();
public BuildCommand(CommandRunnerParams params) {
super(params);
}
@Override
BuildCommandOptions createOptions(BuckConfig buckConfig) {
return new BuildCommandOptions(buckConfig);
}
@Override
@SuppressWarnings("PMD.PrematureDeclaration")
int runCommandWithOptionsInternal(BuildCommandOptions options) throws IOException {
// Set the logger level based on the verbosity option.
Verbosity verbosity = console.getVerbosity();
Logging.setLoggingLevelForVerbosity(verbosity);
// Create artifact cache to initialize Cassandra connection, if appropriate.
ArtifactCache artifactCache = getArtifactCache();
try {
buildTargets = getBuildTargets(options.getArgumentsFormattedAsBuildTargets());
} catch (NoSuchBuildTargetException e) {
console.printBuildFailureWithoutStacktrace(e);
return 1;
}
if (buildTargets.isEmpty()) {
console.printBuildFailure("Must specify at least one build target.");
// If there are aliases defined in .buckconfig, suggest that the user
// build one of them. We show the user only the first 10 aliases.
ImmutableSet<String> aliases = options.getBuckConfig().getAliases();
if (!aliases.isEmpty()) {
console.getStdErr().println(String.format(
"Try building one of the following targets:\n%s",
Joiner.on(' ').join(Iterators.limit(aliases.iterator(), 10))
));
}
return 1;
}
getBuckEventBus().post(BuildEvent.started(buildTargets));
// Parse the build files to create a DependencyGraph.
DependencyGraph dependencyGraph;
try {
dependencyGraph = getParser().parseBuildFilesForTargets(buildTargets,
options.getDefaultIncludes(),
getBuckEventBus());
} catch (BuildTargetException | BuildFileParseException e) {
console.printBuildFailureWithoutStacktrace(e);
return 1;
}
// Create and execute the build.
build = options.createBuild(
options.getBuckConfig(),
dependencyGraph,
getProjectFilesystem(),
artifactCache,
console,
getBuckEventBus(),
Optional.<TargetDevice>absent(),
getCommandRunnerParams().getPlatform());
int exitCode = 0;
try {
exitCode = executeBuildAndPrintAnyFailuresToConsole(build, console);
} finally {
// Shutdown the Executor Service once the build completes.
build.getStepRunner().getListeningExecutorService().shutdownNow();
}
getBuckEventBus().post(BuildEvent.finished(buildTargets,exitCode));
if (exitCode != 0) {
return exitCode;
}
return 0;
}
static int executeBuildAndPrintAnyFailuresToConsole(Build build, Console console) {
Set<BuildRule> rulesToBuild = build.getDependencyGraph().getNodesWithNoIncomingEdges();
int exitCode;
try {
// Get the Future representing the build and then block until everything is built.
build.executeBuild(rulesToBuild).get();
exitCode = 0;
} catch (IOException e) {
console.printBuildFailureWithoutStacktrace(e);
exitCode = 1;
} catch (StepFailedException e) {
console.printBuildFailureWithoutStacktrace(e);
exitCode = e.getExitCode();
} catch (ExecutionException e) {
// This is likely a checked exception that was caught while building a build rule.
Throwable cause = e.getCause();
if (cause instanceof HumanReadableException) {
throw ((HumanReadableException)cause);
} else if (cause instanceof ExceptionWithHumanReadableMessage) {
throw new HumanReadableException((ExceptionWithHumanReadableMessage)cause);
} else {
if (cause instanceof RuntimeException) {
console.printBuildFailureWithStacktrace(e);
} else {
console.printBuildFailureWithoutStacktrace(e);
}
exitCode = 1;
}
} catch (InterruptedException e) {
// This suggests an error in Buck rather than a user error.
// Print the entire stack trace so we can debug it.
console.printBuildFailureWithStacktrace(e);
exitCode = 1;
}
return exitCode;
}
Build getBuild() {
Preconditions.checkNotNull(build);
return build;
}
ImmutableList<BuildTarget> getBuildTargets() {
return ImmutableList.copyOf(buildTargets);
}
@Override
String getUsageIntro() {
return "Specify one build rule to build.";
}
}