depot/third_party/copybara/java/com/google/copybara/InfoCmd.java

167 lines
7.1 KiB
Java
Raw Normal View History

/*
* Copyright (C) 2018 Google 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.google.copybara;
import com.beust.jcommander.Parameters;
import com.google.common.base.Ascii;
import com.google.common.base.Preconditions;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.common.collect.Iterables;
import com.google.copybara.Info.MigrationReference;
import com.google.copybara.config.Config;
import com.google.copybara.config.Migration;
import com.google.copybara.exception.RepoException;
import com.google.copybara.exception.ValidationException;
import com.google.copybara.monitor.EventMonitor.InfoFinishedEvent;
import com.google.copybara.util.ExitCode;
import com.google.copybara.util.TablePrinter;
import com.google.copybara.util.console.Console;
import java.io.IOException;
import java.time.format.DateTimeFormatter;
import java.util.Comparator;
/**
* Reads the last migrated revision in the origin and destination.
*/
@Parameters(separators = "=",
commandDescription = "Reads the last migrated revision in the origin and destination.")
public class InfoCmd implements CopybaraCmd {
private static final int REVISION_MAX_LENGTH = 15;
private static final int DESCRIPTION_MAX_LENGTH = 80;
private static final int AUTHOR_MAX_LENGTH = 40;
private static final DateTimeFormatter DATE_FORMATTER =
DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
private final ConfigLoaderProvider configLoaderProvider;
private final ContextProvider contextProvider;
public InfoCmd(ConfigLoaderProvider configLoaderProvider, ContextProvider contextProvider) {
this.configLoaderProvider = Preconditions.checkNotNull(configLoaderProvider);
this.contextProvider = Preconditions.checkNotNull(contextProvider);
}
@Override
public ExitCode run(CommandEnv commandEnv)
throws ValidationException, IOException, RepoException {
ConfigFileArgs configFileArgs = commandEnv.parseConfigFileArgs(this, /*useSourceRef*/false);
Console console = commandEnv.getOptions().get(GeneralOptions.class).console();
Config config = configLoaderProvider
.newLoader(configFileArgs.getConfigPath(), configFileArgs.getSourceRef())
.load(console);
if (configFileArgs.hasWorkflowName()) {
ImmutableMap<String, String> context =
contextProvider.getContext(config, configFileArgs, configLoaderProvider, console);
info(commandEnv.getOptions(), config, configFileArgs.getWorkflowName(), context);
} else {
showAllMigrations(commandEnv, config);
}
return ExitCode.SUCCESS;
}
private void showAllMigrations(CommandEnv commandEnv, Config config) {
TablePrinter table = new TablePrinter("Name", "Origin", "Destination", "Mode", "Description");
for (Migration m :
config.getMigrations().values().stream()
.sorted(Comparator.comparing(Migration::getName))
.collect(ImmutableList.toImmutableList())) {
table.addRow(
m.getName(),
prettyOriginDestination(m.getOriginDescription()),
prettyOriginDestination(m.getDestinationDescription()),
m.getModeString(),
Strings.nullToEmpty(m.getDescription()));
}
Console console = commandEnv.getOptions().get(GeneralOptions.class).console();
for (String line : table.build()) {
console.info(line);
}
console.info("To get information about the state of any migration run:\n\n"
+ " copybara info " + config.getLocation() + " [workflow_name]"
+ "\n");
}
private static String prettyOriginDestination(ImmutableSetMultimap<String, String> desc) {
return Iterables.getOnlyElement(desc.get("type"))
+ (desc.containsKey("url") ? " (" + Iterables.getOnlyElement(desc.get("url")) + ")" : "");
}
/** Retrieves the {@link Info} of the {@code migrationName} and prints it to the console. */
private static void info(
Options options, Config config, String migrationName, ImmutableMap<String, String> context)
throws ValidationException, RepoException {
@SuppressWarnings("unchecked")
Info<? extends Revision> info = getInfo(migrationName, config);
Console console = options.get(GeneralOptions.class).console();
int outputSize = 0;
for (MigrationReference<? extends Revision> migrationRef : info.migrationReferences()) {
console.info(String.format(
"'%s': last_migrated %s - last_available %s.",
migrationRef.getLabel(),
migrationRef.getLastMigrated() != null
? migrationRef.getLastMigrated().asString() : "None",
migrationRef.getLastAvailableToMigrate() != null
? migrationRef.getLastAvailableToMigrate().asString() : "None"));
ImmutableList<? extends Change<? extends Revision>> availableToMigrate =
migrationRef.getAvailableToMigrate();
int outputLimit = options.get(GeneralOptions.class).getOutputLimit();
if (!availableToMigrate.isEmpty()) {
console.infoFmt(
"Available changes %s:",
availableToMigrate.size() <= outputLimit
? String.format("(%d)", availableToMigrate.size())
: String.format(
"(showing only first %d out of %d)", outputLimit, availableToMigrate.size()));
TablePrinter table = new TablePrinter("Date", "Revision", "Description", "Author");
for (Change<? extends Revision> change :
Iterables.limit(availableToMigrate, outputLimit)) {
outputSize++;
table.addRow(
change.getDateTime().format(DATE_FORMATTER),
Ascii.truncate(change.getRevision().asString(), REVISION_MAX_LENGTH, ""),
Ascii.truncate(change.firstLineMessage(), DESCRIPTION_MAX_LENGTH, "..."),
Ascii.truncate(change.getAuthor().toString(), AUTHOR_MAX_LENGTH, "..."));
}
for (String line : table.build()) {
console.info(line);
}
}
if (outputSize > 100) {
console.infoFmt(
"Use %s to limit the output of the command.", GeneralOptions.OUTPUT_LIMIT_FLAG);
}
}
options.get(GeneralOptions.class).eventMonitor().onInfoFinished(
new InfoFinishedEvent(info, context));
}
/** Returns the {@link Info} of the {@code migrationName}. */
private static Info<? extends Revision> getInfo(String migrationName, Config config)
throws ValidationException, RepoException {
return config.getMigration(migrationName).getInfo();
}
@Override
public String name() {
return "info";
}
}