blob: 4af2bf79bd92edaae03331bf133e8b349ec3ea09 [file] [log] [blame]
// Copyright (C) 2021 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.googlesource.gerrit.plugins.replication.pull.api;
import static com.google.gerrit.httpd.restapi.RestApiServlet.SC_UNPROCESSABLE_ENTITY;
import static com.googlesource.gerrit.plugins.replication.pull.api.HttpServletOps.checkAcceptHeader;
import static com.googlesource.gerrit.plugins.replication.pull.api.HttpServletOps.setResponse;
import static javax.servlet.http.HttpServletResponse.SC_BAD_REQUEST;
import static javax.servlet.http.HttpServletResponse.SC_CONFLICT;
import static javax.servlet.http.HttpServletResponse.SC_CREATED;
import static javax.servlet.http.HttpServletResponse.SC_FORBIDDEN;
import static javax.servlet.http.HttpServletResponse.SC_INTERNAL_SERVER_ERROR;
import static javax.servlet.http.HttpServletResponse.SC_OK;
import static javax.servlet.http.HttpServletResponse.SC_UNAUTHORIZED;
import com.google.common.base.Splitter;
import com.google.common.flogger.FluentLogger;
import com.google.gerrit.extensions.annotations.PluginName;
import com.google.gerrit.extensions.api.projects.HeadInput;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.extensions.restapi.IdString;
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.RestApiException;
import com.google.gerrit.extensions.restapi.TopLevelResource;
import com.google.gerrit.extensions.restapi.UnprocessableEntityException;
import com.google.gerrit.extensions.restapi.Url;
import com.google.gerrit.httpd.AllRequestFilter;
import com.google.gerrit.httpd.restapi.RestApiServlet;
import com.google.gerrit.json.OutputFormat;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.ProjectResource;
import com.google.gerrit.server.restapi.project.ProjectsCollection;
import com.google.gson.Gson;
import com.google.gson.JsonParseException;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.MalformedJsonException;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.TypeLiteral;
import com.googlesource.gerrit.plugins.replication.pull.api.FetchAction.Input;
import com.googlesource.gerrit.plugins.replication.pull.api.data.RevisionInput;
import com.googlesource.gerrit.plugins.replication.pull.api.exception.InitProjectException;
import java.io.BufferedReader;
import java.io.EOFException;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class PullReplicationFilter extends AllRequestFilter {
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
private FetchAction fetchAction;
private ApplyObjectAction applyObjectAction;
private ProjectInitializationAction projectInitializationAction;
private UpdateHeadAction updateHEADAction;
private ProjectDeletionAction projectDeletionAction;
private ProjectsCollection projectsCollection;
private Gson gson;
private Provider<CurrentUser> userProvider;
private String pluginName;
@Inject
public PullReplicationFilter(
FetchAction fetchAction,
ApplyObjectAction applyObjectAction,
ProjectInitializationAction projectInitializationAction,
UpdateHeadAction updateHEADAction,
ProjectDeletionAction projectDeletionAction,
ProjectsCollection projectsCollection,
Provider<CurrentUser> userProvider,
@PluginName String pluginName) {
this.fetchAction = fetchAction;
this.applyObjectAction = applyObjectAction;
this.projectInitializationAction = projectInitializationAction;
this.updateHEADAction = updateHEADAction;
this.projectDeletionAction = projectDeletionAction;
this.projectsCollection = projectsCollection;
this.userProvider = userProvider;
this.pluginName = pluginName;
this.gson = OutputFormat.JSON.newGsonBuilder().create();
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
if (!(request instanceof HttpServletRequest) || !(response instanceof HttpServletResponse)) {
chain.doFilter(request, response);
return;
}
HttpServletResponse httpResponse = (HttpServletResponse) response;
HttpServletRequest httpRequest = (HttpServletRequest) request;
try {
if (isFetchAction(httpRequest)) {
if (userProvider.get().isIdentifiedUser()) {
writeResponse(httpResponse, doFetch(httpRequest));
} else {
httpResponse.sendError(SC_UNAUTHORIZED);
}
} else if (isApplyObjectAction(httpRequest)) {
if (userProvider.get().isIdentifiedUser()) {
writeResponse(httpResponse, doApplyObject(httpRequest));
} else {
httpResponse.sendError(SC_UNAUTHORIZED);
}
} else if (isInitProjectAction(httpRequest)) {
if (userProvider.get().isIdentifiedUser()) {
if (!checkAcceptHeader(httpRequest, httpResponse)) {
return;
}
doInitProject(httpRequest, httpResponse);
} else {
httpResponse.sendError(SC_UNAUTHORIZED);
}
} else if (isUpdateHEADAction(httpRequest)) {
if (userProvider.get().isIdentifiedUser()) {
writeResponse(httpResponse, doUpdateHEAD(httpRequest));
} else {
httpResponse.sendError(SC_UNAUTHORIZED);
}
} else if (isDeleteProjectAction(httpRequest)) {
if (userProvider.get().isIdentifiedUser()) {
writeResponse(httpResponse, doDeleteProject(httpRequest));
} else {
httpResponse.sendError(SC_UNAUTHORIZED);
}
} else {
chain.doFilter(request, response);
}
} catch (AuthException e) {
RestApiServlet.replyError(
httpRequest, httpResponse, SC_FORBIDDEN, e.getMessage(), e.caching(), e);
} catch (MalformedJsonException | JsonParseException e) {
logger.atFine().withCause(e).log("REST call failed on JSON parsing");
RestApiServlet.replyError(
httpRequest, httpResponse, SC_BAD_REQUEST, "Invalid json in request", e);
} catch (BadRequestException e) {
RestApiServlet.replyError(httpRequest, httpResponse, SC_BAD_REQUEST, e.getMessage(), e);
} catch (UnprocessableEntityException e) {
RestApiServlet.replyError(
httpRequest, httpResponse, SC_UNPROCESSABLE_ENTITY, e.getMessage(), e.caching(), e);
} catch (ResourceConflictException e) {
RestApiServlet.replyError(
httpRequest, httpResponse, SC_CONFLICT, e.getMessage(), e.caching(), e);
} catch (InitProjectException | ResourceNotFoundException e) {
RestApiServlet.replyError(
httpRequest, httpResponse, SC_INTERNAL_SERVER_ERROR, e.getMessage(), e.caching(), e);
} catch (Exception e) {
throw new ServletException(e);
}
}
private void doInitProject(HttpServletRequest httpRequest, HttpServletResponse httpResponse)
throws RestApiException, IOException, PermissionBackendException {
String path = httpRequest.getRequestURI();
String projectName = Url.decode(path.substring(path.lastIndexOf('/') + 1));
if (projectInitializationAction.initProject(projectName)) {
setResponse(
httpResponse, HttpServletResponse.SC_CREATED, "Project " + projectName + " initialized");
return;
}
throw new InitProjectException(projectName);
}
@SuppressWarnings("unchecked")
private Response<Map<String, Object>> doApplyObject(HttpServletRequest httpRequest)
throws RestApiException, IOException, PermissionBackendException {
RevisionInput input = readJson(httpRequest, TypeLiteral.get(RevisionInput.class));
IdString id = getProjectName(httpRequest);
ProjectResource projectResource = projectsCollection.parse(TopLevelResource.INSTANCE, id);
return (Response<Map<String, Object>>) applyObjectAction.apply(projectResource, input);
}
@SuppressWarnings("unchecked")
private Response<String> doUpdateHEAD(HttpServletRequest httpRequest) throws Exception {
HeadInput input = readJson(httpRequest, TypeLiteral.get(HeadInput.class));
ProjectResource projectResource =
projectsCollection.parse(TopLevelResource.INSTANCE, getProjectName(httpRequest));
return (Response<String>) updateHEADAction.apply(projectResource, input);
}
@SuppressWarnings("unchecked")
private Response<String> doDeleteProject(HttpServletRequest httpRequest) throws Exception {
ProjectResource projectResource =
projectsCollection.parse(TopLevelResource.INSTANCE, getProjectName(httpRequest));
return (Response<String>)
projectDeletionAction.apply(projectResource, new ProjectDeletionAction.DeleteInput());
}
@SuppressWarnings("unchecked")
private Response<Map<String, Object>> doFetch(HttpServletRequest httpRequest)
throws IOException, RestApiException, PermissionBackendException {
Input input = readJson(httpRequest, TypeLiteral.get(Input.class));
IdString id = getProjectName(httpRequest);
ProjectResource projectResource = projectsCollection.parse(TopLevelResource.INSTANCE, id);
return (Response<Map<String, Object>>) fetchAction.apply(projectResource, input);
}
private <T> void writeResponse(HttpServletResponse httpResponse, Response<T> response)
throws IOException {
String responseJson = gson.toJson(response);
if (response.statusCode() == SC_OK || response.statusCode() == SC_CREATED) {
httpResponse.setContentType("application/json");
httpResponse.setStatus(response.statusCode());
PrintWriter writer = httpResponse.getWriter();
writer.print(new String(RestApiServlet.JSON_MAGIC));
writer.print(responseJson);
} else {
httpResponse.sendError(response.statusCode(), responseJson);
}
}
private <T> T readJson(HttpServletRequest httpRequest, TypeLiteral<T> typeLiteral)
throws IOException, BadRequestException {
try (BufferedReader br = httpRequest.getReader();
JsonReader json = new JsonReader(br)) {
try {
json.setLenient(true);
try {
json.peek();
} catch (EOFException e) {
throw new BadRequestException("Expected JSON object", e);
}
return gson.fromJson(json, typeLiteral.getType());
} finally {
try {
// Reader.close won't consume the rest of the input. Explicitly consume the request
// body.
br.skip(Long.MAX_VALUE);
} catch (Exception e) {
// ignore, e.g. trying to consume the rest of the input may fail if the request was
// cancelled
logger.atFine().withCause(e).log("Exception during the parsing of the request json");
}
}
}
}
/**
* Return project name from request URI. Request URI format:
* /a/projects/<project_name>/pull-replication~apply-object
*
* @param req
* @return project name
*/
private IdString getProjectName(HttpServletRequest req) {
String path = req.getRequestURI();
List<IdString> out = new ArrayList<>();
for (String p : Splitter.on('/').split(path)) {
out.add(IdString.fromUrl(p));
}
if (!out.isEmpty() && out.get(out.size() - 1).isEmpty()) {
out.remove(out.size() - 1);
}
return out.get(3);
}
private boolean isApplyObjectAction(HttpServletRequest httpRequest) {
return httpRequest.getRequestURI().endsWith("pull-replication~apply-object");
}
private boolean isFetchAction(HttpServletRequest httpRequest) {
return httpRequest.getRequestURI().endsWith("pull-replication~fetch");
}
private boolean isInitProjectAction(HttpServletRequest httpRequest) {
return httpRequest.getRequestURI().contains("pull-replication/init-project/");
}
private boolean isUpdateHEADAction(HttpServletRequest httpRequest) {
return httpRequest.getRequestURI().matches("(/a)?/projects/[^/]+/HEAD")
&& "PUT".equals(httpRequest.getMethod());
}
private boolean isDeleteProjectAction(HttpServletRequest httpRequest) {
return httpRequest.getRequestURI().endsWith(String.format("%s~delete-project", pluginName))
&& "DELETE".equals(httpRequest.getMethod());
}
}