// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.net;
import androidx.annotation.RequiresOptIn;
import java.nio.ByteBuffer;
import java.util.concurrent.Executor;
/**
* Controls an HTTP request (GET, PUT, POST etc). Created by {@link UrlRequest.Builder}, which can
* be obtained by calling {@link CronetEngine#newUrlRequestBuilder}. Note: All methods must be
* called on the {@link Executor} passed to {@link CronetEngine#newUrlRequestBuilder}.
*/
public abstract class UrlRequest {
/**
* Builder for {@link UrlRequest}s. Allows configuring requests before constructing them with
* {@link Builder#build}. The builder can be created by calling {@link
* CronetEngine#newUrlRequestBuilder}.
*/
public abstract static class Builder {
/**
* Sets the HTTP method verb to use for this request.
*
* <p>The default when this method is not called is "GET" if the request has
* no body or "POST" if it does.
*
* @param method "GET", "HEAD", "DELETE", "POST" or "PUT".
* @return the builder to facilitate chaining.
*/
public abstract Builder setHttpMethod(String method);
/**
* Adds a request header.
*
* @param header header name.
* @param value header value.
* @return the builder to facilitate chaining.
*/
public abstract Builder addHeader(String header, String value);
/**
* Disables cache for the request. If context is not set up to use cache, this call has no
* effect.
*
* @return the builder to facilitate chaining.
*/
public abstract Builder disableCache();
/** Lowest request priority. Passed to {@link #setPriority}. */
public static final int REQUEST_PRIORITY_IDLE = 0;
/** Very low request priority. Passed to {@link #setPriority}. */
public static final int REQUEST_PRIORITY_LOWEST = 1;
/** Low request priority. Passed to {@link #setPriority}. */
public static final int REQUEST_PRIORITY_LOW = 2;
/**
* Medium request priority. Passed to {@link #setPriority}. This is the default priority
* given to the request.
*/
public static final int REQUEST_PRIORITY_MEDIUM = 3;
/** Highest request priority. Passed to {@link #setPriority}. */
public static final int REQUEST_PRIORITY_HIGHEST = 4;
/**
* Sets priority of the request which should be one of the {@link #REQUEST_PRIORITY_IDLE
* REQUEST_PRIORITY_*} values. The request is given {@link #REQUEST_PRIORITY_MEDIUM}
* priority if this method is not called.
*
* @param priority priority of the request which should be one of the {@link
* #REQUEST_PRIORITY_IDLE REQUEST_PRIORITY_*} values.
* @return the builder to facilitate chaining.
*/
public abstract Builder setPriority(int priority);
/**
* Sets upload data provider. Switches method to "POST" if not explicitly set. Starting the
* request will throw an exception if a Content-Type header is not set.
*
* @param uploadDataProvider responsible for providing the upload data.
* @param executor All {@code uploadDataProvider} methods will be invoked using this {@code
* Executor}. May optionally be the same {@code Executor} the request itself is using.
* @return the builder to facilitate chaining.
*/
public abstract Builder setUploadDataProvider(
UploadDataProvider uploadDataProvider, Executor executor);
/**
* Marks that the executors this request will use to notify callbacks (for {@code
* UploadDataProvider}s and {@code UrlRequest.Callback}s) is intentionally performing inline
* execution, like Guava's directExecutor or {@link
* java.util.concurrent.ThreadPoolExecutor.CallerRunsPolicy}.
*
* <p><b>Warning:</b> This option makes it easy to accidentally block the network thread.
* It should not be used if your callbacks perform disk I/O, acquire locks, or call into
* other code you don't carefully control and audit.
*/
public abstract Builder allowDirectExecutor();
/**
* Associates the annotation object with this request. May add more than one. Passed through
* to a {@link RequestFinishedInfo.Listener}, see {@link
* RequestFinishedInfo#getAnnotations}.
*
* @param annotation an object to pass on to the {@link RequestFinishedInfo.Listener} with a
* {@link RequestFinishedInfo}.
* @return the builder to facilitate chaining.
*/
public Builder addRequestAnnotation(Object annotation) {
return this;
}
/**
* Binds the request to the specified network handle. Cronet will send this request only
* using the network associated to this handle. If this network disconnects the request will
* fail, the exact error will depend on the stage of request processing when the network
* disconnects. Network handles can be obtained through {@code Network#getNetworkHandle}.
* Only available starting from Android Marshmallow.
*
* @param networkHandle the network handle to bind the request to. Specify {@link
* CronetEngine#UNBIND_NETWORK_HANDLE} to unbind.
* @return the builder to facilitate chaining.
*/
public Builder bindToNetwork(long networkHandle) {
return this;
}
/**
* Sets {@link android.net.TrafficStats} tag to use when accounting socket traffic caused by
* this request. See {@link android.net.TrafficStats} for more information. If no tag is set
* (e.g. this method isn't called), then Android accounts for the socket traffic caused by
* this request as if the tag value were set to 0. <p> <b>NOTE:</b>Setting a tag disallows
* sharing of sockets with requests with other tags, which may adversely effect performance
* by prohibiting connection sharing. In other words use of multiplexed sockets (e.g. HTTP/2
* and QUIC) will only be allowed if all requests have the same socket tag.
*
* @param tag the tag value used to when accounting for socket traffic caused by this
* request.
* Tags between 0xFFFFFF00 and 0xFFFFFFFF are reserved and used internally by system
* services like {@link android.app.DownloadManager} when performing traffic on behalf of an
* application.
* @return the builder to facilitate chaining.
*/
public Builder setTrafficStatsTag(int tag) {
return this;
}
/**
* Sets specific UID to use when accounting socket traffic caused by this request. See
* {@link android.net.TrafficStats} for more information. Designed for use when performing
* an operation on behalf of another application. Caller must hold {@code
* MODIFY_NETWORK_ACCOUNTING} permission. By default traffic is attributed to UID of caller.
*
* <p><b>NOTE:</b>Setting a UID disallows sharing of sockets with requests with other UIDs,
* which may adversely effect performance by prohibiting connection sharing. In other words
* use of multiplexed sockets (e.g. HTTP/2 and QUIC) will only be allowed if all requests
* have the same UID set.
*
* @param uid the UID to attribute socket traffic caused by this request.
* @return the builder to facilitate chaining.
*/
public Builder setTrafficStatsUid(int uid) {
return this;
}
/**
* Sets a listener that gets invoked after {@link Callback#onCanceled onCanceled()}, {@link
* Callback#onFailed onFailed()} or {@link Callback#onSucceeded onSucceeded()} return.
*
* <p>The listener is invoked with the request finished info on an
* {@link java.util.concurrent.Executor} provided by {@link
* RequestFinishedInfo.Listener#getExecutor getExecutor()}.
*
* @param listener the listener for finished requests.
* @return the builder to facilitate chaining.
*/
public Builder setRequestFinishedListener(RequestFinishedInfo.Listener listener) {
return this;
}
/**
* Allows Cronet to use the specified compression dictionary for this request. When
* specified, Cronet might signal to the server the availability of said compression
* dictionary. For this to have any effect, the CronetEngine that will execute this request
* must have been configured to enable a compression scheme that supports external
* dictionaries. This partially implements draft-ietf-httpbis-compression-dictionary within
* Cronet. Cronet won't directly handle "Use-As-Dictionary" response headers, it will
* instead rely on the embedder to: either, handle them and later call
* UrlRequest.Builder#setCompressionDictionary for requests which match Use-As-Dictionary's
* rules; or, fetch compression dictionaries via some other out of band mechanism, and later
* call UrlRequest.Builder#setCompressionDictionary. Cronet will interpret the dictionary as
* matching only this UrlRequest. If said request is redirected, and the embedder decides to
* follow the redirect, the dictionary will match also the new URL.
*
* @param dictionarySha256Hash the SHA-256 of the specified compression dictionary.
* @param dictionary the compression dictionary that Cronet can use for this UrlRequest.
* This must be a direct ByteBuffer.
* @param dictionaryId the optional ID associated with this dictionary, must be an empty
* string if missing. If present, this will be sent via the Dictionary-ID header. You
* would need to specify this if the server specified an id in its "Use-As-Dictionary"
* response.
* @return the builder to facilitate chaining.
*/
@Experimental
public Builder setRawCompressionDictionary(
byte[] dictionarySha256Hash, ByteBuffer dictionary, String dictionaryId) {
return this;
}
/**
* Creates a {@link UrlRequest} using configuration within this {@link Builder}. The
* returned {@code UrlRequest} can then be started by calling {@link UrlRequest#start}.
*
* @return constructed {@link UrlRequest} using configuration within this {@link Builder}.
*/
public abstract UrlRequest build();
}
/**
* Users of Cronet extend this class to receive callbacks indicating the progress of a {@link
* UrlRequest} being processed. An instance of this class is passed in to {@link
* UrlRequest.Builder}'s constructor when constructing the {@code UrlRequest}.
* <p>
* Note: All methods will be invoked on the thread of the {@link java.util.concurrent.Executor}
* used during construction of the {@code UrlRequest}.
*/
public abstract static class Callback {
/**
* Invoked whenever a redirect is encountered. This will only be invoked between the call to
* {@link UrlRequest#start} and {@link Callback#onResponseStarted onResponseStarted()}. The
* body of the redirect response, if it has one, will be ignored.
*
* The redirect will not be followed until the URLRequest's {@link
* UrlRequest#followRedirect} method is called, either synchronously or asynchronously.
*
* @param request Request being redirected.
* @param info Response information.
* @param newLocationUrl Location where request is redirected.
* @throws Exception if an error occurs while processing a redirect. {@link #onFailed} will
* be
* called with the thrown exception set as the cause of the {@link CallbackException}.
*/
public abstract void onRedirectReceived(
UrlRequest request, UrlResponseInfo info, String newLocationUrl) throws Exception;
/**
* Invoked when the final set of headers, after all redirects, is received. Will only be
* invoked once for each request.
*
* With the exception of {@link Callback#onCanceled onCanceled()}, no other {@link Callback}
* method will be invoked for the request, including {@link Callback#onSucceeded
* onSucceeded()} and {@link Callback#onFailed onFailed()}, until {@link UrlRequest#read
* UrlRequest.read()} is called to attempt to start reading the response body.
*
* @param request Request that started to get response.
* @param info Response information.
* @throws Exception if an error occurs while processing response start. {@link #onFailed}
* will
* be called with the thrown exception set as the cause of the {@link CallbackException}.
*/
public abstract void onResponseStarted(UrlRequest request, UrlResponseInfo info)
throws Exception;
/**
* Invoked whenever part of the response body has been read. Only part of the buffer may be
* populated, even if the entire response body has not yet been consumed.
*
* With the exception of {@link Callback#onCanceled onCanceled()}, no other {@link Callback}
* method will be invoked for the request, including {@link Callback#onSucceeded
* onSucceeded()} and {@link Callback#onFailed onFailed()}, until {@link UrlRequest#read
* UrlRequest.read()} is called to attempt to continue reading the response body.
*
* @param request Request that received data.
* @param info Response information.
* @param byteBuffer The buffer that was passed in to {@link UrlRequest#read
* UrlRequest.read()},
* now containing the received data. The buffer's position is updated to the end of the
* received data. The buffer's limit is not changed.
* @throws Exception if an error occurs while processing a read completion. {@link
* #onFailed}
* will be called with the thrown exception set as the cause of the {@link
* CallbackException}.
*/
public abstract void onReadCompleted(
UrlRequest request, UrlResponseInfo info, ByteBuffer byteBuffer) throws Exception;
/**
* Invoked when request is completed successfully. Once invoked, no other {@link Callback}
* methods will be invoked.
*
* @param request Request that succeeded.
* @param info Response information.
*/
public abstract void onSucceeded(UrlRequest request, UrlResponseInfo info);
/**
* Invoked if request failed for any reason after {@link UrlRequest#start}. Once invoked, no
* other {@link Callback} methods will be invoked. {@code error} provides information about
* the failure.
*
* @param request Request that failed.
* @param info Response information. May be {@code null} if no response was received.
* @param error information about error.
*/
public abstract void onFailed(
UrlRequest request, UrlResponseInfo info, CronetException error);
/**
* Invoked if request was canceled via {@link UrlRequest#cancel}. Once invoked, no other
* {@link Callback} methods will be invoked. Default implementation takes no action.
*
* @param request Request that was canceled.
* @param info Response information. May be {@code null} if no response was received.
*/
public void onCanceled(UrlRequest request, UrlResponseInfo info) {}
}
/** Request status values returned by {@link #getStatus}. */
public static class Status {
/** This state indicates that the request is completed, canceled, or is not started. */
public static final int INVALID = -1;
/**
* This state corresponds to a resource load that has either not yet begun or is idle
* waiting for the consumer to do something to move things along (e.g. when the consumer of
* a {@link UrlRequest} has not called {@link UrlRequest#read read()} yet).
*/
public static final int IDLE = 0;
/**
* When a socket pool group is below the maximum number of sockets allowed per group, but a
* new socket cannot be created due to the per-pool socket limit, this state is returned by
* all requests for the group waiting on an idle connection, except those that may be
* serviced by a pending new connection.
*/
public static final int WAITING_FOR_STALLED_SOCKET_POOL = 1;
/**
* When a socket pool group has reached the maximum number of sockets allowed per group,
* this state is returned for all requests that don't have a socket, except those that
* correspond to a pending new connection.
*/
public static final int WAITING_FOR_AVAILABLE_SOCKET = 2;
/**
* This state indicates that the URLRequest delegate has chosen to block this request before
* it was sent over the network.
*/
public static final int WAITING_FOR_DELEGATE = 3;
/**
* This state corresponds to a resource load that is blocked waiting for access to a
* resource in the cache. If multiple requests are made for the same resource, the first
* request will be responsible for writing (or updating) the cache entry and the second
* request will be deferred until the first completes. This may be done to optimize for
* cache reuse.
*/
public static final int WAITING_FOR_CACHE = 4;
/**
* This state corresponds to a resource being blocked waiting for the PAC script to be
* downloaded.
*/
public static final int DOWNLOADING_PAC_FILE = 5;
/**
* This state corresponds to a resource load that is blocked waiting for a proxy autoconfig
* script to return a proxy server to use.
*/
public static final int RESOLVING_PROXY_FOR_URL = 6;
/**
* This state corresponds to a resource load that is blocked waiting for a proxy autoconfig
* script to return a proxy server to use, but that proxy script is busy resolving the IP
* address of a host.
*/
public static final int RESOLVING_HOST_IN_PAC_FILE = 7;
/**
* This state indicates that we're in the process of establishing a tunnel through the proxy
* server.
*/
public static final int ESTABLISHING_PROXY_TUNNEL = 8;
/**
* This state corresponds to a resource load that is blocked waiting for a host name to be
* resolved. This could either indicate resolution of the origin server corresponding to the
* resource or to the host name of a proxy server used to fetch the resource.
*/
public static final int RESOLVING_HOST = 9;
/**
* This state corresponds to a resource load that is blocked waiting for a TCP connection
* (or other network connection) to be established. HTTP requests that reuse a keep-alive
* connection skip this state.
*/
public static final int CONNECTING = 10;
/**
* This state corresponds to a resource load that is blocked waiting for the SSL handshake
* to complete.
*/
public static final int SSL_HANDSHAKE = 11;
/**
* This state corresponds to a resource load that is blocked waiting to completely upload a
* request to a server. In the case of a HTTP POST request, this state includes the period
* of time during which the message body is being uploaded.
*/
public static final int SENDING_REQUEST = 12;
/**
* This state corresponds to a resource load that is blocked waiting for the response to a
* network request. In the case of a HTTP transaction, this corresponds to the period after
* the request is sent and before all of the response headers have been received.
*/
public static final int WAITING_FOR_RESPONSE = 13;
/**
* This state corresponds to a resource load that is blocked waiting for a read to complete.
* In the case of a HTTP transaction, this corresponds to the period after the response
* headers have been received and before all of the response body has been downloaded.
* (NOTE: This state only applies for an {@link UrlRequest} while there is an outstanding
* {@link UrlRequest#read read()} operation.)
*/
public static final int READING_RESPONSE = 14;
private Status() {}
}
/** Listener class used with {@link #getStatus} to receive the status of a {@link UrlRequest}. */
public abstract static class StatusListener {
/**
* Invoked on {@link UrlRequest}'s {@link Executor}'s thread when request status is
* obtained.
*
* @param status integer representing the status of the request. It is one of the values
* defined
* in {@link Status}.
*/
public abstract void onStatus(int status);
}
/**
* Starts the request, all callbacks go to {@link Callback}. May only be called once. May not be
* called if {@link #cancel} has been called.
*/
public abstract void start();
/**
* Follows a pending redirect. Must only be called at most once for each invocation of {@link
* Callback#onRedirectReceived onRedirectReceived()}.
*/
public abstract void followRedirect();
/**
* Attempts to read part of the response body into the provided buffer. Must only be called at
* most once in response to each invocation of the {@link Callback#onResponseStarted
* onResponseStarted()} and {@link Callback#onReadCompleted onReadCompleted()} methods of the
* {@link Callback}. Each call will result in an asynchronous call to either the {@link Callback
* Callback's} {@link Callback#onReadCompleted onReadCompleted()} method if data is read, its
* {@link Callback#onSucceeded onSucceeded()} method if there's no more data to read, or its
* {@link Callback#onFailed onFailed()} method if there's an error.
*
* @param buffer {@link ByteBuffer} to write response body to. Must be a direct ByteBuffer. The
* embedder must not read or modify buffer's position, limit, or data between its position and
* limit until the request calls back into the {@link Callback}.
*/
public abstract void read(ByteBuffer buffer);
/**
* Cancels the request. Can be called at any time. {@link Callback#onCanceled onCanceled()} will
* be invoked when cancellation is complete and no further callback methods will be invoked. If
* the request has completed or has not started, calling {@code cancel()} has no effect and
* {@code onCanceled()} will not be invoked. If the {@link Executor} passed in during {@code
* UrlRequest} construction runs tasks on a single thread, and {@code cancel()} is called on
* that thread, no callback methods (besides {@code onCanceled()}) will be invoked after {@code
* cancel()} is called. Otherwise, at most one callback method may be invoked after {@code
* cancel()} has completed.
*/
public abstract void cancel();
/**
* Returns {@code true} if the request was successfully started and is now finished (completed,
* canceled, or failed).
*
* @return {@code true} if the request was successfully started and is now finished (completed,
* canceled, or failed).
*/
public abstract boolean isDone();
/**
* Queries the status of the request.
*
* @param listener a {@link StatusListener} that will be invoked with the request's current
* status. {@code listener} will be invoked back on the {@link Executor} passed in when the
* request was created.
*/
public abstract void getStatus(final StatusListener listener);
// Note: There are deliberately no accessors for the results of the request
// here. Having none removes any ambiguity over when they are populated,
// particularly in the redirect case.
/**
* An annotation for APIs which are not considered stable yet.
*
* <p>Experimental APIs are subject to change, breakage, or removal at any time and may not be
* production ready.
*
* <p>It's highly recommended to reach out to Cronet maintainers (<code>[email protected]
* </code>) before using one of the APIs annotated as experimental outside of debugging and
* proof-of-concept code.
*
* <p>By using an Experimental API, applications acknowledge that they are doing so at their own
* risk.
*/
@RequiresOptIn
public @interface Experimental {}
}