blob: 6f047e6a680efb9398947f09d667cfa050863677 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/cronet/android/chromium_url_request.h"
#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/macros.h"
#include "components/cronet/android/url_request_adapter.h"
#include "components/cronet/android/url_request_context_adapter.h"
#include "jni/ChromiumUrlRequest_jni.h"
#include "net/base/net_errors.h"
#include "net/base/request_priority.h"
#include "net/http/http_response_headers.h"
using base::android::ConvertUTF8ToJavaString;
using base::android::ConvertJavaStringToUTF8;
namespace cronet {
namespace {
net::RequestPriority ConvertRequestPriority(jint request_priority) {
switch (request_priority) {
case REQUEST_PRIORITY_IDLE:
return net::IDLE;
case REQUEST_PRIORITY_LOWEST:
return net::LOWEST;
case REQUEST_PRIORITY_LOW:
return net::LOW;
case REQUEST_PRIORITY_MEDIUM:
return net::MEDIUM;
case REQUEST_PRIORITY_HIGHEST:
return net::HIGHEST;
default:
return net::LOWEST;
}
}
void SetPostContentType(JNIEnv* env,
URLRequestAdapter* request_adapter,
jstring content_type) {
std::string method_post("POST");
request_adapter->SetMethod(method_post);
std::string content_type_header("Content-Type");
std::string content_type_string(ConvertJavaStringToUTF8(env, content_type));
request_adapter->AddHeader(content_type_header, content_type_string);
}
// A delegate of URLRequestAdapter that delivers callbacks to the Java layer.
class JniURLRequestAdapterDelegate
: public URLRequestAdapter::URLRequestAdapterDelegate {
public:
JniURLRequestAdapterDelegate(JNIEnv* env, jobject owner) {
owner_ = env->NewGlobalRef(owner);
}
void OnResponseStarted(URLRequestAdapter* request_adapter) override {
JNIEnv* env = base::android::AttachCurrentThread();
cronet::Java_ChromiumUrlRequest_onResponseStarted(env, owner_);
}
void OnBytesRead(URLRequestAdapter* request_adapter,
int bytes_read) override {
if (bytes_read != 0) {
JNIEnv* env = base::android::AttachCurrentThread();
base::android::ScopedJavaLocalRef<jobject> java_buffer(
env, env->NewDirectByteBuffer(request_adapter->Data(), bytes_read));
cronet::Java_ChromiumUrlRequest_onBytesRead(
env, owner_, java_buffer.obj());
}
}
void OnRequestFinished(URLRequestAdapter* request_adapter) override {
JNIEnv* env = base::android::AttachCurrentThread();
cronet::Java_ChromiumUrlRequest_finish(env, owner_);
}
int ReadFromUploadChannel(net::IOBuffer* buf, int buf_length) override {
JNIEnv* env = base::android::AttachCurrentThread();
base::android::ScopedJavaLocalRef<jobject> java_buffer(
env, env->NewDirectByteBuffer(buf->data(), buf_length));
jint bytes_read = cronet::Java_ChromiumUrlRequest_readFromUploadChannel(
env, owner_, java_buffer.obj());
return bytes_read;
}
protected:
~JniURLRequestAdapterDelegate() override {
JNIEnv* env = base::android::AttachCurrentThread();
env->DeleteGlobalRef(owner_);
}
private:
jobject owner_;
DISALLOW_COPY_AND_ASSIGN(JniURLRequestAdapterDelegate);
};
} // namespace
// Explicitly register static JNI functions.
bool ChromiumUrlRequestRegisterJni(JNIEnv* env) {
return RegisterNativesImpl(env);
}
static jlong CreateRequestAdapter(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_context_adapter,
const JavaParamRef<jstring>& jurl,
jint jrequest_priority) {
URLRequestContextAdapter* context_adapter =
reinterpret_cast<URLRequestContextAdapter*>(jurl_request_context_adapter);
DCHECK(context_adapter);
GURL url(ConvertJavaStringToUTF8(env, jurl));
VLOG(1) << "New chromium network request: " << url.possibly_invalid_spec();
URLRequestAdapter* adapter = new URLRequestAdapter(
context_adapter, new JniURLRequestAdapterDelegate(env, jcaller), url,
ConvertRequestPriority(jrequest_priority));
return reinterpret_cast<jlong>(adapter);
}
// synchronized
static void AddHeader(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jheader_name,
const JavaParamRef<jstring>& jheader_value) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
std::string header_name(ConvertJavaStringToUTF8(env, jheader_name));
std::string header_value(ConvertJavaStringToUTF8(env, jheader_value));
request_adapter->AddHeader(header_name, header_value);
}
static void SetMethod(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jmethod) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
std::string method(ConvertJavaStringToUTF8(env, jmethod));
request_adapter->SetMethod(method);
}
static void SetUploadData(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jcontent_type,
const JavaParamRef<jbyteArray>& jcontent) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
SetPostContentType(env, request_adapter, jcontent_type);
if (jcontent != NULL) {
jsize size = env->GetArrayLength(jcontent);
if (size > 0) {
jbyte* content_bytes = env->GetByteArrayElements(jcontent, NULL);
request_adapter->SetUploadContent(
reinterpret_cast<const char*>(content_bytes), size);
env->ReleaseByteArrayElements(jcontent, content_bytes, 0);
}
}
}
static void SetUploadChannel(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jcontent_type,
jlong jcontent_length) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
SetPostContentType(env, request_adapter, jcontent_type);
request_adapter->SetUploadChannel(env, jcontent_length);
}
static void EnableChunkedUpload(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jcontent_type) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
SetPostContentType(env, request_adapter, jcontent_type);
request_adapter->EnableChunkedUpload();
}
static void AppendChunk(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jobject>& jchunk_byte_buffer,
jint jchunk_size,
jboolean jis_last_chunk) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
DCHECK(jchunk_byte_buffer);
void* chunk = env->GetDirectBufferAddress(jchunk_byte_buffer);
request_adapter->AppendChunk(reinterpret_cast<const char*>(chunk),
jchunk_size, jis_last_chunk);
}
/* synchronized */
static void Start(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
if (request_adapter != NULL)
request_adapter->Start();
}
/* synchronized */
static void DestroyRequestAdapter(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
if (request_adapter != NULL)
request_adapter->Destroy();
}
/* synchronized */
static void Cancel(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
if (request_adapter != NULL)
request_adapter->Cancel();
}
static jint GetErrorCode(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
int error_code = request_adapter->error_code();
switch (error_code) {
// TODO(mef): Investigate returning success on positive values, too, as
// they technically indicate success.
case net::OK:
return REQUEST_ERROR_SUCCESS;
// TODO(mef): Investigate this. The fact is that Chrome does not do this,
// and this library is not just being used for downloads.
// Comment from src/content/browser/download/download_resource_handler.cc:
// ERR_CONTENT_LENGTH_MISMATCH and ERR_INCOMPLETE_CHUNKED_ENCODING are
// allowed since a number of servers in the wild close the connection too
// early by mistake. Other browsers - IE9, Firefox 11.0, and Safari 5.1.4 -
// treat downloads as complete in both cases, so we follow their lead.
case net::ERR_CONTENT_LENGTH_MISMATCH:
case net::ERR_INCOMPLETE_CHUNKED_ENCODING:
return REQUEST_ERROR_SUCCESS;
case net::ERR_INVALID_URL:
case net::ERR_DISALLOWED_URL_SCHEME:
case net::ERR_UNKNOWN_URL_SCHEME:
return REQUEST_ERROR_MALFORMED_URL;
case net::ERR_CONNECTION_TIMED_OUT:
return REQUEST_ERROR_CONNECTION_TIMED_OUT;
case net::ERR_NAME_NOT_RESOLVED:
return REQUEST_ERROR_UNKNOWN_HOST;
case net::ERR_TOO_MANY_REDIRECTS:
return REQUEST_ERROR_TOO_MANY_REDIRECTS;
}
return REQUEST_ERROR_UNKNOWN;
}
static ScopedJavaLocalRef<jstring> GetErrorString(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
int error_code = request_adapter->error_code();
char buffer[200];
std::string error_string = net::ErrorToString(error_code);
snprintf(buffer,
sizeof(buffer),
"System error: %s(%d)",
error_string.c_str(),
error_code);
return ConvertUTF8ToJavaString(env, buffer);
}
static jint GetHttpStatusCode(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
return request_adapter->http_status_code();
}
static ScopedJavaLocalRef<jstring> GetHttpStatusText(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
return ConvertUTF8ToJavaString(env, request_adapter->http_status_text());
}
static ScopedJavaLocalRef<jstring> GetContentType(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
std::string type = request_adapter->content_type();
if (!type.empty()) {
return ConvertUTF8ToJavaString(env, type.c_str());
} else {
return ScopedJavaLocalRef<jstring>();
}
}
static jlong GetContentLength(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
return request_adapter->content_length();
}
static ScopedJavaLocalRef<jstring> GetHeader(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jstring>& jheader_name) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
std::string header_name = ConvertJavaStringToUTF8(env, jheader_name);
std::string header_value = request_adapter->GetHeader(header_name);
if (!header_value.empty())
return ConvertUTF8ToJavaString(env, header_value.c_str());
return ScopedJavaLocalRef<jstring>();
}
static void GetAllHeaders(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter,
const JavaParamRef<jobject>& jheaders_map) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
net::HttpResponseHeaders* headers = request_adapter->GetResponseHeaders();
if (headers == NULL)
return;
size_t iter = 0;
std::string header_name;
std::string header_value;
while (headers->EnumerateHeaderLines(&iter, &header_name, &header_value)) {
ScopedJavaLocalRef<jstring> name =
ConvertUTF8ToJavaString(env, header_name);
ScopedJavaLocalRef<jstring> value =
ConvertUTF8ToJavaString(env, header_value);
Java_ChromiumUrlRequest_onAppendResponseHeader(env, jcaller, jheaders_map,
name.obj(), value.obj());
}
// Some implementations (notably HttpURLConnection) include a mapping for the
// null key; in HTTP's case, this maps to the HTTP status line.
ScopedJavaLocalRef<jstring> status_line =
ConvertUTF8ToJavaString(env, headers->GetStatusLine());
Java_ChromiumUrlRequest_onAppendResponseHeader(env, jcaller, jheaders_map,
NULL, status_line.obj());
}
static ScopedJavaLocalRef<jstring> GetNegotiatedProtocol(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
std::string negotiated_protocol = request_adapter->GetNegotiatedProtocol();
return ConvertUTF8ToJavaString(env, negotiated_protocol.c_str());
}
static jboolean GetWasCached(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jurl_request_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jurl_request_adapter);
DCHECK(request_adapter);
bool was_cached = request_adapter->GetWasCached();
return was_cached ? JNI_TRUE : JNI_FALSE;
}
static void DisableRedirects(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jlong jrequest_adapter) {
URLRequestAdapter* request_adapter =
reinterpret_cast<URLRequestAdapter*>(jrequest_adapter);
DCHECK(request_adapter);
request_adapter->DisableRedirects();
}
} // namespace cronet