/*
* Copyright OpenSearch Contributors
* SPDX-License-Identifier: Apache-2.0
*
* The OpenSearch Contributors require contributions made to
* this file be licensed under the Apache-2.0 license or a
* compatible open source license.
*/
package org.opensearch.sdk.api;
import org.opensearch.action.ActionType;
import org.opensearch.action.ActionRequest;
import org.opensearch.action.ActionResponse;
import org.opensearch.action.RequestValidators;
import org.opensearch.action.admin.indices.alias.IndicesAliasesRequest;
import org.opensearch.action.admin.indices.mapping.put.PutMappingRequest;
import org.opensearch.action.support.ActionFilter;
import org.opensearch.action.support.TransportAction;
import org.opensearch.action.support.TransportActions;
import org.opensearch.core.common.Strings;
import org.opensearch.common.util.concurrent.ThreadContext;
import org.opensearch.rest.RestHeaderDefinition;
import org.opensearch.sdk.Extension;
import org.opensearch.sdk.rest.ExtensionRestHandler;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
import java.util.function.UnaryOperator;
import java.util.stream.Collectors;
/**
* An additional extension point for {@link Extension}s that extends OpenSearch's scripting functionality. Implement it like this:
*
{@code
* {@literal @}Override
* public List> getActions() {
* return Arrays.asList(new ActionHandler<>(ReindexAction.INSTANCE, TransportReindexAction.class),
* new ActionHandler<>(UpdateByQueryAction.INSTANCE, TransportUpdateByQueryAction.class),
* new ActionHandler<>(DeleteByQueryAction.INSTANCE, TransportDeleteByQueryAction.class),
* new ActionHandler<>(RethrottleAction.INSTANCE, TransportRethrottleAction.class));
* }
* }
*/
public interface ActionExtension {
/**
* Actions added by this extension.
*
* @return a list of ActionHandler instances representing the added actions.
*/
default List> getActions() {
return Collections.emptyList();
}
/**
* Client actions added by this extension. This defaults to all of the {@linkplain ActionType} in
* {@linkplain ActionExtension#getActions()}.
*
* @return a list of ActionType instances representing the added client actions.
*/
default List> getClientActions() {
return getActions().stream().map(a -> a.action).collect(Collectors.toList());
}
/**
* ActionType filters added by this extension.
*
* @return a list of ActionFilter instances representing the added action filters.
*/
default List getActionFilters() {
return Collections.emptyList();
}
/**
* Gets a list of {@link ExtensionRestHandler} implementations this extension handles.
*
* @return a list of REST handlers (REST actions) this extension handles.
*/
default List getExtensionRestHandlers() {
return Collections.emptyList();
}
/**
* Returns headers which should be copied through rest requests on to internal requests.
*
* @return a collection of RestHeaderDefinition instances representing the headers to be copied.
*/
default Collection getRestHeaders() {
return Collections.emptyList();
}
/**
* Returns headers which should be copied from internal requests into tasks.
*
* @return a collection of String instances representing the headers to be copied.
*/
default Collection getTaskHeaders() {
return Collections.emptyList();
}
/**
* Returns a function used to wrap each rest request before handling the request.
* The returned {@link UnaryOperator} is called for every incoming rest request and receives
* the original rest handler as it's input. This allows adding arbitrary functionality around
* rest request handlers to do for instance logging or authentication.
* A simple example of how to only allow GET request is here:
*
* {@code
* UnaryOperator getRestHandlerWrapper(ThreadContext threadContext) {
* return originalHandler -> (RestHandler) (request) -> {
* if (request.method() != Method.GET) {
* throw new IllegalStateException("only GET requests are allowed");
* }
* originalHandler.handleRequest(request);
* };
* }
* }
*
*
* @param threadContext The Thread Context which can be used by the operator
* @return a function used to wrap each rest request
*/
default UnaryOperator getRestHandlerWrapper(ThreadContext threadContext) {
return null;
}
/**
* Class responsible for handing Transport Actions
*/
final class ActionHandler {
private final ActionType action;
private final Class extends TransportAction> transportAction;
private final Class>[] supportTransportActions;
/**
* Create a record of an action, the {@linkplain TransportAction} that handles it, and any supporting {@linkplain TransportActions}
* that are needed by that {@linkplain TransportAction}.
* @param action the action for the handler.
* @param transportAction transport action.
* @param supportTransportActions supported transport actions.
*/
public ActionHandler(
ActionType action,
Class extends TransportAction> transportAction,
Class>... supportTransportActions
) {
this.action = action;
this.transportAction = transportAction;
this.supportTransportActions = supportTransportActions;
}
/**
* Returns the ActionType associated with this ActionHandler.
*
* @return the ActionType associated with this ActionHandler
*/
public ActionType getAction() {
return action;
}
/**
* Returns the TransportAction associated with this ActionHandler.
*
* @return the TransportAction associated with this ActionHandler
*/
public Class extends TransportAction> getTransportAction() {
return transportAction;
}
/**
* Returns the array of supporting TransportActions associated with this ActionHandler.
*
* @return the array of supporting TransportActions associated with this ActionHandler
*/
public Class>[] getSupportTransportActions() {
return supportTransportActions;
}
@Override
public String toString() {
StringBuilder b = new StringBuilder().append(action.name()).append(" is handled by ").append(transportAction.getName());
if (supportTransportActions.length > 0) {
b.append('[').append(Strings.arrayToCommaDelimitedString(supportTransportActions)).append(']');
}
return b.toString();
}
@Override
public boolean equals(Object obj) {
if (obj == null || obj.getClass() != ActionHandler.class) {
return false;
}
ActionHandler, ?> other = (ActionHandler, ?>) obj;
return Objects.equals(action, other.action)
&& Objects.equals(transportAction, other.transportAction)
&& Objects.deepEquals(supportTransportActions, other.supportTransportActions);
}
@Override
public int hashCode() {
return Objects.hash(action, transportAction, supportTransportActions);
}
}
/**
* Returns a collection of validators that are used by {@link RequestValidators} to validate a
* {@link org.opensearch.action.admin.indices.mapping.put.PutMappingRequest} before the executing it.
*/
default Collection> mappingRequestValidators() {
return Collections.emptyList();
}
/**
* Returns a collection of validators that are used by {@link RequestValidators} to validate a
* {@link org.opensearch.action.admin.indices.alias.IndicesAliasesRequest} before the executing it.
*/
default Collection> indicesAliasesRequestValidators() {
return Collections.emptyList();
}
}