org.apache.sling.testing.clients.SlingClient Maven / Gradle / Ivy
Show all versions of org.apache.sling.testing.clients Show documentation
/*
* 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.sling.testing.clients;
import static org.apache.http.HttpStatus.SC_CREATED;
import static org.apache.http.HttpStatus.SC_NOT_FOUND;
import static org.apache.http.HttpStatus.SC_NOT_IMPLEMENTED;
import static org.apache.http.HttpStatus.SC_OK;
import java.io.File;
import java.net.URI;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import com.fasterxml.jackson.databind.JsonNode;
import org.apache.commons.lang3.StringUtils;
import org.apache.http.HttpEntity;
import org.apache.http.HttpRequestInterceptor;
import org.apache.http.HttpResponseInterceptor;
import org.apache.http.NameValuePair;
import org.apache.http.annotation.Contract;
import org.apache.http.annotation.ThreadingBehavior;
import org.apache.http.client.CookieStore;
import org.apache.http.client.CredentialsProvider;
import org.apache.http.client.RedirectStrategy;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.mime.MultipartEntityBuilder;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.cookie.BasicClientCookie;
import org.apache.sling.testing.clients.exceptions.TestingValidationException;
import org.apache.sling.testing.clients.interceptors.*;
import org.apache.sling.testing.clients.util.*;
import org.apache.sling.testing.clients.util.poller.AbstractPoller;
import org.apache.sling.testing.clients.util.poller.Polling;
import org.apache.sling.testing.timeouts.TimeoutsProvider;
/**
* The Base class for all Integration Test Clients. It provides generic methods to send HTTP requests to a server.
*
* It has methods to perform simple node operations on the server like creating and deleting nodes, etc.
* on the server using requests.
*/
@Contract(threading = ThreadingBehavior.IMMUTABLE)
public class SlingClient extends AbstractSlingClient {
public static final String DEFAULT_NODE_TYPE = "sling:OrderedFolder";
public static final String CLIENT_CONNECTION_TIMEOUT_PROP = "sling.client.connection.timeout.seconds";
public static final String SUDO_COOKIE_NAME = "sling.sudo.cookie.name";
/**
* Constructor used by Builders and adaptTo(). Should never be called directly from the code.
*
* @param http the underlying HttpClient to be used
* @param config sling specific configs
* @throws ClientException if the client could not be created
*
* @see AbstractSlingClient#AbstractSlingClient(CloseableHttpClient, SlingClientConfig)
*/
public SlingClient(CloseableHttpClient http, SlingClientConfig config) throws ClientException {
super(http, config);
}
/**
* Handy constructor easy to use in simple tests. Creates a client that uses basic authentication.
*
* For constructing clients with complex configurations, use a {@link InternalBuilder}
*
* For constructing clients with the same configuration, but a different class, use {@link #adaptTo(Class)}
*
* @param url url of the server (including context path)
* @param user username for basic authentication
* @param password password for basic authentication
* @throws ClientException never, kept for uniformity with the other constructors
*/
public SlingClient(URI url, String user, String password) throws ClientException {
super(Builder.create(url, user, password).buildHttpClient(), Builder.create(url, user, password).buildSlingClientConfig());
}
/**
* Moves a sling path to a new location (:operation move)
*
* @param srcPath source path
* @param destPath destination path
* @param expectedStatus list of accepted status codes in response
* @return the response
* @throws ClientException if an error occurs during operation
*/
public SlingHttpResponse move(String srcPath, String destPath, int... expectedStatus) throws ClientException {
UrlEncodedFormEntity entity = FormEntityBuilder.create()
.addParameter(":operation", "move")
.addParameter(":dest", destPath)
.build();
return this.doPost(srcPath, entity, expectedStatus);
}
/**
* Deletes a sling path (:operation delete)
*
* @param path path to be deleted
* @param expectedStatus list of accepted status codes in response
* @return the response
* @throws ClientException if an error occurs during operation
*/
public SlingHttpResponse deletePath(String path, int... expectedStatus) throws ClientException {
HttpEntity entity = FormEntityBuilder.create().addParameter(":operation", "delete").build();
return this.doPost(path, entity, expectedStatus);
}
/**
* Recursively creates all the none existing nodes in the given path using the {@link SlingClient#createNode(String, String)} method.
* All the created nodes will have the given node type.
*
* @param path the path to use for creating all the none existing nodes
* @param nodeType the node type to use for the created nodes
* @return the response to the creation of the leaf node
* @throws ClientException if one of the nodes can't be created
*/
public SlingHttpResponse createNodeRecursive(final String path, final String nodeType) throws ClientException {
final String parentPath = getParentPath(path);
if (!parentPath.isEmpty() && !exists(parentPath)) {
createNodeRecursive(parentPath, nodeType);
}
return createNode(path, nodeType);
}
/**
* Creates the node specified by a given path with the given node type.
* If the given node type is {@code null}, the node will be created with the default type: {@value DEFAULT_NODE_TYPE}.
* If the node already exists, the method will return null, with no errors.
* The method ignores trailing slashes so a path like this /a/b/c/// is accepted and will create the c node if the rest of
* the path exists.
*
* @param path the path to the node to create
* @param nodeType the type of the node to create
* @return the sling HTTP response or null if the path already existed
* @throws ClientException if the node can't be created
*/
public SlingHttpResponse createNode(final String path, final String nodeType) throws ClientException {
if (!exists(path)) {
String nodeTypeValue = nodeType;
if (nodeTypeValue == null) {
nodeTypeValue = DEFAULT_NODE_TYPE;
}
// Use the property for creating the actual node for working around the Sling issue with dot containing node names.
// The request will be similar with doing:
// curl -F "nodeName/jcr:primaryType=nodeTypeValue" -u admin:admin http://localhost:8080/nodeParentPath
final String nodeName = getNodeNameFromPath(path);
final String nodeParentPath = getParentPath(path);
final HttpEntity entity = FormEntityBuilder.create().addParameter(nodeName + "/jcr:primaryType", nodeTypeValue).build();
return this.doPost(nodeParentPath, entity, SC_OK, SC_CREATED);
} else {
return null;
}
}
/**
* End the impersonation of the current user.
* @return the SlingClient
*/
public SlingClient endImpersonation() {
BasicClientCookie c = new BasicClientCookie(getSudoCookieName(), "");
c.setPath("/");
c.setDomain(getUrl().getHost());
// setting expiry date in the past will remove the cookie
c.setExpiryDate(new Date(0));
getCookieStore().addCookie(c);
return this;
}
/**
* Checks whether a path exists or not by making a GET request to that path with the {@code json} extension
* @param path path to be checked
* @return true if GET response returns 200
* @throws ClientException if the request could not be performed
*/
public boolean exists(String path) throws ClientException {
SlingHttpResponse response = this.doGet(path + ".json", SC_OK, SC_CREATED, SC_NOT_FOUND,SC_NOT_IMPLEMENTED);
final int status = response.getStatusLine().getStatusCode();
return status == SC_OK;
}
/**
* Extracts the parent path from the given String
*
* @param path string containing the path
* @return the parent path if exists or empty string otherwise
*/
protected String getParentPath(final String path) {
// TODO define more precisely what is the parent of a folder and of a file
final String normalizedPath = StringUtils.removeEnd(path, "/"); // remove trailing slash in case of folders
return StringUtils.substringBeforeLast(normalizedPath, "/");
}
/**
* Extracts the node from path
*
* @param path string containing the path
* @return the node without parent path
*/
protected String getNodeNameFromPath(final String path) {
// TODO define the output for all the cases (e.g. paths with trailing slash)
final String normalizedPath = StringUtils.removeEnd(path, "/"); // remove trailing slash in case of folders
final int pos = normalizedPath.lastIndexOf('/');
if (pos != -1) {
return normalizedPath.substring(pos + 1, normalizedPath.length());
}
return normalizedPath;
}
/**
* Checks whether a path exists or not by making a GET request to that path with the {@code json extension}
* It polls the server and waits until the path exists
*
* @deprecated use {@link #waitExists(String, long, long)} instead.
*
* @param path path to be checked
* @param waitMillis time to wait between retries
* @param retryCount number of retries before throwing an exception
* @throws TestingValidationException if the path was not found
* @throws InterruptedException to mark this operation as "waiting"
*/
@Deprecated
public void waitUntilExists(final String path, final long waitMillis, int retryCount)
throws TestingValidationException, InterruptedException {
AbstractPoller poller = new AbstractPoller(waitMillis, retryCount) {
boolean found = false;
public boolean call() {
try {
found = exists(path);
} catch (ClientException e) {
// maybe log
found = false;
}
return true;
}
public boolean condition() {
return found;
}
};
boolean found = poller.callUntilCondition();
if (!found) {
throw new TestingValidationException("path " + path + " does not exist after " + retryCount + " retries");
}
}
/**
* Waits until a path exists by making successive GET requests to that path with the {@code json extension}
* Polls the server until the path exists or until timeout is reached
* @param path path to be checked
* @param timeout max total time to wait, in milliseconds
* @param delay time to wait between checks, in milliseconds
* @throws TimeoutException if the path was not found before timeout
* @throws InterruptedException to mark this operation as "waiting", should be rethrown by callers
* @since 1.1.0
*/
public void waitExists(final String path, final long timeout, final long delay)
throws TimeoutException, InterruptedException {
Polling p = new Polling() {
@Override
public Boolean call() throws Exception {
return exists(path);
}
@Override
protected String message() {
return "Path " + path + " does not exist after %1$d ms";
}
};
p.poll(timeout, delay);
}
/**
* Sets String component property on a node.
*
* @param nodePath path to the node to be edited
* @param propName name of the property to be edited
* @param propValue value of the property to be edited
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 200 is assumed.
* @return the response object
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse setPropertyString(String nodePath, String propName, String propValue, int... expectedStatus)
throws ClientException {
// prepare the form
HttpEntity formEntry = FormEntityBuilder.create().addParameter(propName, propValue).build();
// send the request
return this.doPost(nodePath, formEntry, HttpUtils.getExpectedStatus(SC_OK, expectedStatus));
}
/**
* Sets a String[] component property on a node.
*
* @param nodePath path to the node to be edited
* @param propName name of the property to be edited
* @param propValueList List of String values
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 200 is assumed.
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse setPropertyStringArray(String nodePath, String propName, List propValueList, int... expectedStatus)
throws ClientException {
// prepare the form
FormEntityBuilder formEntry = FormEntityBuilder.create();
for (String propValue : (propValueList != null) ? propValueList : new ArrayList(0)) {
formEntry.addParameter(propName, propValue);
}
// send the request and return the sling response
return this.doPost(nodePath, formEntry.build(), HttpUtils.getExpectedStatus(SC_OK, expectedStatus));
}
/**
* Sets multiple String properties on a node in a single request
* @param nodePath path to the node to be edited
* @param properties list of NameValue pairs with the name and value for each property. String[] properties can be defined
* by adding multiple time the same property name with different values
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 200 is assumed.
* @return the response
* @throws ClientException if the operation could not be completed
*/
public SlingHttpResponse setPropertiesString(String nodePath, List properties, int... expectedStatus)
throws ClientException {
// prepare the form
HttpEntity formEntry = FormEntityBuilder.create().addAllParameters(properties).build();
// send the request and return the sling response
return this.doPost(nodePath, formEntry, HttpUtils.getExpectedStatus(SC_OK, expectedStatus));
}
/**
* Returns the JSON content of a node already mapped to a {@link com.fasterxml.jackson.databind.JsonNode}.
* Waits max 10 seconds for the node to be created.
*
* @deprecated use {@link #waitExists(String, long, long)} and {@link #doGetJson(String, int, int...)} instead
* @param path the path to the content node
* @param depth the number of levels to go down the tree, -1 for infinity
* @return a {@link com.fasterxml.jackson.databind.JsonNode} mapping to the requested content node.
* @throws ClientException if something fails during request/response processing
* @throws InterruptedException to mark this operation as "waiting"
*/
@Deprecated
public JsonNode getJsonNode(String path, int depth) throws ClientException, InterruptedException {
return getJsonNode(path, depth, 500, 20);
}
/**
* Returns JSON format of a content node already mapped to a {@link com.fasterxml.jackson.databind.JsonNode}.
*
* @deprecated use {@link #waitExists(String, long, long)} and {@link #doGetJson(String, int, int...)} instead
* @param path the path to the content node
* @param depth the number of levels to go down the tree, -1 for infinity
* @param waitMillis how long it should wait between requests
* @param retryNumber number of retries before throwing an exception
* @param expectedStatus list of allowed HTTP Status to be returned. If not set,
* http status 200 (OK) is assumed.
* @return a {@link com.fasterxml.jackson.databind.JsonNode} mapping to the requested content node.
* @throws ClientException if something fails during request/response cycle
* @throws InterruptedException to mark this operation as "waiting"
*/
@Deprecated
public JsonNode getJsonNode(String path, int depth, final long waitMillis, final int retryNumber, int... expectedStatus)
throws ClientException, InterruptedException {
// check if path exist and wait if needed
waitUntilExists(path, waitMillis, retryNumber);
// check for infinity
if (depth == -1) {
path += ".infinity.json";
} else {
path += "." + depth + ".json";
}
// request the JSON for the page node
SlingHttpResponse response = this.doGet(path);
HttpUtils.verifyHttpStatus(response, HttpUtils.getExpectedStatus(SC_OK, expectedStatus));
return JsonUtils.getJsonNodeFromString(response.getContent());
}
/**
* Returns the {@link com.fasterxml.jackson.databind.JsonNode} object corresponding to a content node.
*
* @param path the path to the content node
* @param depth the number of levels to go down the tree, -1 for infinity
* @param expectedStatus list of allowed HTTP Status to be returned. If not set, 200 (OK) is assumed.
*
* @return a {@link com.fasterxml.jackson.databind.JsonNode} mapping to the requested content node.
* @throws ClientException if the path does not exist or something fails during request/response cycle
* @since 1.1.0
*/
public JsonNode doGetJson(String path, int depth, int... expectedStatus) throws ClientException {
// check for infinity
if (depth == -1) {
path += ".infinity.json";
} else {
path += "." + depth + ".json";
}
// request the JSON for the node
SlingHttpResponse response = this.doGet(path, HttpUtils.getExpectedStatus(SC_OK, expectedStatus));
return JsonUtils.getJsonNodeFromString(response.getContent());
}
/**
* Uploads a file to the repository. It creates a leaf node typed {@code nt:file}. The intermediary nodes are created with
* type "sling:OrderedFolder" if parameter {@code createFolders} is true
*
* @param file the file to be uploaded
* @param mimeType the MIME Type of the file
* @param toPath the complete path of the file in the repository including file name
* @param createFolders if true, all non existing parent nodes will be created using node type {@code sling:OrderedFolder}
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 201 is assumed.
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse upload(File file, String mimeType, String toPath, boolean createFolders, int... expectedStatus)
throws ClientException {
// Determine filename and parent folder, depending on whether toPath is a folder or a file
String toFileName;
String toFolder;
if (toPath.endsWith("/")) {
toFileName = file.getName();
toFolder = toPath;
} else {
toFileName = getNodeNameFromPath(toPath);
toFolder = getParentPath(toPath);
}
if (createFolders) {
createNodeRecursive(toFolder, "sling:OrderedFolder");
}
if (mimeType == null) {
mimeType = "application/octet-stream";
}
HttpEntity entity = MultipartEntityBuilder.create()
.addBinaryBody(toFileName, file, ContentType.create(mimeType), toFileName)
.build();
// return the sling response
return this.doPost(toFolder, entity, HttpUtils.getExpectedStatus(SC_CREATED, expectedStatus));
}
/**
* Creates a new Folder of type sling:OrderedFolder. Same as using {@code New Folder...} in the Site Admin.
*
* @param folderName The name of the folder to be used in the URL.
* @param folderTitle Title of the Folder to be set in jcr:title
* @param parentPath The parent path where the folder gets added.
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 201 is assumed.
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse createFolder(String folderName, String folderTitle, String parentPath, int... expectedStatus)
throws ClientException {
// we assume the parentPath is a folder, even though it doesn't end with a slash
parentPath = StringUtils.appendIfMissing(parentPath, "/");
String folderPath = parentPath + folderName;
HttpEntity feb = FormEntityBuilder.create()
.addParameter("./jcr:primaryType", "sling:OrderedFolder") // set primary type for folder node
.addParameter("./jcr:content/jcr:primaryType", "nt:unstructured") // add jcr:content as sub node
.addParameter("./jcr:content/jcr:title", folderTitle) //set the title
.build();
// execute request and return the sling response
return this.doPost(folderPath, feb, HttpUtils.getExpectedStatus(SC_CREATED, expectedStatus));
}
/**
* Create a tree structure under {@code parentPath} by providing a {@code content} in one
* of the supported formats: xml, jcr.xml, json, jar, zip.
*
* This is the implementation of {@code :operation import}, as documented in
* importing-content-structures
*
* @param parentPath path where the tree is created
* @param contentType format of the content
* @param content string expressing the structure to be created, in the specified format
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 201 is assumed
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse importContent(String parentPath, String contentType, String content, int... expectedStatus)
throws ClientException {
HttpEntity entity = FormEntityBuilder.create()
.addParameter(":operation", "import")
.addParameter(":contentType", contentType)
.addParameter(":content", content)
.build();
// execute request and return the sling response
return this.doPost(parentPath, entity, HttpUtils.getExpectedStatus(SC_CREATED, expectedStatus));
}
/**
* Create a tree structure under {@code parentPath} by providing a {@code contentFile} in one
* of the supported formats: xml, jcr.xml, json, jar, zip.
*
* This is the implementation of {@code :operation import}, as documented in
* importing-content-structures
*
* @param parentPath path where the tree is created
* @param contentType format of the content
* @param contentFile file containing the structure to be created, in the specified format
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 200 is assumed
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse importContent(String parentPath, String contentType, File contentFile, int... expectedStatus)
throws ClientException {
HttpEntity entity = MultipartEntityBuilder.create()
.addTextBody(":operation", "import")
.addTextBody(":contentType", contentType)
.addBinaryBody(":contentFile", contentFile)
.build();
// execute request and return the sling response
return this.doPost(parentPath, entity, HttpUtils.getExpectedStatus(SC_CREATED, expectedStatus));
}
/**
* Wrapper method over {@link #importContent(String, String, String, int...)} for directly importing a json node
* @param parentPath path where the tree is created
* @param json json node with the desired structure
* @param expectedStatus list of expected HTTP Status to be returned, if not set, 201 is assumed
* @return the response
* @throws ClientException if something fails during the request/response cycle
*/
public SlingHttpResponse importJson(String parentPath, JsonNode json, int... expectedStatus)
throws ClientException {
return importContent(parentPath, "json", json.toString(), expectedStatus);
}
private String getSudoCookieName() {
return Optional.ofNullable(this.getValue(SUDO_COOKIE_NAME)).orElse("sling.sudo");
}
/**
* Get the UUID of a repository path
*
* @param path path in repository
* @return uuid as String or null if path does not exist
* @throws ClientException if something fails during request/response cycle
*/
public String getUUID(String path) throws ClientException {
if (!exists(path)) {
return null;
}
JsonNode jsonNode = doGetJson(path, -1);
return getUUId(jsonNode);
}
/**
* Get the UUID from a node that was already parsed in a {@link JsonNode}
*
* @param jsonNode {@link JsonNode} object of the repository node
* @return UUID as String or null if jsonNode is null or if the UUID was not found
* @throws ClientException if something fails during request/response cycle
*/
// TODO make this method static
public String getUUId(JsonNode jsonNode) throws ClientException {
if (jsonNode == null) {
return null;
}
JsonNode uuidNode = jsonNode.get("jcr:uuid");
if (uuidNode == null) {
return null;
}
//TODO write test to ensure uuidNode.asText() == uuidNode.getValueAsText(), to avoid regression
//return uuidNode.getValueAsText();
return uuidNode.asText();
}
@Override
public String getUser() {
// get the username from the sudo cookie or default from client config
return getCookieStore().getCookies().stream().filter(c -> c.getName().equals(getSudoCookieName())).findFirst()
.map(c -> c.getValue().replace("\"", "")).orElse(super.getUser());
}
/**
* Impersonate user with the given userId
*
* By impersonating a user SlingClient can access content from the perspective of that user.
*
*Passing a null
will clear impersonation.
*
* @param userId the user to impersonate. A null
value clears impersonation
* @return the slingClient with the impersonation applied
*/
public SlingClient impersonate(String userId) {
if(userId == null){
endImpersonation();
return this;
}
BasicClientCookie c = new BasicClientCookie(getSudoCookieName(), userId);
c.setPath("/");
c.setDomain(getUrl().getHost());
getCookieStore().addCookie(c);
return this;
}
//
// InternalBuilder class and builder related methods
//
/**
* Extensible InternalBuilder for SlingClient. Can be used by calling: {@code SlingClient.builder().create(...).build()}.
* Between create() and build(), any number of set methods can be called to customize the client.
* It also exposes the underling httpClientBuilder through {@link #httpClientBuilder()} which can be used to customize the client
* at http level.
*
*
* The InternalBuilder is created to be easily extensible. A class, e.g. {@code MyClient extends SlingClient}, can have its own InternalBuilder.
* This is worth creating if MyClient has fields that need to be initialized. The Skeleton of such InternalBuilder (created inside MyClient) is:
*
*
* {@code
* public static abstract class InternalBuilder extends SlingClient.InternalBuilder {
* private String additionalField;
*
* public InternalBuilder(URI url, String user, String password) { super(url, user, password); }
*
* public InternalBuilder setAdditionalField(String s) { additionalField = s; }
* }
* }
*
* Besides this, two more methods need to be implemented directly inside {@code MyClient}:
*
* {@code
* public static InternalBuilder> builder(URI url, String user, String password) {
* return new InternalBuilder(url, user, password) {
* {@literal @}Override
* public MyClient build() throws ClientException { return new MyClient(this); }
* };
* }
*
* protected MyClient(InternalBuilder builder) throws ClientException {
* super(builder);
* additionalField = builder.additionalField;
* }
* }
*
* Of course, the Clients and InternalBuilder are extensible on several levels, so MyClient.InternalBuilder can be further extended.
*
* @param type extending SlingClient
*/
public static abstract class InternalBuilder {
private final SlingClientConfig.Builder configBuilder;
private final HttpClientBuilder httpClientBuilder;
protected InternalBuilder(URI url, String user, String password) {
this.httpClientBuilder = HttpClientBuilder.create();
this.configBuilder = SlingClientConfig.Builder.create().setUrl(url).setUser(user).setPassword(password);
setDefaults();
}
public InternalBuilder setUrl(URI url) {
this.configBuilder.setUrl(url);
return this;
}
public InternalBuilder setUser(String user) {
this.configBuilder.setUser(user);
return this;
}
public InternalBuilder setPassword(String password) {
this.configBuilder.setPassword(password);
return this;
}
public InternalBuilder setCredentialsProvider(CredentialsProvider cp) {
this.configBuilder.setCredentialsProvider(cp);
return this;
}
public InternalBuilder setPreemptiveAuth(boolean isPreemptiveAuth) {
this.configBuilder.setPreemptiveAuth(isPreemptiveAuth);
return this;
}
public InternalBuilder setCookieStore(CookieStore cs) {
this.configBuilder.setCookieStore(cs);
return this;
}
public HttpClientBuilder httpClientBuilder() {
return httpClientBuilder;
}
public abstract T build() throws ClientException;
protected CloseableHttpClient buildHttpClient() {
return httpClientBuilder.build();
}
protected SlingClientConfig buildSlingClientConfig() throws ClientException {
return configBuilder.build();
}
/**
* Sets defaults to the builder.
*
* @return this
*/
private InternalBuilder setDefaults() {
httpClientBuilder.useSystemProperties();
httpClientBuilder.setUserAgent(SystemPropertiesConfig.getDefaultUserAgent());
// Connection
httpClientBuilder.setMaxConnPerRoute(10);
httpClientBuilder.setMaxConnTotal(100);
// Interceptors
httpClientBuilder.addInterceptorLast(new TestDescriptionInterceptor());
httpClientBuilder.addInterceptorLast(new UserAgentInterceptor());
httpClientBuilder.addInterceptorLast(new DelayRequestInterceptor(SystemPropertiesConfig.getHttpDelay()));
// HTTP request strategy
httpClientBuilder.setServiceUnavailableRetryStrategy(new ServerErrorRetryStrategy());
// connection timeouts
int timeoutSeconds = TimeoutsProvider.getInstance().getTimeout(CLIENT_CONNECTION_TIMEOUT_PROP, -1);
if (timeoutSeconds > 0) {
int timeoutMs = (int)TimeUnit.SECONDS.toMillis(timeoutSeconds);
RequestConfig config = RequestConfig.custom()
.setConnectTimeout(timeoutMs)
.setConnectionRequestTimeout(timeoutMs)
.setSocketTimeout(timeoutMs).build();
this.httpClientBuilder.setDefaultRequestConfig(config);
}
return this;
}
//
// HttpClientBuilder delegating methods
//
public final InternalBuilder addInterceptorFirst(final HttpResponseInterceptor itcp) {
httpClientBuilder.addInterceptorFirst(itcp);
return this;
}
/**
* Adds this protocol interceptor to the tail of the protocol processing list.
*
* Please note this value can be overridden by the {@link HttpClientBuilder#setHttpProcessor(
* org.apache.http.protocol.HttpProcessor)} method.
*
*
* @param itcp the interceptor
* @return this
*/
public final InternalBuilder addInterceptorLast(final HttpResponseInterceptor itcp) {
httpClientBuilder.addInterceptorLast(itcp);
return this;
}
/**
* Adds this protocol interceptor to the head of the protocol processing list.
*
* Please note this value can be overridden by the {@link HttpClientBuilder#setHttpProcessor(
* org.apache.http.protocol.HttpProcessor)} method.
*
*
* @param itcp the interceptor
* @return this
*/
public final InternalBuilder addInterceptorFirst(final HttpRequestInterceptor itcp) {
httpClientBuilder.addInterceptorFirst(itcp);
return this;
}
/**
* Adds this protocol interceptor to the tail of the protocol processing list.
*
* Please note this value can be overridden by the {@link HttpClientBuilder#setHttpProcessor(
* org.apache.http.protocol.HttpProcessor)} method.
*
*
* @param itcp the interceptor
* @return this
*/
public final InternalBuilder addInterceptorLast(final HttpRequestInterceptor itcp) {
httpClientBuilder.addInterceptorLast(itcp);
return this;
}
/**
* Adds this protocol interceptor to the head of the protocol processing list for both requests and responses
*
* Please note this value can be overridden by the {@link HttpClientBuilder#setHttpProcessor(
* org.apache.http.protocol.HttpProcessor)} method.
*
*
* @param itcp the request and response interceptor
* @return this
*/
public final InternalBuilder addInterceptorFirst(final HttpRequestResponseInterceptor itcp) {
httpClientBuilder.addInterceptorFirst((HttpRequestInterceptor) itcp);
httpClientBuilder.addInterceptorFirst((HttpResponseInterceptor) itcp);
return this;
}
/**
* Adds this protocol interceptor to the tail of the protocol processing list for both requests and responses
*
* Please note this value can be overridden by the {@link HttpClientBuilder#setHttpProcessor(
* org.apache.http.protocol.HttpProcessor)} method.
*
*
* @param itcp the request and response interceptor
* @return this
*/
public final InternalBuilder addInterceptorLast(final HttpRequestResponseInterceptor itcp) {
httpClientBuilder.addInterceptorLast((HttpRequestInterceptor) itcp);
httpClientBuilder.addInterceptorLast((HttpResponseInterceptor) itcp);
return this;
}
/**
* Assigns {@link RedirectStrategy} instance.
* Please note this value can be overridden by the {@link #disableRedirectHandling()} method.
*
* @param redirectStrategy custom redirect strategy
* @return this
*/
public final InternalBuilder setRedirectStrategy(final RedirectStrategy redirectStrategy) {
httpClientBuilder.setRedirectStrategy(redirectStrategy);
return this;
}
/**
* Disables automatic redirect handling.
*
* @return this
*/
public final InternalBuilder disableRedirectHandling() {
httpClientBuilder.disableRedirectHandling();
return this;
}
}
public final static class Builder extends InternalBuilder {
private Builder(URI url, String user, String password) {
super(url, user, password);
}
@Override
public SlingClient build() throws ClientException {
return new SlingClient(buildHttpClient(), buildSlingClientConfig());
}
public static Builder create(URI url, String user, String password) {
return new Builder(url, user, password);
}
}
}