com.google.devtools.build.lib.rules.extra.ExtraAction.java Source code

Java tutorial

Introduction

Here is the source code for com.google.devtools.build.lib.rules.extra.ExtraAction.java

Source

// Copyright 2014 The Bazel Authors. All rights reserved.
//
// 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.google.devtools.build.lib.rules.extra;

import com.google.common.base.Function;
import com.google.common.base.Predicates;
import com.google.common.collect.Collections2;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.actions.AbstractAction;
import com.google.devtools.build.lib.actions.Action;
import com.google.devtools.build.lib.actions.ActionExecutionContext;
import com.google.devtools.build.lib.actions.ActionExecutionException;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.ArtifactResolver;
import com.google.devtools.build.lib.actions.DelegateSpawn;
import com.google.devtools.build.lib.actions.PackageRootResolutionException;
import com.google.devtools.build.lib.actions.PackageRootResolver;
import com.google.devtools.build.lib.actions.Spawn;
import com.google.devtools.build.lib.actions.SpawnActionContext;
import com.google.devtools.build.lib.analysis.actions.CommandLine;
import com.google.devtools.build.lib.analysis.actions.SpawnAction;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder;
import com.google.devtools.build.lib.collect.nestedset.Order;
import com.google.devtools.build.lib.util.Preconditions;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.PathFragment;
import java.io.IOException;
import java.util.Collection;
import java.util.Map;
import java.util.Set;
import javax.annotation.Nullable;
import javax.annotation.concurrent.GuardedBy;

/**
 * Action used by extra_action rules to create an action that shadows an existing action. Runs a
 * command-line using {@link SpawnActionContext} for executions.
 */
public final class ExtraAction extends SpawnAction {
    private final Action shadowedAction;
    private final boolean createDummyOutput;
    private final ImmutableMap<PathFragment, Artifact> runfilesManifests;
    private final ImmutableSet<Artifact> extraActionInputs;
    // This can be read/written from multiple threads, and so accesses should be synchronized.
    @GuardedBy("this")
    private boolean inputsKnown;

    /**
     * A long way to say (ExtraAction xa) -> xa.getShadowedAction().
     */
    public static final Function<ExtraAction, Action> GET_SHADOWED_ACTION = new Function<ExtraAction, Action>() {
        @Nullable
        @Override
        public Action apply(@Nullable ExtraAction extraAction) {
            return extraAction != null ? extraAction.getShadowedAction() : null;
        }
    };

    public ExtraAction(ImmutableSet<Artifact> extraActionInputs, Map<PathFragment, Artifact> runfilesManifests,
            Collection<Artifact> outputs, Action shadowedAction, boolean createDummyOutput, CommandLine argv,
            Map<String, String> environment, Set<String> clientEnvironmentVariables,
            Map<String, String> executionInfo, String progressMessage, String mnemonic) {
        super(shadowedAction.getOwner(), ImmutableList.<Artifact>of(),
                createInputs(shadowedAction.getInputs(), extraActionInputs), outputs,
                AbstractAction.DEFAULT_RESOURCE_SET, argv, ImmutableMap.copyOf(environment),
                ImmutableSet.copyOf(clientEnvironmentVariables), ImmutableMap.copyOf(executionInfo),
                progressMessage, getManifests(shadowedAction), mnemonic, false, null);
        this.shadowedAction = shadowedAction;
        this.runfilesManifests = ImmutableMap.copyOf(runfilesManifests);
        this.createDummyOutput = createDummyOutput;

        this.extraActionInputs = extraActionInputs;
        inputsKnown = shadowedAction.inputsKnown();
        if (createDummyOutput) {
            // Expecting just a single dummy file in the outputs.
            Preconditions.checkArgument(outputs.size() == 1, outputs);
        }
    }

    private static ImmutableMap<PathFragment, Artifact> getManifests(Action shadowedAction) {
        // If the shadowed action is a SpawnAction, then we also add the input manifests to this
        // action's input manifests.
        // TODO(bazel-team): Also handle other action classes correctly.
        if (shadowedAction instanceof SpawnAction) {
            return ((SpawnAction) shadowedAction).getInputManifests();
        }
        return ImmutableMap.of();
    }

    @Override
    public boolean discoversInputs() {
        return shadowedAction.discoversInputs();
    }

