com.mooltiverse.oss.nyx.services.gitlab.RESTv4 Maven / Gradle / Ivy
/*
* Copyright 2020 Mooltiverse
*
* 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.mooltiverse.oss.nyx.services.gitlab;
import static com.mooltiverse.oss.nyx.log.Markers.SERVICE;
import static com.mooltiverse.oss.nyx.services.gitlab.GitLab.logger;
import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Paths;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLEncoder;
import java.net.http.HttpClient;
import java.net.http.HttpClient.Redirect;
import java.net.http.HttpClient.Version;
import java.net.http.HttpRequest;
import java.net.http.HttpRequest.BodyPublishers;
import java.net.http.HttpResponse;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.mooltiverse.oss.nyx.entities.Attachment;
import com.mooltiverse.oss.nyx.io.TransportException;
import com.mooltiverse.oss.nyx.services.SecurityException;
/**
* The REST API v4 implementation as per GitLab API.
*/
class RESTv4 extends API {
/**
* The suggested content type to set in the 'content-type' header, as per GitHub API.
*/
private static final String CONTENT_TYPE = "application/json";
/**
* The security token to use to authenticate to the remote service.
* If {@code null} or invalid all authentication protected operations will fail with a
* {@link SecurityException}
*/
private final String authenticationToken;
/**
* The instance of a standard client builder with all reusable attributes set. Each client can
* be created from a copy of this builder.
*/
protected HttpClient.Builder clientBuilderTemplate = null;
/**
* The default API base URL, as per GitLab API.
* This value can be overridden by passing an entry with the {@link #BASE_URI_OPTION_NAME} key
* in the options map used to get an instance of this service.
* Value: {@value}
*/
public static final String BASE_URI_DEFAULT_VALUE = "https://gitlab.com/api/v4";
/**
* Builds this object using the given uri to use as the remote endpoint.
*
* @param uri the remote API endpoint. It can't be {@code null}
* @param authenticationToken the token (Personal Access Token, OAuth) to use for authentication. It may be {@code null}.
* If {@code null} the service will not be able to authenticate and perform any of the authentication protected operations.
*/
RESTv4(URI uri, String authenticationToken) {
super(uri);
this.authenticationToken = authenticationToken;
}
/**
* {@inheritDoc}
*/
@Override
Map createRepository(String name, String description, boolean restricted, boolean initialize)
throws TransportException, SecurityException {
// See: https://docs.gitlab.com/ee/api/projects.html#create-project
Objects.requireNonNull(name, "The name of the repository to create cannot be null");
URI uri = newRequestURI("/projects");
Map requestParameters = new HashMap();
requestParameters.put("name", name);
requestParameters.put("description", description);
requestParameters.put("visibility", restricted ? "private" : "public");
requestParameters.put("initialize_with_readme", initialize);
HttpResponse response = null;
try {
response = sendRequest(getRequestBuilder(true).uri(uri).POST(BodyPublishers.ofString(new ObjectMapper().writeValueAsString(requestParameters))).build());
}
catch (IOException | InterruptedException e) {
throw new TransportException(e);
}
// See: https://docs.gitlab.com/ee/api/#status-codes
if (response.statusCode() != 201) {
if (response.statusCode() == 401 || response.statusCode() == 403)
throw new SecurityException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
else throw new TransportException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
}
return unmarshalJSONBody(response.body());
}
/**
* {@inheritDoc}
*/
@Override
void deleteRepository(String name)
throws TransportException, SecurityException {
// See: https://docs.gitlab.com/ee/api/projects.html#delete-project
Objects.requireNonNull(name, "The name of the repository to delete cannot be null");
URI uri = newRequestURI("/projects/"+URLEncoder.encode(name, StandardCharsets.UTF_8));
HttpResponse response = null;
try {
response = sendRequest(getRequestBuilder(true).uri(uri).DELETE().build());
}
catch (IOException | InterruptedException e) {
throw new TransportException(e);
}
// See: https://docs.gitlab.com/ee/api/#status-codes
if (response.statusCode() != 202 && response.statusCode() != 204) {
if (response.statusCode() == 403)
throw new SecurityException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
else throw new TransportException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
}
}
/**
* {@inheritDoc}
*/
@Override
synchronized HttpClient.Builder getClientBuilder() {
if (clientBuilderTemplate == null) {
clientBuilderTemplate = HttpClient.newBuilder().version(Version.HTTP_1_1).followRedirects(Redirect.NORMAL);
}
return clientBuilderTemplate;
}
/**
* {@inheritDoc}
*/
@Override
Map getReleaseByTag(String owner, String repository, String tag)
throws SecurityException, TransportException {
// See: https://docs.gitlab.com/ee/api/releases/#get-a-release-by-a-tag-name
Objects.requireNonNull(owner, "The release repository owner cannot be null");
Objects.requireNonNull(repository, "The release repository name cannot be null");
Objects.requireNonNull(tag, "The release tag cannot be null");
URI uri = newRequestURI("/projects/"+URLEncoder.encode(owner+"/"+repository, StandardCharsets.UTF_8)+"/releases/"+URLEncoder.encode(tag, StandardCharsets.UTF_8));
HttpResponse response = null;
try {
response = sendRequest(getRequestBuilder(true).uri(uri).GET().build());
}
catch (IOException | InterruptedException e) {
throw new TransportException(e);
}
// See: https://docs.gitlab.com/ee/api/#status-codes
if (response.statusCode() != 200) {
if (response.statusCode() == 404)
return null; // the object just doesn't exist
else if (response.statusCode() == 401 || response.statusCode() == 403)
throw new SecurityException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
else throw new TransportException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
}
return unmarshalJSONBody(response.body());
}
/**
* {@inheritDoc}
*/
@Override
synchronized HttpRequest.Builder getRequestBuilder(boolean authenticate)
throws SecurityException {
// See https://docs.gitlab.com/ee/api/
HttpRequest.Builder builder = HttpRequest.newBuilder()
.version(Version.HTTP_1_1)
.uri(uri)
.setHeader("content-type", CONTENT_TYPE);
if (authenticate) {
if (Objects.isNull(authenticationToken))
throw new SecurityException("No authentication token provided");
builder.setHeader("Authorization", "Bearer "+authenticationToken); // Set the authentication token
}
return builder.copy();
}
/**
* {@inheritDoc}
*/
@Override
Map getUserAttributes(String userID)
throws TransportException, SecurityException {
// See: https://docs.gitlab.com/ee/api/users.html
URI uri = userID == null ? newRequestURI("/user") : newRequestURI("/users/"+URLEncoder.encode(userID, StandardCharsets.UTF_8));
HttpResponse response = null;
try {
response = sendRequest(getRequestBuilder(true).uri(uri).GET().build());
}
catch (IOException | InterruptedException e) {
throw new TransportException(e);
}
// See: https://docs.gitlab.com/ee/api/#status-codes
if (response.statusCode() != 200) {
if (response.statusCode() == 404)
return null; // no such user
else if (response.statusCode() == 401 || response.statusCode() == 403)
throw new SecurityException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
else throw new TransportException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
}
return unmarshalJSONBody(response.body());
}
/**
* {@inheritDoc}
*/
@Override
Set listReleaseAssets(String owner, String repository, String tag)
throws SecurityException, TransportException {
// See: https://docs.gitlab.com/ee/api/releases/links.html#get-links
Objects.requireNonNull(owner, "The release repository owner cannot be null");
Objects.requireNonNull(repository, "The release repository name cannot be null");
Objects.requireNonNull(tag, "The release ID cannot be null");
URI uri = newRequestURI("/projects/"+URLEncoder.encode(owner+"/"+repository, StandardCharsets.UTF_8)+"/releases/"+URLEncoder.encode(tag, StandardCharsets.UTF_8)+"/assets/links");
HttpResponse response = null;
try {
response = sendRequest(getRequestBuilder(true).uri(uri).GET().build());
}
catch (IOException | InterruptedException e) {
throw new TransportException(e);
}
if (response.statusCode() != 200) {
if (response.statusCode() == 404)
return null; // the object just doesn't exist
else if (response.statusCode() == 401 || response.statusCode() == 403)
throw new SecurityException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
else throw new TransportException(String.format("Request returned a status code '%d': %s", response.statusCode(), response.body()));
}
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy