Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.codename1.io.rest.RequestBuilder Maven / Gradle / Ivy
/*
* Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores
* CA 94065 USA or visit www.oracle.com if you need additional information or
* have any questions.
*/
package com.codename1.io.rest;
import com.codename1.io.ConnectionRequest;
import com.codename1.io.Data;
import com.codename1.io.JSONParser;
import com.codename1.io.Log;
import com.codename1.io.NetworkEvent;
import com.codename1.io.gzip.GZConnectionRequest;
import com.codename1.properties.PropertyBusinessObject;
import com.codename1.ui.CN;
import com.codename1.ui.events.ActionListener;
import com.codename1.ui.util.EventDispatcher;
import com.codename1.util.Base64;
import com.codename1.util.Callback;
import com.codename1.util.FailureCallback;
import com.codename1.util.OnComplete;
import com.codename1.util.SuccessCallback;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* This class is used to build, invoke the http request and to get the http
* response
*
* @author Chen Fishbein
*/
public class RequestBuilder {
private String method;
private String url;
private Map queryParams = new HashMap();
private Map headers = new HashMap();
private Map pathParams = new HashMap();
private Integer timeout, readTimeout;
private Boolean cookiesEnabled;
private Data body;
private String contentType;
private ErrorCodeHandler byteArrayErrorCallback;
private ErrorCodeHandler jsonErrorCallback;
private ErrorCodeHandler stringErrorCallback;
private ErrorCodeHandler propertyErrorCallback;
private Class errorHandlerPropertyType;
//private ActionListener errorCallback;
private ArrayList> errorCallbacks = new ArrayList>();
private ConnectionRequest.CachingMode cache;
private boolean fetched;
private Boolean postParameters;
private Byte priority;
private boolean insecure;
RequestBuilder(String method, String url) {
this.method = method;
this.url = url;
}
private void checkFetched() {
if(fetched) {
throw new RuntimeException("This method can't be invoked after a request was sent");
}
}
/**
* Turns off checking to make sure that SSL certificate is valid.
*
* @param insecure true to disable ssl certificate checking
* @return this request builder
*/
public RequestBuilder insecure(boolean insecure) {
this.insecure = insecure;
return this;
}
/**
* Sets the caching mode for this request, see {@link com.codename1.io.ConnectionRequest#getCacheMode()}
* @param cache the cache mode
* @return RequestBuilder instance
*/
public RequestBuilder cacheMode(ConnectionRequest.CachingMode cache) {
this.cache = cache;
return this;
}
/**
* Overrides the default behavior of methods so they can be sent using the post/get method
* @param postParameters true to force post, false to use get method. Defaults to true for all methods other than GET
* @return RequestBuilder instance
*/
public RequestBuilder postParameters(Boolean postParameters) {
this.postParameters = postParameters;
return this;
}
/**
* Sets the value of the content type
* @param s the content type
* @return RequestBuilder instance
*/
public RequestBuilder contentType(String s) {
checkFetched();
contentType = s;
return this;
}
/**
* Sets the priority of the request.
* @param priority The priority.
* @return RequestBuilder instance.
* @see ConnectionRequest#setPriority(byte)
* @since 8.0
*/
public RequestBuilder priority(byte priority) {
checkFetched();
this.priority = priority;
return this;
}
/**
* Sets the cookiesEnabled parameter.
* @param cookiesEnabled True to enable cookies. False to disable.
* @return RequestBuilder instance.
* @since 8.0
*/
public RequestBuilder cookiesEnabled(boolean cookiesEnabled) {
checkFetched();
this.cookiesEnabled = cookiesEnabled;
return this;
}
/**
* Add a path param to the request.
* For example if the request url is: http://domain.com/users/{id}
* The path param can be - key="id", value="1"
* When the request executes the path would be: http://domain.com/users/1
*
* @param key the identifier key in the request.
* @param value the value to replace in the url
* @return RequestBuilder instance
*/
public RequestBuilder pathParam(String key, String value) {
checkFetched();
pathParams.put(key, value);
return this;
}
/**
* Add a query parameter to the request
*
* @param key param key
* @param value param value
* @return RequestBuilder instance
*/
public RequestBuilder queryParam(String key, String value) {
checkFetched();
queryParams.put(key, value);
return this;
}
/**
* Add multiple query parameter values to the request using same key.
*
* @param key param key
* @param values param values
* @return RequestBuilder instance
* @since 8.0
*/
public RequestBuilder queryParam(String key, String[] values) {
checkFetched();
queryParams.put(key, values);
return this;
}
/**
* Add a header to the request
*
* @param key
* @param value
* @return RequestBuilder instance
*/
public RequestBuilder header(String key, String value) {
checkFetched();
// .toString() is used to trigger an NPE early for null headers
headers.put(key.toString(), value.toString());
return this;
}
/**
* Sets the request body
*
* @param body request bodyContent
* @return RequestBuilder instance
*/
public RequestBuilder body(final String bodyContent) {
checkFetched();
this.body = new Data() {
@Override
public void appendTo(OutputStream output) throws IOException {
output.write(bodyContent.getBytes("UTF-8"));
}
@Override
public long getSize() throws IOException {
return bodyContent.getBytes("UTF-8").length;
}
};
return this;
}
/**
* Sets the request body lazily.
* @param body Wrapper for the request body that knows how to append to an output stream.
* @return RequestBuilder instances
* @since 7.0
* @see #body(java.lang.String)
*/
public RequestBuilder body(Data body) {
checkFetched();
this.body = body;
return this;
}
/**
* Sets the request body to the JSON matching the given object
*
* @param body request body
* @return RequestBuilder instance
*/
public RequestBuilder body(PropertyBusinessObject body) {
body(body.getPropertyIndex().toJSON());
return this;
}
/**
* In case of an error this method is invoked asynchronously to process
* the error content with the byte array data
* @param err the content of the error response
* @return RequestBuilder instance
*/
public RequestBuilder onErrorCodeBytes(ErrorCodeHandler err) {
checkFetched();
byteArrayErrorCallback = err;
return this;
}
/**
* In case of an error this method is invoked asynchronously to process
* the error content with the JSON data
* @param err the content of the error response
* @return RequestBuilder instance
*/
public RequestBuilder onErrorCodeJSON(ErrorCodeHandler err) {
checkFetched();
jsonErrorCallback = err;
return this;
}
/**
* In case of an error this method is invoked asynchronously to process
* the error content with the JSON data and places it into a business
* object in the callback
* @param err the content of the error response
* @param errorClass the class of the business object into which the data is parsed
* @return RequestBuilder instance
*/
public RequestBuilder onErrorCode(ErrorCodeHandler err, Class errorClass) {
checkFetched();
propertyErrorCallback = err;
errorHandlerPropertyType = errorClass;
return this;
}
/**
* In case of an error this method is invoked asynchronously to process
* the error content with the JSON data
* @param err the content of the error response
* @return RequestBuilder instance
*/
public RequestBuilder onErrorCodeString(ErrorCodeHandler err) {
checkFetched();
stringErrorCallback = err;
return this;
}
/**
* Invoked for exceptions or failures such as disconnect. Replaces any existing
* callbacks previously registered with {@link #onError(com.codename1.ui.events.ActionListener) }
* @param error callback for a networking error
* @return RequestBuilder instance
* @see #onError(com.codename1.ui.events.ActionListener, boolean)
*
*/
public RequestBuilder onError(ActionListener error) {
return onError(error, true);
}
/**
* Invoked for exceptions or failures such as disconnect
* @param error callback for a networking error
* @param replace If true, replaces the existing errorCallback(s) with the handler
* provided.
* @return RequestBuilder instance
* @since 7.0
*/
public RequestBuilder onError(ActionListener error, boolean replace) {
checkFetched();
if (replace) {
errorCallbacks.clear();
}
errorCallbacks.add(error);
return this;
}
/**
* Sets the request timeout
*
* @param timeout request timeout in milliseconds
* @return RequestBuilder instance
*/
public RequestBuilder timeout(int timeout) {
checkFetched();
this.timeout = timeout;
return this;
}
/**
* Sets the request read timeout. Only used if {@link ConnectionRequest#isReadTimeoutSupported() }
* is true on this platform.
* @param timeout The timeout.
* @return RequestBuilder instance.
*/
public RequestBuilder readTimeout(int timeout) {
checkFetched();
this.readTimeout = timeout;
return this;
}
/**
* Sets the request to be a gzip request
*
* @return RequestBuilder instance
* @deprecated this API was implemented incorrectly
*/
public RequestBuilder gzip() {
header("Accept-Encoding", "gzip");
return this;
}
/**
* Add accept json header to the request
*
* @return RequestBuilder instance
*/
public RequestBuilder acceptJson() {
header("Accept", "application/json");
return this;
}
/**
* Sets both the content type and accept headers to "application/json"
* @return RequestBuilder instance
*/
public RequestBuilder jsonContent() {
return contentType("application/json").
header("Accept", "application/json");
}
/**
* Add a basic authentication Authorization header
*
* @return RequestBuilder instance
*/
public RequestBuilder basicAuth(String username, String password) {
header("Authorization", "Basic " + Base64.encodeNoNewline((username + ":" + password).getBytes()));
return this;
}
/**
* Add an authorization bearer header, this is shorthand for
* {@code header("Authorization", "Bearer " + token)}
*
* @param token the authorization token
* @return RequestBuilder instance
*/
public RequestBuilder bearer(String token) {
header("Authorization", "Bearer " + token);
return this;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback invoked with the result of the builder query
* @return the ConnectionRequest instance
*/
public ConnectionRequest fetchAsString(final OnComplete> callback) {
return getAsStringAsyncImpl(callback);
}
private ConnectionRequest getAsStringAsyncImpl(final Object callback) {
final Connection request = createRequest(false);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if(request.errorCode) {
return;
}
Response res = null;
try {
res = new Response(evt.getResponseCode(), new String(evt.getConnectionRequest().getResponseData(), "UTF-8"), evt.getMessage());
if(callback instanceof Callback) {
((Callback)callback).onSucess(res);
} else {
((OnComplete>)callback).completed(res);
}
} catch (UnsupportedEncodingException ex) {
ex.printStackTrace();
}
}
});
fetched = true;
CN.addToQueue(request);
return request;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @deprecated please use {@link #fetchAsString(com.codename1.util.OnComplete)} instead
*/
public void getAsStringAsync(final Callback> callback) {
getAsStringAsyncImpl(callback);
}
/**
* Executes the request synchronously
*
* @return Response Object
*/
public Response getAsString() {
ConnectionRequest request = createRequest(false);
fetched = true;
CN.addToQueueAndWait(request);
Response res = null;
try {
byte[] respData = request.getResponseData();
String resp = null;
if(respData != null) {
resp = new String(respData, "UTF-8");
}
res = new Response(request.getResponseCode(), resp,
request.getResponseErrorMessage());
} catch (UnsupportedEncodingException ex) {
ex.printStackTrace();
}
return res;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @return the connection request instance
*/
public ConnectionRequest fetchAsBytes(final OnComplete> callback) {
return getAsBytesAsyncImpl(callback);
}
private ConnectionRequest getAsBytesAsyncImpl(final Object callback) {
final Connection request = createRequest(false);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if(request.errorCode) {
return;
}
Response res = null;
res = new Response(evt.getResponseCode(), evt.getConnectionRequest().getResponseData(), evt.getMessage());
if(callback instanceof Callback) {
((Callback)callback).onSucess(res);
} else {
((OnComplete)callback).completed(res);
}
}
});
fetched = true;
CN.addToQueue(request);
return request;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @deprecated use {@link #fetchAsBytes(com.codename1.util.OnComplete)} instead
*/
public void getAsBytesAsync(final Callback> callback) {
getAsBytesAsyncImpl(callback);
}
/**
* Executes the request synchronously
*
* @return Response Object
*/
public Response getAsBytes() {
ConnectionRequest request = createRequest(false);
fetched = true;
CN.addToQueueAndWait(request);
Response res = new Response(request.getResponseCode(), request.getResponseData(), request.getResponseErrorMessage());
return res;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @return returns the Connection Request object so it can be killed if necessary
*/
public ConnectionRequest fetchAsJsonMap(final OnComplete> callback) {
final Connection request = createRequest(true);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if(request.errorCode) {
return;
}
Response res = null;
Map response = (Map)evt.getMetaData();
res = new Response(evt.getResponseCode(), response, evt.getMessage());
callback.completed(res);
}
});
fetched = true;
CN.addToQueue(request);
return request;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback. This fetches JSON data and parses it into a properties business object
* @param callback writes the response to this callback
* @param type the class of the business object returned
* @return returns the Connection Request object so it can be killed if necessary
*/
public ConnectionRequest fetchAsProperties(final OnComplete> callback, final Class type) {
final Connection request = createRequest(true);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if(request.errorCode) {
return;
}
Response res = null;
Map response = (Map)evt.getMetaData();
try {
PropertyBusinessObject pb = (PropertyBusinessObject)type.newInstance();
pb.getPropertyIndex().populateFromMap(response);
res = new Response(evt.getResponseCode(), pb, evt.getMessage());
callback.completed(res);
} catch(Exception err) {
Log.e(err);
throw new RuntimeException(err.toString());
}
}
});
fetched = true;
CN.addToQueue(request);
return request;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @return returns the Connection Request object so it can be killed if necessary
* @deprecated use {@link #fetchAsJsonMap(com.codename1.util.OnComplete)} instead
*/
public ConnectionRequest getAsJsonMap(final SuccessCallback> callback) {
return getAsJsonMap(callback, null);
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @param onError the error callback
* @return returns the Connection Request object so it can be killed if necessary
* @deprecated use {@link #fetchAsJsonMap(com.codename1.util.OnComplete)} instead
*/
public ConnectionRequest getAsJsonMap(final SuccessCallback> callback, final FailureCallback extends Object> onError) {
final Connection request = createRequest(true);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if(request.errorCode) {
return;
}
if(onError != null) {
// this is an error response code and should be handled as an error
if(evt.getResponseCode() > 310) {
return;
}
}
Response res = null;
Map response = (Map)evt.getMetaData();
res = new Response(evt.getResponseCode(), response, evt.getMessage());
callback.onSucess(res);
}
});
bindOnError(request, onError);
fetched = true;
CN.addToQueue(request);
return request;
}
private void bindOnError(final ConnectionRequest req, final FailureCallback extends Object> f) {
if(f == null) {
return;
}
req.addResponseCodeListener(new ActionListener() {
public void actionPerformed(NetworkEvent evt) {
evt.consume();
f.onError(null, evt.getError(), evt.getResponseCode(), evt.getMessage());
}
});
req.addExceptionListener(new ActionListener() {
public void actionPerformed(NetworkEvent evt) {
evt.consume();
f.onError(null, evt.getError(), evt.getResponseCode(), evt.getMessage());
}
});
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback
* @param callback writes the response to this callback
* @deprecated use {@link #fetchAsJsonMap(com.codename1.util.OnComplete)} instead
*/
public void getAsJsonMapAsync(final Callback> callback) {
getAsJsonMap(callback, callback);
}
/**
* Executes the request synchronously
*
* @return Response Object
*/
public Response getAsJsonMap() {
ConnectionRequest request = createRequest(true);
fetched = true;
CN.addToQueueAndWait(request);
Map response = ((Connection)request).json;
return new Response(request.getResponseCode(), response, request.getResponseErrorMessage());
}
/**
* Executes the request synchronously
*
* @param type the type of the business object to create
* @return Response Object
*/
public Response getAsProperties(Class type) {
ConnectionRequest request = createRequest(true);
fetched = true;
CN.addToQueueAndWait(request);
Map response = ((Connection)request).json;
try {
PropertyBusinessObject pb = (PropertyBusinessObject)type.newInstance();
pb.getPropertyIndex().populateFromMap(response);
return new Response(request.getResponseCode(), pb, request.getResponseErrorMessage());
} catch(Exception err) {
Log.e(err);
throw new RuntimeException(err.toString());
}
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback. This fetches JSON data and parses it into a properties business object
* @param callback writes the response to this callback
* @param type the class of the business object returned
* @param root the root element's key of the structured content
* @return returns the Connection Request object so it can be killed if necessary
*/
public ConnectionRequest fetchAsPropertyList(final OnComplete>> callback, final Class type, final String root) {
final Connection request = createRequest(true);
request.addResponseListener(new ActionListener() {
@Override
public void actionPerformed(NetworkEvent evt) {
if (request.errorCode) {
return;
}
Response res = null;
Map response = (Map) evt.getMetaData();
List lst = (List) response.get(root);
if (lst == null) {
return;
}
try {
List result = new ArrayList();
for (Map m : lst) {
PropertyBusinessObject pb = (PropertyBusinessObject) type.newInstance();
pb.getPropertyIndex().populateFromMap(m);
result.add(pb);
}
res = new Response(evt.getResponseCode(), result, evt.getMessage());
callback.completed(res);
} catch (Exception err) {
Log.e(err);
throw new RuntimeException(err.toString());
}
}
});
fetched = true;
CN.addToQueue(request);
return request;
}
/**
* Executes the request asynchronously and writes the response to the provided
* Callback. This fetches JSON data and parses it into a properties business object
* @param callback writes the response to this callback
* @param type the class of the business object returned
* @return returns the Connection Request object so it can be killed if necessary
*/
public ConnectionRequest fetchAsPropertyList(final OnComplete>> callback, final Class type) {
return fetchAsPropertyList(callback, type, "root");
}
/**
* Executes the request synchronously
*
* @param type the type of the business object to create
* @param root the root element's key of the structured content
* @return Response Object
*/
public Response> getAsPropertyList(Class type, String root) {
ConnectionRequest request = createRequest(true);
fetched = true;
CN.addToQueueAndWait(request);
Map response = ((Connection) request).json;
try {
List lst = (List) response.get(root);
List result = new ArrayList();
for (Map m : lst) {
PropertyBusinessObject pb = (PropertyBusinessObject) type.newInstance();
pb.getPropertyIndex().populateFromMap(m);
result.add(pb);
}
return new Response(request.getResponseCode(), result, request.getResponseErrorMessage());
} catch (Exception err) {
Log.e(err);
throw new RuntimeException(err.toString());
}
}
/**
* Executes the request synchronously
*
* @param type the type of the business object to create
* @return Response Object
*/
public Response> getAsPropertyList(Class type) {
return getAsPropertyList(type, "root");
}
public String getRequestUrl() {
return this.url;
}
class Connection extends GZConnectionRequest {
private boolean parseJSON;
boolean errorCode;
Map json;
private ErrorCodeHandler errorHandler;
private Object errorObject;
public Connection(boolean parseJSON) {
this.parseJSON = parseJSON;
}
@Override
protected void handleErrorResponseCode(int code, String message) {
if(byteArrayErrorCallback != null || stringErrorCallback != null ||
jsonErrorCallback != null || propertyErrorCallback != null) {
errorCode = true;
} else {
super.handleErrorResponseCode(code, message);
}
}
@Override
protected void readUnzipedResponse(InputStream input) throws IOException {
if(errorCode) {
if(byteArrayErrorCallback != null) {
super.readUnzipedResponse(input);
errorObject = getResponseData();
errorHandler = byteArrayErrorCallback;
return;
}
if(stringErrorCallback != null) {
super.readUnzipedResponse(input);
errorObject = new String(getResponseData(), "UTF-8");
errorHandler = stringErrorCallback;
return;
}
if(jsonErrorCallback != null) {
JSONParser jp = new JSONParser();
errorObject = jp.parseJSON(new InputStreamReader(input, "UTF-8"));
errorHandler = jsonErrorCallback;
return;
}
if(propertyErrorCallback != null) {
try {
JSONParser jp = new JSONParser();
Map m = jp.parseJSON(new InputStreamReader(input, "UTF-8"));
PropertyBusinessObject pb = (PropertyBusinessObject)errorHandlerPropertyType.newInstance();
pb.getPropertyIndex().populateFromMap(m);
errorObject = pb;
errorHandler = propertyErrorCallback;
} catch(InstantiationException err) {
Log.e(err);
throw new IOException(err.toString());
} catch(IllegalAccessException err) {
Log.e(err);
throw new IOException(err.toString());
}
}
return;
}
if(parseJSON) {
JSONParser parser = new JSONParser();
json = parser.parseJSON(new InputStreamReader(input, "UTF-8"));
if(hasResponseListeners() && !isKilled()) {
fireResponseListener(new NetworkEvent(this, json));
}
return;
}
super.readUnzipedResponse(input);
}
@Override
protected void postResponse() {
if(errorHandler != null) {
errorHandler.onError(new Response(getResponseCode(), errorObject, getResponseErrorMessage()));
}
}
}
private Connection createRequest(boolean parseJson) {
Connection req = new Connection(parseJson);
for (String key : pathParams.keySet()) {
url = com.codename1.util.StringUtil.replaceAll(url, "{" + key + "}", pathParams.get(key));
}
if(contentType != null) {
req.setContentType(contentType);
}
req.setFailSilently(false);
if(cache != null) {
req.setCacheMode(cache);
}
req.setReadResponseForErrors(true);
req.setDuplicateSupported(true);
req.setUrl(url);
req.setHttpMethod(method);
if(postParameters == null) {
req.setPost(method.equalsIgnoreCase("POST") || method.equalsIgnoreCase("PUT") || method.equalsIgnoreCase("PATCH"));
} else {
req.setPost(postParameters);
}
if(body != null){
req.setRequestBody(body);
req.setWriteRequest(true);
}
if(timeout != null){
req.setTimeout(timeout);
}
if (readTimeout != null) {
req.setReadTimeout(readTimeout);
}
for (String key : queryParams.keySet()) {
Object value = queryParams.get(key);
if (value instanceof String[]) {
req.addArgument(key, (String[])value);
} else {
req.addArgument(key, (String)value);
}
}
for (String key : headers.keySet()) {
req.addRequestHeader(key, headers.get(key));
}
for (ActionListener l : errorCallbacks) {
req.addExceptionListener(l);
}
req.setInsecure(insecure);
if (cookiesEnabled != null) {
req.setCookiesEnabled(cookiesEnabled);
}
if (priority != null) {
req.setPriority(priority);
}
return req;
}
}