|  | // Copyright (C) 2013 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.restapi.change; | 
|  |  | 
|  | import static com.google.gerrit.git.ObjectIds.abbreviateName; | 
|  | import static java.nio.charset.StandardCharsets.UTF_8; | 
|  |  | 
|  | import com.google.gerrit.extensions.restapi.BinaryResult; | 
|  | import com.google.gerrit.extensions.restapi.ResourceConflictException; | 
|  | import com.google.gerrit.extensions.restapi.ResourceNotFoundException; | 
|  | import com.google.gerrit.extensions.restapi.Response; | 
|  | import com.google.gerrit.extensions.restapi.RestReadView; | 
|  | import com.google.gerrit.server.change.RevisionResource; | 
|  | import com.google.gerrit.server.git.GitRepositoryManager; | 
|  | import com.google.inject.Inject; | 
|  | import java.io.IOException; | 
|  | import java.io.OutputStream; | 
|  | import java.text.SimpleDateFormat; | 
|  | import java.util.Calendar; | 
|  | import java.util.Locale; | 
|  | import java.util.zip.ZipEntry; | 
|  | import java.util.zip.ZipOutputStream; | 
|  | import org.eclipse.jgit.diff.DiffFormatter; | 
|  | import org.eclipse.jgit.lib.PersonIdent; | 
|  | import org.eclipse.jgit.lib.Repository; | 
|  | import org.eclipse.jgit.revwalk.RevCommit; | 
|  | import org.eclipse.jgit.revwalk.RevWalk; | 
|  | import org.eclipse.jgit.treewalk.filter.PathFilter; | 
|  | import org.kohsuke.args4j.Option; | 
|  |  | 
|  | public class GetPatch implements RestReadView<RevisionResource> { | 
|  | private final GitRepositoryManager repoManager; | 
|  |  | 
|  | private final String FILE_NOT_FOUND = "File not found: %s."; | 
|  |  | 
|  | @Option(name = "--zip") | 
|  | private boolean zip; | 
|  |  | 
|  | @Option(name = "--download") | 
|  | private boolean download; | 
|  |  | 
|  | @Option(name = "--path") | 
|  | private String path; | 
|  |  | 
|  | @Inject | 
|  | GetPatch(GitRepositoryManager repoManager) { | 
|  | this.repoManager = repoManager; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public Response<BinaryResult> apply(RevisionResource rsrc) | 
|  | throws ResourceConflictException, IOException, ResourceNotFoundException { | 
|  | final Repository repo = repoManager.openRepository(rsrc.getProject()); | 
|  | boolean close = true; | 
|  | try { | 
|  | final RevWalk rw = new RevWalk(repo); | 
|  | try { | 
|  | final RevCommit commit = rw.parseCommit(rsrc.getPatchSet().commitId()); | 
|  | RevCommit[] parents = commit.getParents(); | 
|  | if (parents.length > 1) { | 
|  | throw new ResourceConflictException("Revision has more than 1 parent."); | 
|  | } else if (parents.length == 0) { | 
|  | throw new ResourceConflictException("Revision has no parent."); | 
|  | } | 
|  | final RevCommit base = parents[0]; | 
|  | rw.parseBody(base); | 
|  |  | 
|  | BinaryResult bin = | 
|  | new BinaryResult() { | 
|  | @Override | 
|  | public void writeTo(OutputStream out) throws IOException { | 
|  | if (zip) { | 
|  | ZipOutputStream zos = new ZipOutputStream(out); | 
|  | ZipEntry e = new ZipEntry(fileName(rw, commit)); | 
|  | e.setTime(commit.getCommitTime() * 1000L); | 
|  | zos.putNextEntry(e); | 
|  | format(zos); | 
|  | zos.closeEntry(); | 
|  | zos.finish(); | 
|  | } else { | 
|  | format(out); | 
|  | } | 
|  | } | 
|  |  | 
|  | private void format(OutputStream out) throws IOException { | 
|  | // Only add header if no path is specified | 
|  | if (path == null) { | 
|  | out.write(formatEmailHeader(commit).getBytes(UTF_8)); | 
|  | } | 
|  | try (DiffFormatter fmt = new DiffFormatter(out)) { | 
|  | fmt.setRepository(repo); | 
|  | if (path != null) { | 
|  | fmt.setPathFilter(PathFilter.create(path)); | 
|  | } | 
|  | fmt.format(base.getTree(), commit.getTree()); | 
|  | fmt.flush(); | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public void close() throws IOException { | 
|  | rw.close(); | 
|  | repo.close(); | 
|  | } | 
|  | }; | 
|  |  | 
|  | if (path != null && bin.asString().isEmpty()) { | 
|  | throw new ResourceNotFoundException(String.format(FILE_NOT_FOUND, path)); | 
|  | } | 
|  |  | 
|  | if (zip) { | 
|  | bin.disableGzip() | 
|  | .setContentType("application/zip") | 
|  | .setAttachmentName(fileName(rw, commit) + ".zip"); | 
|  | } else { | 
|  | bin.base64() | 
|  | .setContentType("application/mbox") | 
|  | .setAttachmentName(download ? fileName(rw, commit) + ".base64" : null); | 
|  | } | 
|  |  | 
|  | close = false; | 
|  | return Response.ok(bin); | 
|  | } finally { | 
|  | if (close) { | 
|  | rw.close(); | 
|  | } | 
|  | } | 
|  | } finally { | 
|  | if (close) { | 
|  | repo.close(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | public GetPatch setPath(String path) { | 
|  | this.path = path; | 
|  | return this; | 
|  | } | 
|  |  | 
|  | private static String formatEmailHeader(RevCommit commit) { | 
|  | StringBuilder b = new StringBuilder(); | 
|  | PersonIdent author = commit.getAuthorIdent(); | 
|  | String subject = commit.getShortMessage(); | 
|  | String msg = commit.getFullMessage().substring(subject.length()); | 
|  | if (msg.startsWith("\n\n")) { | 
|  | msg = msg.substring(2); | 
|  | } | 
|  | b.append("From ") | 
|  | .append(commit.getName()) | 
|  | .append(' ') | 
|  | .append( | 
|  | "Mon Sep 17 00:00:00 2001\n") // Fixed timestamp to match output of C Git's format-patch | 
|  | .append("From: ") | 
|  | .append(author.getName()) | 
|  | .append(" <") | 
|  | .append(author.getEmailAddress()) | 
|  | .append(">\n") | 
|  | .append("Date: ") | 
|  | .append(formatDate(author)) | 
|  | .append('\n') | 
|  | .append("Subject: [PATCH] ") | 
|  | .append(subject) | 
|  | .append('\n') | 
|  | .append('\n') | 
|  | .append(msg); | 
|  | if (!msg.endsWith("\n")) { | 
|  | b.append('\n'); | 
|  | } | 
|  | return b.append("---\n\n").toString(); | 
|  | } | 
|  |  | 
|  | private static String formatDate(PersonIdent author) { | 
|  | SimpleDateFormat df = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss Z", Locale.US); | 
|  | df.setCalendar(Calendar.getInstance(author.getTimeZone(), Locale.US)); | 
|  | return df.format(author.getWhen()); | 
|  | } | 
|  |  | 
|  | private static String fileName(RevWalk rw, RevCommit commit) throws IOException { | 
|  | return abbreviateName(commit, rw.getObjectReader()) + ".diff"; | 
|  | } | 
|  | } |