    @Nullable
    @Override
    public Iterable<Artifact> discoverInputs(ActionExecutionContext actionExecutionContext)
            throws ActionExecutionException, InterruptedException {
        Preconditions.checkState(discoversInputs(), this);
        // We need to update our inputs to take account of any additional
        // inputs the shadowed action may need to do its work.
        if (shadowedAction.discoversInputs() && shadowedAction instanceof AbstractAction) {
            Iterable<Artifact> additionalInputs = ((AbstractAction) shadowedAction)
                    .getInputFilesForExtraAction(actionExecutionContext);
            updateInputs(createInputs(additionalInputs, extraActionInputs));
            return ImmutableSet.copyOf(additionalInputs);
        }
        return null;
    }

    @Override
    public synchronized boolean inputsKnown() {
        return inputsKnown;
    }

    private static NestedSet<Artifact> createInputs(Iterable<Artifact> shadowedActionInputs,
            ImmutableSet<Artifact> extraActionInputs) {
        NestedSetBuilder<Artifact> result = new NestedSetBuilder<>(Order.STABLE_ORDER);
        if (shadowedActionInputs instanceof NestedSet) {
            result.addTransitive((NestedSet<Artifact>) shadowedActionInputs);
        } else {
            result.addAll(shadowedActionInputs);
        }
        return result.addAll(extraActionInputs).build();
    }

    @Override
    public synchronized void updateInputs(Iterable<Artifact> discoveredInputs) {
        setInputs(discoveredInputs);
        inputsKnown = true;
    }

    @Nullable
    @Override
    public Iterable<Artifact> resolveInputsFromCache(ArtifactResolver artifactResolver,
            PackageRootResolver resolver, Collection<PathFragment> inputPaths)
            throws PackageRootResolutionException, InterruptedException {
        // We update the inputs directly from the shadowed action.
        Set<PathFragment> extraActionPathFragments = ImmutableSet
                .copyOf(Artifact.asPathFragments(extraActionInputs));
        return shadowedAction.resolveInputsFromCache(artifactResolver, resolver,
                Collections2.filter(inputPaths, Predicates.in(extraActionPathFragments)));
    }

    /**
     * @InheritDoc
     *
     * This method calls in to {@link AbstractAction#getInputFilesForExtraAction} and
     * {@link Action#getExtraActionInfo} of the action being shadowed from the thread executing this
     * ExtraAction. It assumes these methods are safe to call from a different thread than the thread
     * responsible for the execution of the action being shadowed.
     */
    @Override
    public void execute(ActionExecutionContext actionExecutionContext)
            throws ActionExecutionException, InterruptedException {
        // PHASE 2: execution of extra_action.

        super.execute(actionExecutionContext);

        // PHASE 3: create dummy output.
        // If the user didn't specify output, we need to create dummy output
        // to make blaze schedule this action.
        if (createDummyOutput) {
            for (Artifact output : getOutputs()) {
                try {
                    FileSystemUtils.touchFile(output.getPath());
                } catch (IOException e) {
                    throw new ActionExecutionException(e.getMessage(), e, this, false);
                }
            }
        }
        synchronized (this) {
            inputsKnown = true;
        }
    }

    /**
     * The spawn command for ExtraAction needs to be slightly modified from
     * regular SpawnActions:
     * -the extraActionInfo file needs to be added to the list of inputs.
     * -the extraActionInfo file that is an output file of this task is created
     * before the SpawnAction so should not be listed as one of its outputs.
     */
    // TODO(bazel-team): Add more tests that execute this code path!
    @Override
    public Spawn getSpawn(Map<String, String> clientEnv) {
        final Spawn base = super.getSpawn(clientEnv);
        return new DelegateSpawn(base) {
            @Override
            public ImmutableMap<PathFragment, Artifact> getRunfilesManifests() {
                ImmutableMap.Builder<PathFragment, Artifact> builder = ImmutableMap.builder();
                builder.putAll(super.getRunfilesManifests());
                builder.putAll(runfilesManifests);
                return builder.build();
            }

            @Override
            public String getMnemonic() {
                return ExtraAction.this.getMnemonic();
            }
        };
    }

    /**
     * Returns the action this extra action is 'shadowing'.
     */
    public Action getShadowedAction() {
        return shadowedAction;
    }
}