blob: e7289ac8e82aef4ded3c3c4504a00f60b0c2d825 [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.googlesource.gerrit.plugins.replication.pull.PullReplicationLogger.repLog;
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_FORBIDDEN;
import com.google.common.base.Strings;
import com.google.common.flogger.FluentLogger;
import com.google.common.net.MediaType;
import com.google.gerrit.entities.Project;
import com.google.gerrit.entities.RefNames;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.extensions.restapi.Url;
import com.google.gerrit.index.project.ProjectIndexer;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.permissions.GlobalPermission;
import com.google.gerrit.server.permissions.PermissionBackend;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.ProjectCache;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.Singleton;
import com.googlesource.gerrit.plugins.replication.LocalFS;
import com.googlesource.gerrit.plugins.replication.pull.GerritConfigOps;
import com.googlesource.gerrit.plugins.replication.pull.api.data.RevisionsInput;
import com.googlesource.gerrit.plugins.replication.pull.api.exception.MissingParentObjectException;
import com.googlesource.gerrit.plugins.replication.pull.api.exception.RefUpdateException;
import com.googlesource.gerrit.plugins.replication.pull.api.util.PayloadSerDes;
import java.io.IOException;
import java.util.Arrays;
import java.util.Objects;
import java.util.Optional;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jgit.transport.URIish;
@Singleton
public class ProjectInitializationAction extends HttpServlet {
private static final long serialVersionUID = 1L;
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
public static final String PROJECT_NAME = "project-name";
private final GerritConfigOps gerritConfigOps;
private final Provider<CurrentUser> userProvider;
private final PermissionBackend permissionBackend;
private final ProjectIndexer projectIndexer;
private final ApplyObjectCommand applyObjectCommand;
private final ProjectCache projectCache;
@Inject
ProjectInitializationAction(
GerritConfigOps gerritConfigOps,
Provider<CurrentUser> userProvider,
PermissionBackend permissionBackend,
ProjectIndexer projectIndexer,
ApplyObjectCommand applyObjectCommand,
ProjectCache projectCache) {
this.gerritConfigOps = gerritConfigOps;
this.userProvider = userProvider;
this.permissionBackend = permissionBackend;
this.projectIndexer = projectIndexer;
this.applyObjectCommand = applyObjectCommand;
this.projectCache = projectCache;
}
@Override
protected void doPut(
HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse)
throws ServletException, IOException {
if (!checkAcceptHeader(httpServletRequest, httpServletResponse)) {
return;
}
String gitRepositoryName = getGitRepositoryName(httpServletRequest);
try {
boolean initProjectStatus;
String contentType = httpServletRequest.getContentType();
if (checkContentType(contentType, MediaType.JSON_UTF_8)) {
// init project request includes project configuration in JSON format.
initProjectStatus = initProjectWithConfiguration(httpServletRequest, gitRepositoryName);
} else if (checkContentType(contentType, MediaType.PLAIN_TEXT_UTF_8)) {
// init project request does not include project configuration.
initProjectStatus = initProject(gitRepositoryName);
} else {
setResponse(
httpServletResponse,
SC_BAD_REQUEST,
String.format(
"Invalid Content Type. Only %s or %s is supported.",
MediaType.JSON_UTF_8.toString(), MediaType.PLAIN_TEXT_UTF_8.toString()));
return;
}
if (initProjectStatus) {
setResponse(
httpServletResponse,
HttpServletResponse.SC_CREATED,
"Project " + gitRepositoryName + " initialized");
return;
}
setResponse(
httpServletResponse,
HttpServletResponse.SC_INTERNAL_SERVER_ERROR,
"Cannot initialize project " + gitRepositoryName);
} catch (BadRequestException | IllegalArgumentException e) {
logExceptionAndUpdateResponse(httpServletResponse, e, SC_BAD_REQUEST, gitRepositoryName);
} catch (RefUpdateException | MissingParentObjectException e) {
logExceptionAndUpdateResponse(httpServletResponse, e, SC_CONFLICT, gitRepositoryName);
} catch (AuthException | PermissionBackendException e) {
logExceptionAndUpdateResponse(httpServletResponse, e, SC_FORBIDDEN, gitRepositoryName);
}
}
public boolean initProject(String gitRepositoryName)
throws AuthException, PermissionBackendException {
if (initProject(gitRepositoryName, true)) {
repLog.info("Init project API from {}", gitRepositoryName);
return true;
}
return false;
}
private boolean initProjectWithConfiguration(
HttpServletRequest httpServletRequest, String gitRepositoryName)
throws AuthException, PermissionBackendException, IOException, BadRequestException,
MissingParentObjectException, RefUpdateException {
RevisionsInput input = PayloadSerDes.parseRevisionsInput(httpServletRequest);
validateInput(input);
if (!initProject(gitRepositoryName, false)) {
return false;
}
String projectName = gitRepositoryName.replace(".git", "");
applyObjectCommand.applyObjects(
Project.nameKey(projectName),
input.getRefName(),
input.getRevisionsData(),
input.getLabel(),
input.getEventCreatedOn());
projectCache.onCreateProject(Project.nameKey(projectName));
repLog.info(
"Init project API from {} for {}:{} - {}",
input.getLabel(),
projectName,
input.getRefName(),
Arrays.toString(input.getRevisionsData()));
return true;
}
private boolean initProject(String gitRepositoryName, boolean needsProjectReindexing)
throws AuthException, PermissionBackendException {
// When triggered internally(for example by consuming stream events) user is not provided
// and internal user is returned. Project creation should be always allowed for internal user.
if (!userProvider.get().isInternalUser()) {
permissionBackend.user(userProvider.get()).check(GlobalPermission.CREATE_PROJECT);
}
Optional<URIish> maybeUri = gerritConfigOps.getGitRepositoryURI(gitRepositoryName);
if (!maybeUri.isPresent()) {
logger.atSevere().log("Cannot initialize project '%s'", gitRepositoryName);
return false;
}
LocalFS localFS = new LocalFS(maybeUri.get());
Project.NameKey projectNameKey = Project.NameKey.parse(gitRepositoryName);
if (localFS.createProject(projectNameKey, RefNames.HEAD)) {
if (needsProjectReindexing) {
projectIndexer.index(projectNameKey);
}
return true;
}
return false;
}
private void validateInput(RevisionsInput input) {
if (Strings.isNullOrEmpty(input.getLabel())) {
throw new IllegalArgumentException("Source label cannot be null or empty");
}
if (!Objects.equals(input.getRefName(), RefNames.REFS_CONFIG)) {
throw new IllegalArgumentException(
String.format("Ref-update refname should be %s", RefNames.REFS_CONFIG));
}
input.validate();
}
private String getGitRepositoryName(HttpServletRequest httpServletRequest) {
String path = httpServletRequest.getRequestURI();
return Url.decode(path.substring(path.lastIndexOf('/') + 1));
}
private void logExceptionAndUpdateResponse(
HttpServletResponse httpServletResponse,
Exception e,
int statusCode,
String gitRepositoryName)
throws IOException {
repLog.error("Init Project API FAILED for {}", gitRepositoryName, e);
setResponse(httpServletResponse, statusCode, e.getMessage());
}
private boolean checkContentType(String contentType, MediaType mediaType) {
try {
return MediaType.parse(contentType).is(mediaType);
} catch (Exception e) {
return false;
}
}
}