blob: 4de88439f301fdbf182feeb2763128948bf0b7c9 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.drill.exec.server.rest;
import io.swagger.v3.oas.annotations.ExternalDocumentation;
import io.swagger.v3.oas.annotations.Operation;
import org.apache.drill.common.logical.StoragePluginConfig.AuthMode;
import org.apache.drill.common.logical.StoragePluginConfig;
import org.apache.drill.common.logical.security.CredentialsProvider;
import org.apache.drill.exec.server.rest.DrillRestServer.UserAuthEnabled;
import org.apache.drill.exec.server.rest.StorageResources.StoragePluginModel;
import org.apache.drill.exec.store.StoragePluginRegistry;
import org.apache.drill.exec.store.StoragePluginRegistry.PluginException;
import org.apache.drill.exec.store.StoragePluginRegistry.PluginFilter;
import org.glassfish.jersey.server.mvc.Viewable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.security.RolesAllowed;
import javax.inject.Inject;
import javax.servlet.http.HttpServletRequest;
import javax.ws.rs.Consumes;
import javax.ws.rs.FormParam;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.Status;
import javax.ws.rs.core.SecurityContext;
import javax.xml.bind.annotation.XmlRootElement;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Spliterator;
import java.util.Spliterators;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
import static org.apache.drill.exec.server.rest.auth.DrillUserPrincipal.AUTHENTICATED_ROLE;
@Path("/")
@RolesAllowed(AUTHENTICATED_ROLE)
public class CredentialResources {
private static final Logger logger = LoggerFactory.getLogger(CredentialResources.class);
private static final Comparator<PluginConfigWrapper> PLUGIN_COMPARATOR =
Comparator.comparing(PluginConfigWrapper::getName);
private static final String ALL_PLUGINS = "all";
private static final String ENABLED_PLUGINS = "enabled";
private static final String DISABLED_PLUGINS = "disabled";
private static final String TRANSLATES_USERS = "translates_users";
@Inject
UserAuthEnabled authEnabled;
@Inject
StoragePluginRegistry storage;
@Inject
SecurityContext sc;
@Inject
HttpServletRequest request;
@GET
@Path("/credentials")
@Produces(MediaType.TEXT_HTML)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Viewable getPlugins() {
List<StoragePluginModel> model = getPluginsJSON().stream()
.map(plugin -> new StoragePluginModel(plugin, request, sc))
.collect(Collectors.toList());
// Creating an empty model with CSRF token, if there are no storage plugins
if (model.isEmpty()) {
model.add(new StoragePluginModel(null, request, sc));
}
return ViewableWithPermissions.create(authEnabled.get(), "/rest/credentials/list.ftl", sc, model);
}
@GET
@Path("/credentials.json")
@Produces(MediaType.APPLICATION_JSON)
public List<PluginConfigWrapper> getPluginsJSON() {
return getConfigsFor(TRANSLATES_USERS);
}
@GET
@Path("/credentials{group: (/[^/]+?)*}-plugins.json")
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public List<PluginConfigWrapper> getConfigsFor(@PathParam("group") String pluginGroup) {
PluginFilter filter;
switch (pluginGroup.trim()) {
case ALL_PLUGINS:
filter = PluginFilter.ALL;
break;
case ENABLED_PLUGINS:
filter = PluginFilter.ENABLED;
break;
case DISABLED_PLUGINS:
filter = PluginFilter.DISABLED;
break;
case TRANSLATES_USERS:
filter = PluginFilter.TRANSLATES_USERS;
break;
default:
return Collections.emptyList();
}
List<PluginConfigWrapper> results = StreamSupport.stream(
Spliterators.spliteratorUnknownSize(storage.storedConfigs(filter).entrySet().iterator(), Spliterator.ORDERED), false)
.map(entry -> new PluginConfigWrapper(entry.getKey(), entry.getValue()))
.sorted(PLUGIN_COMPARATOR)
.collect(Collectors.toList());
if (results.isEmpty()) {
return Collections.emptyList();
} else {
return results;
}
}
@POST
@Path("/credentials/update_credentials")
@Consumes(MediaType.APPLICATION_FORM_URLENCODED)
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response createOrUpdateCredentials(@FormParam("plugin") String pluginName,
@FormParam("username") String username,
@FormParam("password") String password) {
String queryUser = sc.getUserPrincipal().getName();
pluginName = pluginName.trim();
if (pluginName.isEmpty()) {
return Response.status(Response.Status.BAD_REQUEST)
.entity(message("A storage config name may not be empty"))
.build();
}
// Get the config
StoragePluginConfig config = storage.getStoredConfig(pluginName);
if (config.getAuthMode() != AuthMode.USER_TRANSLATION) {
return Response.status(Status.INTERNAL_SERVER_ERROR)
.entity(message(pluginName + " does not support per user translation."))
.build();
}
// Get the credential provider
CredentialsProvider credentialProvider = config.getCredentialsProvider();
credentialProvider.setUserCredentials(username, password, queryUser);
// Since the config classes are not accessible from java-exec, we have to serialize them,
// replace the credential provider with the updated one, and update the storage plugin registry
StoragePluginConfig newConfig = config.updateCredentialProvider(credentialProvider);
newConfig.setEnabled(config.isEnabled());
try {
storage.validatedPut(pluginName, newConfig);
// Force re-caching
storage.setEnabled(pluginName, newConfig.isEnabled());
} catch (PluginException e) {
logger.error("Error while saving plugin", e);
return Response.status(Response.Status.INTERNAL_SERVER_ERROR)
.entity(message("Error while saving plugin: %s", e.getMessage()))
.build();
}
return Response.ok().entity(message("Success")).build();
}
@POST
@Path("/credentials/{pluginName}/update_credentials.json")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response createOrUpdatePlugin(@PathParam("pluginName") String pluginName, UsernamePasswordContainer credentials) {
String queryUser = sc.getUserPrincipal().getName();
String cleanPluginName;
if (pluginName.isEmpty()) {
return Response.status(Response.Status.BAD_REQUEST)
.entity(message("A storage config name may not be empty"))
.build();
}
cleanPluginName = pluginName.trim();
StoragePluginConfig config = storage.getStoredConfig(cleanPluginName);
if (config.getAuthMode() != AuthMode.USER_TRANSLATION) {
return Response.status(Status.INTERNAL_SERVER_ERROR)
.entity(message(cleanPluginName + " does not have user translation enabled."))
.build();
}
CredentialsProvider credentialProvider = config.getCredentialsProvider();
credentialProvider.setUserCredentials(credentials.getUsername(), credentials.getPassword(), queryUser);
// Since the config classes are not accessible from java-exec, we have to serialize them,
// replace the credential provider with the updated one, and update the storage plugin registry
StoragePluginConfig newConfig = config.updateCredentialProvider(credentialProvider);
newConfig.setEnabled(config.isEnabled());
try {
storage.validatedPut(cleanPluginName, newConfig);
// Force re-caching
storage.setEnabled(cleanPluginName, newConfig.isEnabled());
} catch (PluginException e) {
logger.error("Error while saving plugin", e);
return Response.status(Response.Status.INTERNAL_SERVER_ERROR)
.entity(message("Error while updating plugin credentials: %s", e.getMessage()))
.build();
}
return Response.status(Status.OK)
.entity("Credentials have been updated.")
.build();
}
@POST
@Path("/credentials/{name}/update_refresh_token")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response updateRefreshToken(@PathParam("name") String name, OAuthTokenContainer tokens) {
return OAuthRequests.updateRefreshToken(name, tokens, storage, authEnabled, sc);
}
@POST
@Path("/credentials/{name}/update_access_token")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response updateAccessToken(@PathParam("name") String name, OAuthTokenContainer tokens) {
return OAuthRequests.updateAccessToken(name, tokens, storage, authEnabled, sc);
}
@POST
@Path("/credentials/{name}/update_oauth_tokens")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response updateOAuthTokens(@PathParam("name") String name,
OAuthTokenContainer tokenContainer) {
return OAuthRequests.updateOAuthTokens(name, tokenContainer, storage, authEnabled, sc);
}
@GET
@Path("/credentials/{name}/update_oauth2_authtoken")
@Produces(MediaType.TEXT_HTML)
@Operation(externalDocs = @ExternalDocumentation(description = "Apache Drill REST API documentation:", url = "https://drill.apache.org/docs/rest-api-introduction/"))
public Response updateAuthToken(@PathParam("name") String name, @QueryParam("code") String code) {
return OAuthRequests.updateAuthToken(name, code, request, storage, authEnabled, sc);
}
private JsonResult message(String message, Object... args) {
return new JsonResult(String.format(message, args));
}
@XmlRootElement
public static class JsonResult {
private final String result;
public JsonResult(String result) {
this.result = result;
}
public String getResult() {
return result;
}
}
}