All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.patch.MagicFile Maven / Gradle / Ivy

There is a newer version: 3.11.0
Show newest version
// Copyright (C) 2020 The Android Open Source Project
//
// 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.gerrit.server.patch;

import com.google.auto.value.AutoValue;
import com.google.common.base.CharMatcher;
import com.google.gerrit.git.ObjectIds;
import java.io.IOException;
import java.text.SimpleDateFormat;
import org.eclipse.jgit.lib.AnyObjectId;
import org.eclipse.jgit.lib.ObjectReader;
import org.eclipse.jgit.lib.PersonIdent;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;

/** Representation of a magic file which appears as a file with content to Gerrit users. */
@AutoValue
public abstract class MagicFile {

  public static MagicFile forCommitMessage(ObjectReader reader, AnyObjectId commitId)
      throws IOException {
    try (RevWalk rw = new RevWalk(reader)) {
      RevCommit c;
      if (commitId instanceof RevCommit) {
        c = (RevCommit) commitId;
      } else {
        c = rw.parseCommit(commitId);
      }

      String header = createCommitMessageHeader(reader, rw, c);
      String message = c.getFullMessage();
      return MagicFile.builder().generatedContent(header).modifiableContent(message).build();
    }
  }

  private static String createCommitMessageHeader(ObjectReader reader, RevWalk rw, RevCommit c)
      throws IOException {
    StringBuilder b = new StringBuilder();
    switch (c.getParentCount()) {
      case 0:
        break;
      case 1:
        {
          RevCommit p = c.getParent(0);
          rw.parseBody(p);
          b.append("Parent:     ");
          b.append(abbreviateName(p, reader));
          b.append(" (");
          b.append(p.getShortMessage());
          b.append(")\n");
          break;
        }
      default:
        for (int i = 0; i < c.getParentCount(); i++) {
          RevCommit p = c.getParent(i);
          rw.parseBody(p);
          b.append(i == 0 ? "Merge Of:   " : "            ");
          b.append(abbreviateName(p, reader));
          b.append(" (");
          b.append(p.getShortMessage());
          b.append(")\n");
        }
    }
    appendPersonIdent(b, "Author", c.getAuthorIdent());
    appendPersonIdent(b, "Commit", c.getCommitterIdent());
    b.append("\n");
    return b.toString();
  }

  public static MagicFile forMergeList(
      ComparisonType comparisonType, ObjectReader reader, AnyObjectId commitId) throws IOException {
    try (RevWalk rw = new RevWalk(reader)) {
      RevCommit c = rw.parseCommit(commitId);
      StringBuilder b = new StringBuilder();
      switch (c.getParentCount()) {
        case 0:
          break;
        case 1:
          {
            break;
          }
        default:
          int uninterestingParent =
              comparisonType.isAgainstParent() ? comparisonType.getParentNum().get() : 1;

          b.append("Merge List:\n\n");
          for (RevCommit commit : MergeListBuilder.build(rw, c, uninterestingParent)) {
            b.append("* ");
            b.append(abbreviateName(commit, reader));
            b.append(" ");
            b.append(commit.getShortMessage());
            b.append("\n");
          }
      }
      return MagicFile.builder().generatedContent(b.toString()).build();
    }
  }

  private static String abbreviateName(RevCommit p, ObjectReader reader) throws IOException {
    return ObjectIds.abbreviateName(p, 8, reader);
  }

  private static void appendPersonIdent(StringBuilder b, String field, PersonIdent person) {
    if (person != null) {
      b.append(field).append(":    ");
      if (person.getName() != null) {
        b.append(" ");
        b.append(person.getName());
      }
      if (person.getEmailAddress() != null) {
        b.append(" <");
        b.append(person.getEmailAddress());
        b.append(">");
      }
      b.append("\n");

      SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss ZZZ");
      sdf.setTimeZone(person.getTimeZone());
      b.append(field).append("Date: ");
      b.append(sdf.format(person.getWhen()));
      b.append("\n");
    }
  }

  /** Generated part of the file. Any generated contents should go here. Can be empty. */
  public abstract String generatedContent();

  /**
   * Non-generated part of the file. This should correspond to some actual content derived from
   * somewhere else which can also be modified (e.g. by suggested fixes). Can be empty.
   */
  public abstract String modifiableContent();

  /** Whole content of the file as it appears to users. */
  public String getFileContent() {
    return generatedContent() + modifiableContent();
  }

  /** Returns the start line of the modifiable content. Assumes that line counting starts at 1. */
  public int getStartLineOfModifiableContent() {
    int numHeaderLines = CharMatcher.is('\n').countIn(generatedContent());
    // Lines start at 1 and not 0. -> Add 1.
    return 1 + numHeaderLines;
  }

  static Builder builder() {
    return new AutoValue_MagicFile.Builder().generatedContent("").modifiableContent("");
  }

  @AutoValue.Builder
  abstract static class Builder {

    /** See {@link #generatedContent()}. Use an empty string to denote no such content. */
    public abstract Builder generatedContent(String content);

    /** See {@link #modifiableContent()}. Use an empty string to denote no such content. */
    public abstract Builder modifiableContent(String content);

    abstract String generatedContent();

    abstract String modifiableContent();

    abstract MagicFile autoBuild();

    public MagicFile build() {
      // Normalize each content part to end with a newline character, which simplifies further
      // handling.
      if (!generatedContent().isEmpty() && !generatedContent().endsWith("\n")) {
        generatedContent(generatedContent() + "\n");
      }
      if (!modifiableContent().isEmpty() && !modifiableContent().endsWith("\n")) {
        modifiableContent(modifiableContent() + "\n");
      }
      return autoBuild();
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy