/* * Copyright 2013-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.jvm.java; import static com.facebook.buck.zip.ZipCompressionLevel.DEFAULT_COMPRESSION_LEVEL; import com.facebook.buck.maven.AetherUtil; import com.facebook.buck.model.BuildTargets; import com.facebook.buck.model.Flavor; import com.facebook.buck.model.InternalFlavor; import com.facebook.buck.rules.AbstractBuildRule; import com.facebook.buck.rules.AddToRuleKey; import com.facebook.buck.rules.BuildContext; import com.facebook.buck.rules.BuildRule; import com.facebook.buck.rules.BuildRuleParams; import com.facebook.buck.rules.BuildableContext; import com.facebook.buck.rules.ExplicitBuildTargetSourcePath; import com.facebook.buck.rules.SourcePath; import com.facebook.buck.shell.ShellStep; import com.facebook.buck.step.ExecutionContext; import com.facebook.buck.step.Step; import com.facebook.buck.step.fs.MakeCleanDirectoryStep; import com.facebook.buck.step.fs.MkdirStep; import com.facebook.buck.step.fs.RmStep; import com.facebook.buck.step.fs.WriteFileStep; import com.facebook.buck.zip.ZipCompressionLevel; import com.facebook.buck.zip.ZipStep; import com.google.common.base.Joiner; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import java.io.File; import java.nio.file.Path; import java.util.Optional; public class Javadoc extends AbstractBuildRule implements MavenPublishable { public static final Flavor DOC_JAR = InternalFlavor.of("doc"); @AddToRuleKey private final ImmutableSet<SourcePath> sources; @AddToRuleKey private final Optional<String> mavenCoords; @AddToRuleKey private final Optional<SourcePath> mavenPomTemplate; @AddToRuleKey private final Iterable<HasMavenCoordinates> mavenDeps; private final Path output; private final Path scratchDir; protected Javadoc( BuildRuleParams buildRuleParams, Optional<String> mavenCoords, Optional<SourcePath> mavenPomTemplate, Iterable<HasMavenCoordinates> mavenDeps, ImmutableSet<SourcePath> sources) { super(buildRuleParams); this.mavenCoords = mavenCoords.map(coord -> AetherUtil.addClassifier(coord, "javadoc")); this.mavenPomTemplate = mavenPomTemplate; this.mavenDeps = mavenDeps; this.sources = sources; this.output = BuildTargets.getGenPath( getProjectFilesystem(), getBuildTarget(), String.format("%%s/%s-javadoc.jar", getBuildTarget().getShortName())); this.scratchDir = BuildTargets.getScratchPath( getProjectFilesystem(), getBuildTarget(), String.format("%%s/%s-javadoc.tmp", getBuildTarget().getShortName())); } @Override public ImmutableList<Step> getBuildSteps( BuildContext context, BuildableContext buildableContext) { buildableContext.recordArtifact(output); ImmutableList.Builder<Step> steps = ImmutableList.builder(); steps.add(MkdirStep.of(getProjectFilesystem(), output.getParent())); steps.add(RmStep.of(getProjectFilesystem(), output)); // Fast path: nothing to do so just create an empty zip and return. if (sources.isEmpty()) { steps.add( new ZipStep( getProjectFilesystem(), output, ImmutableSet.<Path>of(), /* junk paths */ false, ZipCompressionLevel.MIN_COMPRESSION_LEVEL, output)); return steps.build(); } Path sourcesListFilePath = scratchDir.resolve("all-sources.txt"); steps.addAll(MakeCleanDirectoryStep.of(getProjectFilesystem(), scratchDir)); // Write an @-file with all the source files in steps.add( new WriteFileStep( getProjectFilesystem(), Joiner.on("\n") .join( sources .stream() .map(context.getSourcePathResolver()::getAbsolutePath) .map(Path::toString) .iterator()), sourcesListFilePath, /* can execute */ false)); Path atArgs = scratchDir.resolve("options"); // Write an @-file with the classpath StringBuilder argsBuilder = new StringBuilder("-classpath "); Joiner.on(File.pathSeparator) .appendTo( argsBuilder, getBuildDeps() .stream() .filter(HasClasspathEntries.class::isInstance) .flatMap(rule -> ((HasClasspathEntries) rule).getTransitiveClasspaths().stream()) .map(context.getSourcePathResolver()::getAbsolutePath) .map(Object::toString) .iterator()); steps.add( new WriteFileStep( getProjectFilesystem(), argsBuilder.toString(), atArgs, /* can execute */ false)); Path uncompressedOutputDir = scratchDir.resolve("docs"); steps.addAll(MakeCleanDirectoryStep.of(getProjectFilesystem(), uncompressedOutputDir)); steps.add( new ShellStep(getProjectFilesystem().resolve(scratchDir)) { @Override protected ImmutableList<String> getShellCommandInternal(ExecutionContext context) { return ImmutableList.of( "javadoc", "-Xdoclint:none", "-notimestamp", "-d", uncompressedOutputDir.getFileName().toString(), "@" + getProjectFilesystem().resolve(atArgs), "@" + getProjectFilesystem().resolve(sourcesListFilePath)); } @Override public String getShortName() { return "javadoc"; } }); steps.add( new ZipStep( getProjectFilesystem(), output, ImmutableSet.of(), /* junk paths */ false, DEFAULT_COMPRESSION_LEVEL, uncompressedOutputDir)); return steps.build(); } @Override public SourcePath getSourcePathToOutput() { return new ExplicitBuildTargetSourcePath(getBuildTarget(), output); } @Override public Optional<String> getMavenCoords() { return mavenCoords; } @Override public Iterable<HasMavenCoordinates> getMavenDeps() { return mavenDeps; } @Override public Iterable<BuildRule> getPackagedDependencies() { return ImmutableSet.of(this); // I think that this is right } @Override public Optional<SourcePath> getPomTemplate() { return mavenPomTemplate; } }