com.ibm.cloud.objectstorage.DefaultRequest Maven / Gradle / Ivy
/*
* Copyright 2010-2022 Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0
*
* or in the "license" file accompanying this file. This file 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.ibm.cloud.objectstorage;
import com.ibm.cloud.objectstorage.annotation.NotThreadSafe;
import com.ibm.cloud.objectstorage.event.ProgressInputStream;
import com.ibm.cloud.objectstorage.handlers.HandlerContextKey;
import com.ibm.cloud.objectstorage.http.HttpMethodName;
import com.ibm.cloud.objectstorage.util.AWSRequestMetrics;
import com.ibm.cloud.objectstorage.util.json.Jackson;
import java.io.InputStream;
import java.net.URI;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
/**
* Default implementation of the {@linkplain com.ibm.cloud.objectstorage.Request} interface.
*
* This class is only intended for internal use inside the AWS client libraries.
* Callers shouldn't ever interact directly with objects of this class.
*/
@NotThreadSafe
public class DefaultRequest implements Request {
/** The resource path being requested */
private String resourcePath;
/**
* Map of the parameters being sent as part of this request.
*
* Note that a LinkedHashMap is used, since we want to preserve the
* insertion order so that members of a list parameter will still be ordered
* by their indices when they are marshalled into the query string.
*
* Lists values in this Map must use an implementation that allows
* null values to be present.
*/
private Map> parameters = new LinkedHashMap>();
/** Map of the headers included in this request */
private Map headers = new TreeMap(String.CASE_INSENSITIVE_ORDER);
/** The service endpoint to which this request should be sent */
private URI endpoint;
/** The name of the service to which this request is being sent */
private String serviceName;
/**
* The original, user facing request object which this internal request
* object is representing
*/
private final AmazonWebServiceRequest originalRequest;
/** The HTTP method to use when sending this request. */
private HttpMethodName httpMethod = HttpMethodName.POST;
/** An optional stream from which to read the request payload. */
private InputStream content;
/** An optional time offset to account for clock skew */
private int timeOffset;
/** All AWS Request metrics are collected into this object. */
private AWSRequestMetrics metrics;
/**
* Context associated with a request. Mainly used to transfer
* information between different {@link com.ibm.cloud.objectstorage.handlers.RequestHandler2}
*/
private final Map, Object> handlerContext = new
HashMap, Object>();
/**
* Constructs a new DefaultRequest with the specified service name and the
* original, user facing request object.
*
* @param serviceName
* The name of the service to which this request is being sent.
* @param originalRequest
* The original, user facing, AWS request being represented by
* this internal request object.
*/
public DefaultRequest(AmazonWebServiceRequest originalRequest, String serviceName) {
this.serviceName = serviceName;
this.originalRequest = originalRequest == null ? AmazonWebServiceRequest.NOOP
: originalRequest;
this.handlerContext.putAll(this.originalRequest.getHandlerContext());
}
/**
* Constructs a new DefaultRequest with the specified service name and no
* specified original, user facing request object.
*
* @param serviceName
* The name of the service to which this request is being sent.
*/
public DefaultRequest(String serviceName) {
this(null, serviceName);
}
/**
* Returns the original, user facing request object which this internal
* request object is representing.
*
* @return The original, user facing request object which this request
* object is representing.
*/
public AmazonWebServiceRequest getOriginalRequest() {
return originalRequest;
}
/**
* @see com.ibm.cloud.objectstorage.Request#addHeader(java.lang.String, java.lang.String)
*/
public void addHeader(String name, String value) {
headers.put(name, value);
}
/**
* @see com.ibm.cloud.objectstorage.Request#getHeaders()
*/
public Map getHeaders() {
return headers;
}
/**
* @see com.ibm.cloud.objectstorage.Request#setResourcePath(java.lang.String)
*/
public void setResourcePath(String resourcePath) {
this.resourcePath = resourcePath;
}
/**
* @see com.ibm.cloud.objectstorage.Request#getResourcePath()
*/
public String getResourcePath() {
return resourcePath;
}
/**
* @see com.ibm.cloud.objectstorage.Request#addParameter(java.lang.String, java.lang.String)
*/
public void addParameter(String name, String value) {
List paramList = parameters.get(name);
if (paramList == null) {
paramList = new ArrayList();
parameters.put(name, paramList);
}
paramList.add(value);
}
/**
*
* @see com.ibm.cloud.objectstorage.Request#addParameters(java.lang.String, java.util.List)
*/
public void addParameters(String name, List values) {
if (values == null) return;
for (String value : values) {
addParameter(name, value);
}
}
/**
* @see com.ibm.cloud.objectstorage.Request#getParameters()
*/
public Map> getParameters() {
return parameters;
}
/**
* @see com.ibm.cloud.objectstorage.Request#withParameter(java.lang.String, java.lang.String)
*/
public Request withParameter(String name, String value) {
addParameter(name, value);
return this;
}
/**
* @see com.ibm.cloud.objectstorage.Request#getHttpMethod()
*/
public HttpMethodName getHttpMethod() {
return httpMethod;
}
/**
* @see com.ibm.cloud.objectstorage.Request#setHttpMethod(com.ibm.cloud.objectstorage.http.HttpMethodName)
*/
public void setHttpMethod(HttpMethodName httpMethod) {
this.httpMethod = httpMethod;
}
/**
* @see com.ibm.cloud.objectstorage.Request#setEndpoint(java.net.URI)
*/
public void setEndpoint(URI endpoint) {
this.endpoint = endpoint;
}
/**
* @see com.ibm.cloud.objectstorage.Request#getEndpoint()
*/
public URI getEndpoint() {
return endpoint;
}
/**
* @see com.ibm.cloud.objectstorage.Request#getServiceName()
*/
public String getServiceName() {
return serviceName;
}
/**
* @see com.ibm.cloud.objectstorage.Request#getContent()
*/
public InputStream getContent() {
return content;
}
/**
* @see com.ibm.cloud.objectstorage.Request#setContent(java.io.InputStream)
*/
public void setContent(InputStream content) {
this.content = content;
}
/**
* @see com.ibm.cloud.objectstorage.Request#setHeaders(java.util.Map)
*/
public void setHeaders(Map headers) {
this.headers.clear();
this.headers.putAll(headers);
}
/**
* @see com.ibm.cloud.objectstorage.Request#setParameters(java.util.Map)
*/
public void setParameters(Map> parameters) {
this.parameters.clear();
this.parameters.putAll(parameters);
}
/**
* @see com.ibm.cloud.objectstorage.Request#getTimeOffset
*/
public int getTimeOffset() {
return timeOffset;
}
/**
* @see Request#setTimeOffset(int)
*/
public void setTimeOffset(int timeOffset) {
this.timeOffset = timeOffset;
}
/**
* @see Request#setTimeOffset(int)
*/
public Request withTimeOffset(int timeOffset) {
setTimeOffset(timeOffset);
return this;
}
@Override
public String toString() {
final StringBuilder builder = new StringBuilder();
builder.append(getHttpMethod()).append(" ");
builder.append(getEndpoint()).append(" ");
String resourcePath = getResourcePath();
if (resourcePath == null) {
builder.append("/");
}
else {
if (!resourcePath.startsWith("/")) {
builder.append("/");
}
builder.append(resourcePath);
}
builder.append(" ");
if (!getParameters().isEmpty()) {
builder.append("Parameters: (")
.append(Jackson.toJsonString(parameters));
}
if (!getHeaders().isEmpty()) {
builder.append("Headers: (");
for (String key : getHeaders().keySet()) {
String value = getHeaders().get(key);
builder.append(key).append(": ").append(value).append(", ");
}
builder.append(") ");
}
return builder.toString();
}
@Override
public AWSRequestMetrics getAWSRequestMetrics() {
return metrics;
}
@Override
public void setAWSRequestMetrics(AWSRequestMetrics metrics) {
if (this.metrics == null) {
this.metrics = metrics;
} else {
throw new IllegalStateException("AWSRequestMetrics has already been set on this request");
}
}
@Override
public void addHandlerContext(HandlerContextKey key, X value) {
handlerContext.put(key, value);
}
@Override
public X getHandlerContext(HandlerContextKey key) {
return (X) handlerContext.get(key);
}
@SuppressWarnings("resource")
@Override
public InputStream getContentUnwrapped() {
InputStream is = getContent();
if (is == null)
return null;
// We want to disable the progress reporting when the stream is
// consumed for signing purpose.
while (is instanceof ProgressInputStream) {
ProgressInputStream pris = (ProgressInputStream)is;
is = pris.getWrappedInputStream();
}
return is;
}
@Override
public ReadLimitInfo getReadLimitInfo() {
return originalRequest;
}
@Override
public Object getOriginalRequestObject() {
return originalRequest;
}
}