com.facebook.buck.cxx.CxxTestDescription.java Source code

Java tutorial

Introduction

Here is the source code for com.facebook.buck.cxx.CxxTestDescription.java

Source

/*
 * Copyright 2014-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.cxx;

import com.facebook.buck.model.BuildTarget;
import com.facebook.buck.model.Flavor;
import com.facebook.buck.model.FlavorDomain;
import com.facebook.buck.model.Flavored;
import com.facebook.buck.model.MacroException;
import com.facebook.buck.parser.NoSuchBuildTargetException;
import com.facebook.buck.rules.BuildRule;
import com.facebook.buck.rules.BuildRuleParams;
import com.facebook.buck.rules.BuildRuleResolver;
import com.facebook.buck.rules.CellPathResolver;
import com.facebook.buck.rules.Description;
import com.facebook.buck.rules.ImplicitDepsInferringDescription;
import com.facebook.buck.rules.Label;
import com.facebook.buck.rules.MetadataProvidingDescription;
import com.facebook.buck.rules.SourcePathResolver;
import com.facebook.buck.rules.SourcePathRuleFinder;
import com.facebook.buck.rules.SourcePaths;
import com.facebook.buck.rules.TargetGraph;
import com.facebook.buck.util.HumanReadableException;
import com.facebook.buck.util.MoreCollectors;
import com.facebook.buck.versions.VersionRoot;
import com.facebook.infer.annotation.SuppressFieldNotInitialized;
import com.google.common.base.Preconditions;
import com.google.common.base.Supplier;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Maps;
import com.google.common.collect.Ordering;
import com.google.common.collect.Sets;

import java.nio.file.Path;
import java.util.Optional;

public class CxxTestDescription implements Description<CxxTestDescription.Arg>, Flavored,
        ImplicitDepsInferringDescription<CxxTestDescription.Arg>,
        MetadataProvidingDescription<CxxTestDescription.Arg>, VersionRoot<CxxTestDescription.Arg> {

    private static final CxxTestType DEFAULT_TEST_TYPE = CxxTestType.GTEST;

    private final CxxBuckConfig cxxBuckConfig;
    private final CxxPlatform defaultCxxPlatform;
    private final FlavorDomain<CxxPlatform> cxxPlatforms;
    private final Optional<Long> defaultTestRuleTimeoutMs;

    public CxxTestDescription(CxxBuckConfig cxxBuckConfig, CxxPlatform defaultCxxPlatform,
            FlavorDomain<CxxPlatform> cxxPlatforms, Optional<Long> defaultTestRuleTimeoutMs) {
        this.cxxBuckConfig = cxxBuckConfig;
        this.defaultCxxPlatform = defaultCxxPlatform;
        this.cxxPlatforms = cxxPlatforms;
        this.defaultTestRuleTimeoutMs = defaultTestRuleTimeoutMs;
    }

    @Override
    public Arg createUnpopulatedConstructorArg() {
        return new Arg();
    }

    @SuppressWarnings("PMD.PrematureDeclaration")
    @Override
    public <A extends Arg> BuildRule createBuildRule(TargetGraph targetGraph, BuildRuleParams inputParams,
            final BuildRuleResolver resolver, final A args) throws NoSuchBuildTargetException {
        Optional<StripStyle> flavoredStripStyle = StripStyle.FLAVOR_DOMAIN.getValue(inputParams.getBuildTarget());
        Optional<LinkerMapMode> flavoredLinkerMapMode = LinkerMapMode.FLAVOR_DOMAIN
                .getValue(inputParams.getBuildTarget());
        inputParams = CxxStrip.removeStripStyleFlavorInParams(inputParams, flavoredStripStyle);
        inputParams = LinkerMapMode.removeLinkerMapModeFlavorInParams(inputParams, flavoredLinkerMapMode);
        final BuildRuleParams params = inputParams;

        Optional<CxxPlatform> platform = cxxPlatforms.getValue(params.getBuildTarget());
        CxxPlatform cxxPlatform = platform.orElse(defaultCxxPlatform);
        SourcePathRuleFinder ruleFinder = new SourcePathRuleFinder(resolver);
        SourcePathResolver pathResolver = new SourcePathResolver(ruleFinder);

        if (params.getBuildTarget().getFlavors().contains(CxxCompilationDatabase.COMPILATION_DATABASE)) {
            return CxxDescriptionEnhancer.createCompilationDatabase(params, resolver, pathResolver, ruleFinder,
                    cxxBuckConfig, cxxPlatform, args);
        }

        if (params.getBuildTarget().getFlavors().contains(CxxCompilationDatabase.UBER_COMPILATION_DATABASE)) {
            return CxxDescriptionEnhancer.createUberCompilationDatabase(
                    platform.isPresent() ? params : params.withFlavor(cxxPlatform.getFlavor()), resolver);
        }

        if (params.getBuildTarget().getFlavors().contains(CxxDescriptionEnhancer.SANDBOX_TREE_FLAVOR)) {
            return CxxDescriptionEnhancer.createSandboxTreeBuildRule(resolver, args, cxxPlatform, params);
        }

        // Generate the link rule that builds the test binary.
        final CxxLinkAndCompileRules cxxLinkAndCompileRules = CxxDescriptionEnhancer
                .createBuildRulesForCxxBinaryDescriptionArg(params, resolver, cxxBuckConfig, cxxPlatform, args,
                        flavoredStripStyle, flavoredLinkerMapMode);

        // Construct the actual build params we'll use, notably with an added dependency on the
        // CxxLink rule above which builds the test binary.
        BuildRuleParams testParams = params.appendExtraDeps(cxxLinkAndCompileRules.executable.getDeps(ruleFinder));
        testParams = CxxStrip.restoreStripStyleFlavorInParams(testParams, flavoredStripStyle);
        testParams = LinkerMapMode.restoreLinkerMapModeFlavorInParams(testParams, flavoredLinkerMapMode);

        // Supplier which expands macros in the passed in test environment.
        Supplier<ImmutableMap<String, String>> testEnv = () -> ImmutableMap
                .copyOf(Maps.transformValues(args.env, CxxDescriptionEnhancer.MACRO_HANDLER
                        .getExpander(params.getBuildTarget(), params.getCellRoots(), resolver)));

        // Supplier which expands macros in the passed in test arguments.
        Supplier<ImmutableList<String>> testArgs = () -> args.args.stream().map(CxxDescriptionEnhancer.MACRO_HANDLER
                .getExpander(params.getBuildTarget(), params.getCellRoots(), resolver)::apply)
                .collect(MoreCollectors.toImmutableList());

        Supplier<ImmutableSortedSet<BuildRule>> additionalDeps = () -> {
            ImmutableSortedSet.Builder<BuildRule> deps = ImmutableSortedSet.naturalOrder();

            // It's not uncommon for users to add dependencies onto other binaries that they run
            // during the test, so make sure to add them as runtime deps.
            deps.addAll(Sets.difference(params.getDeps(), cxxLinkAndCompileRules.getBinaryRule().getDeps()));

            // Add any build-time from any macros embedded in the `env` or `args` parameter.
            for (String part : Iterables.concat(args.args, args.env.values())) {
                try {
                    deps.addAll(CxxDescriptionEnhancer.MACRO_HANDLER.extractBuildTimeDeps(params.getBuildTarget(),
                            params.getCellRoots(), resolver, part));
                } catch (MacroException e) {
                    throw new HumanReadableException(e, "%s: %s", params.getBuildTarget(), e.getMessage());
                }
            }

            return deps.build();
        };

        CxxTest test;

        CxxTestType type = args.framework.orElse(getDefaultTestType());
        switch (type) {
        case GTEST: {
            test = new CxxGtestTest(testParams, pathResolver, ruleFinder, cxxLinkAndCompileRules.getBinaryRule(),
                    cxxLinkAndCompileRules.executable, testEnv, testArgs,
                    FluentIterable.from(args.resources)
                            .transform(SourcePaths.toSourcePath(params.getProjectFilesystem()))
                            .toSortedSet(Ordering.natural()),
                    additionalDeps, args.labels, args.contacts, args.runTestSeparately.orElse(false),
                    args.testRuleTimeoutMs.map(Optional::of).orElse(defaultTestRuleTimeoutMs),
                    cxxBuckConfig.getMaximumTestOutputSize());
            break;
        }
        case BOOST: {
            test = new CxxBoostTest(testParams, pathResolver, ruleFinder, cxxLinkAndCompileRules.getBinaryRule(),
                    cxxLinkAndCompileRules.executable, testEnv, testArgs,
                    FluentIterable.from(args.resources)
                            .transform(SourcePaths.toSourcePath(params.getProjectFilesystem()))
                            .toSortedSet(Ordering.natural()),
                    additionalDeps, args.labels, args.contacts, args.runTestSeparately.orElse(false),
                    args.testRuleTimeoutMs.map(Optional::of).orElse(defaultTestRuleTimeoutMs));
            break;
        }
        default: {
            Preconditions.checkState(false, "Unhandled C++ test type: %s", type);
            throw new RuntimeException();
        }
        }

        return test;
    }

    @Override
    public Iterable<BuildTarget> findDepsForTargetFromConstructorArgs(BuildTarget buildTarget,
            CellPathResolver cellRoots, Arg constructorArg) {

        ImmutableSet.Builder<BuildTarget> deps = ImmutableSet.builder();

        // Get any parse time deps from the C/C++ platforms.
        deps.addAll(CxxPlatforms
                .getParseTimeDeps(cxxPlatforms.getValue(buildTarget.getFlavors()).orElse(defaultCxxPlatform)));

        // Extract parse time deps from flags, args, and environment parameters.
        Iterable<Iterable<String>> macroStrings = ImmutableList.<Iterable<String>>builder()
                .add(constructorArg.linkerFlags).addAll(constructorArg.platformLinkerFlags.getValues())
                .add(constructorArg.args).add(constructorArg.env.values()).build();
        for (String macroString : Iterables.concat(macroStrings)) {
            try {
                deps.addAll(CxxDescriptionEnhancer.MACRO_HANDLER.extractParseTimeDeps(buildTarget, cellRoots,
                        macroString));
            } catch (MacroException e) {
                throw new HumanReadableException(e, "%s: %s", buildTarget, e.getMessage());
            }
        }

        CxxTestType type = constructorArg.framework.orElse(getDefaultTestType());
        switch (type) {
        case GTEST: {
            deps.add(cxxBuckConfig.getGtestDep());
            boolean useDefaultTestMain = constructorArg.useDefaultTestMain.orElse(true);
            if (useDefaultTestMain) {
                deps.add(cxxBuckConfig.getGtestDefaultTestMainDep());
            }
            break;
        }
        case BOOST: {
            deps.add(cxxBuckConfig.getBoostTestDep());
            break;
        }
        default: {
            break;
        }
        }

        return deps.build();
    }

    public CxxTestType getDefaultTestType() {
        return DEFAULT_TEST_TYPE;
    }

    @Override
    public boolean hasFlavors(ImmutableSet<Flavor> flavors) {

        if (flavors.isEmpty()) {
            return true;
        }

        if (flavors.contains(CxxCompilationDatabase.COMPILATION_DATABASE)) {
            return true;
        }

        if (flavors.contains(CxxCompilationDatabase.UBER_COMPILATION_DATABASE)) {
            return true;
        }

        if (StripStyle.FLAVOR_DOMAIN.containsAnyOf(flavors)) {
            return true;
        }

        if (LinkerMapMode.FLAVOR_DOMAIN.containsAnyOf(flavors)) {
            return true;
        }

        for (Flavor flavor : cxxPlatforms.getFlavors()) {
            if (flavors.equals(ImmutableSet.of(flavor))) {
                return true;
            }
        }

        return false;
    }

    @Override
    public <A extends Arg, U> Optional<U> createMetadata(BuildTarget buildTarget, BuildRuleResolver resolver,
            A args, final Class<U> metadataClass) throws NoSuchBuildTargetException {
        if (!metadataClass.isAssignableFrom(CxxCompilationDatabaseDependencies.class)
                || !buildTarget.getFlavors().contains(CxxCompilationDatabase.COMPILATION_DATABASE)) {
            return Optional.empty();
        }
        return CxxDescriptionEnhancer
                .createCompilationDatabaseDependencies(buildTarget, cxxPlatforms, resolver, args)
                .map(metadataClass::cast);
    }

    @Override
    public boolean isVersionRoot(ImmutableSet<Flavor> flavors) {
        return true;
    }

    @SuppressFieldNotInitialized
    public static class Arg extends CxxBinaryDescription.Arg {
        public ImmutableSet<String> contacts = ImmutableSet.of();
        public ImmutableSet<Label> labels = ImmutableSet.of();
        public Optional<CxxTestType> framework;
        public ImmutableMap<String, String> env = ImmutableMap.of();
        public ImmutableList<String> args = ImmutableList.of();
        public Optional<Boolean> runTestSeparately;
        public Optional<Boolean> useDefaultTestMain;
        public Optional<Long> testRuleTimeoutMs;
        public ImmutableSortedSet<Path> resources = ImmutableSortedSet.of();
    }

}