com.google.devtools.build.lib.analysis.actions.ParameterFileWriteAction.java Source code

Java tutorial

Introduction

Here is the source code for com.google.devtools.build.lib.analysis.actions.ParameterFileWriteAction.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.analysis.actions;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.devtools.build.lib.actions.ActionExecutionContext;
import com.google.devtools.build.lib.actions.ActionOwner;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.Artifact.ArtifactExpander;
import com.google.devtools.build.lib.actions.ParameterFile.ParameterFileType;
import com.google.devtools.build.lib.analysis.actions.AbstractFileWriteAction.DeterministicWriter;
import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable;
import com.google.devtools.build.lib.util.Fingerprint;
import com.google.devtools.build.lib.util.Preconditions;
import com.google.devtools.build.lib.util.ShellEscaper;

import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.nio.charset.Charset;

/**
 * Action to write a parameter file for a {@link CommandLine}.
 */
@Immutable // if commandLine and charset are immutable
public final class ParameterFileWriteAction extends AbstractFileWriteAction {

    private static final String GUID = "45f678d8-e395-401e-8446-e795ccc6361f";

    private final CommandLine commandLine;
    private final ParameterFileType type;
    private final Charset charset;
    private final boolean hasInputArtifactToExpand;

    /**
     * Creates a new instance.
     *
     * @param owner the action owner
     * @param output the Artifact that will be created by executing this Action
     * @param commandLine the contents to be written to the file
     * @param type the type of the file
     * @param charset the charset of the file
     */
    public ParameterFileWriteAction(ActionOwner owner, Artifact output, CommandLine commandLine,
            ParameterFileType type, Charset charset) {
        this(owner, ImmutableList.<Artifact>of(), output, commandLine, type, charset);
    }

    /**
     * Creates a new instance.
     *
     * @param owner the action owner
     * @param inputs the list of TreeArtifacts that must be resolved and expanded before evaluating
     *     the contents of {@link commandLine}.
     * @param output the Artifact that will be created by executing this Action
     * @param commandLine the contents to be written to the file
     * @param type the type of the file
     * @param charset the charset of the file
     */
    public ParameterFileWriteAction(ActionOwner owner, Iterable<Artifact> inputs, Artifact output,
            CommandLine commandLine, ParameterFileType type, Charset charset) {
        super(owner, ImmutableList.copyOf(inputs), output, false);
        this.commandLine = commandLine;
        this.type = type;
        this.charset = charset;
        this.hasInputArtifactToExpand = !Iterables.isEmpty(inputs);
    }

    /**
     * Returns the list of options written to the parameter file. Don't use this
     * method outside tests - the list is often huge, resulting in significant
     * garbage collection overhead.
     */
    @VisibleForTesting
    public Iterable<String> getContents() {
        Preconditions.checkState(!hasInputArtifactToExpand,
                "This action contains a CommandLine with TreeArtifacts: %s, which must be expanded using "
                        + "ArtifactExpander first before we can evaluate the CommandLine.",
                getInputs());
        return commandLine.arguments();
    }

    @VisibleForTesting
    public Iterable<String> getContents(ArtifactExpander artifactExpander) {
        return commandLine.arguments(artifactExpander);
    }

    @Override
    public DeterministicWriter newDeterministicWriter(ActionExecutionContext ctx) {
        return new ParamFileWriter(Preconditions.checkNotNull(ctx.getArtifactExpander()));
    }

    private class ParamFileWriter implements DeterministicWriter {
        private final ArtifactExpander artifactExpander;

        ParamFileWriter(ArtifactExpander artifactExpander) {
            this.artifactExpander = artifactExpander;
        }

        @Override
        public void writeOutputFile(OutputStream out) throws IOException {
            Iterable<String> arguments = commandLine.arguments(artifactExpander);

            switch (type) {
            case SHELL_QUOTED:
                writeContentQuoted(out, arguments);
                break;
            case UNQUOTED:
                writeContentUnquoted(out, arguments);
                break;
            default:
                throw new AssertionError();
            }
        }

        /**
         * Writes the arguments from the list into the parameter file.
         */
        private void writeContentUnquoted(OutputStream outputStream, Iterable<String> arguments)
                throws IOException {
            OutputStreamWriter out = new OutputStreamWriter(outputStream, charset);
            for (String line : arguments) {
                out.write(line);
                out.write('\n');
            }
            out.flush();
        }

        /**
         * Writes the arguments from the list into the parameter file with shell
         * quoting (if required).
         */
        private void writeContentQuoted(OutputStream outputStream, Iterable<String> arguments) throws IOException {
            OutputStreamWriter out = new OutputStreamWriter(outputStream, charset);
            for (String line : ShellEscaper.escapeAll(arguments)) {
                out.write(line);
                out.write('\n');
            }
            out.flush();
        }
    }

    @Override
    protected String computeKey() {
        Fingerprint f = new Fingerprint();
        f.addString(GUID);
        f.addString(String.valueOf(makeExecutable));
        f.addStrings(commandLine.arguments());
        return f.hexDigestAndReset();
    }
}