Index: net/cronet/android/org_chromium_net_UrlRequest.cc |
diff --git a/net/cronet/android/org_chromium_net_UrlRequest.cc b/net/cronet/android/org_chromium_net_UrlRequest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..b46e6a6a440ac7137f8c02c600c36d751959a449 |
--- /dev/null |
+++ b/net/cronet/android/org_chromium_net_UrlRequest.cc |
@@ -0,0 +1,364 @@ |
+// Copyright 2013 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 "net/cronet/android/org_chromium_net_UrlRequest.h" |
+ |
+#include <stdio.h> |
+ |
+#include "base/macros.h" |
+#include "net/base/net_errors.h" |
+#include "net/base/request_priority.h" |
+#include "net/cronet/android/org_chromium_net_UrlRequestContext.h" |
+#include "net/cronet/android/url_request_context_peer.h" |
+#include "net/cronet/android/url_request_peer.h" |
+ |
+// TODO(mef): Replace following definitions with generated UrlRequest_jni.h |
+//#include "jni/UrlRequest_jni.h" |
+namespace { |
+ |
+jclass g_class; |
+jmethodID g_method_finish; |
+jmethodID g_method_onAppendChunkCompleted; |
+jmethodID g_method_onResponseStarted; |
+jmethodID g_method_onReadBytes; |
+jclass g_class_OutputStream; |
+jmethodID g_method_write; |
+jfieldID g_request_field; |
+ |
+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; |
+ } |
+} |
+ |
+// Stores a reference to the request in a java field. |
+void SetNativeObject(JNIEnv* env, jobject object, URLRequestPeer* request) { |
+ env->SetLongField(object, g_request_field, reinterpret_cast<jlong>(request)); |
+} |
+ |
+// Returns a reference to the request, which is stored in a field of the Java |
+// object. |
+URLRequestPeer* GetNativeObject(JNIEnv* env, jobject object) { |
+ return reinterpret_cast<URLRequestPeer*>( |
+ env->GetLongField(object, g_request_field)); |
+} |
+ |
+void SetPostContentType(JNIEnv* env, |
+ URLRequestPeer* request, |
+ jstring content_type) { |
+ DCHECK(request != NULL); |
+ |
+ std::string method_post("POST"); |
+ request->SetMethod(method_post); |
+ |
+ std::string content_type_header("Content-Type"); |
+ |
+ const char* content_type_utf8 = env->GetStringUTFChars(content_type, NULL); |
+ std::string content_type_string(content_type_utf8); |
+ env->ReleaseStringUTFChars(content_type, content_type_utf8); |
+ |
+ request->AddHeader(content_type_header, content_type_string); |
+} |
+ |
+} // namespace |
+ |
+// Find Java classes and retain them. |
+bool UrlRequestRegisterJni(JNIEnv* env) { |
+ g_class = reinterpret_cast<jclass>( |
+ env->NewGlobalRef(env->FindClass("org/chromium/net/UrlRequest"))); |
+ g_method_finish = env->GetMethodID(g_class, "finish", "()V"); |
+ g_method_onAppendChunkCompleted = |
+ env->GetMethodID(g_class, "onAppendChunkCompleted", "()V"); |
+ g_method_onResponseStarted = |
+ env->GetMethodID(g_class, "onResponseStarted", "()V"); |
+ g_method_onReadBytes = |
+ env->GetMethodID(g_class, "onBytesRead", "(Ljava/nio/ByteBuffer;)V"); |
+ g_request_field = env->GetFieldID(g_class, "mRequest", "J"); |
+ |
+ g_class_OutputStream = reinterpret_cast<jclass>( |
+ env->NewGlobalRef(env->FindClass("java/io/OutputStream"))); |
+ g_method_write = env->GetMethodID(g_class_OutputStream, "write", "([BII)V"); |
+ |
+ if (!g_class || !g_method_finish || !g_method_onAppendChunkCompleted || |
+ !g_method_onResponseStarted || !g_method_onReadBytes || |
+ !g_request_field || !g_class_OutputStream || !g_method_write) { |
+ return false; |
+ } |
+ return true; |
+} |
+ |
+// A delegate of URLRequestPeer that delivers callbacks to the Java layer. |
+class JniURLRequestPeerDelegate |
+ : public URLRequestPeer::URLRequestPeerDelegate { |
+ public: |
+ JniURLRequestPeerDelegate(JNIEnv* env, jobject owner) { |
+ owner_ = env->NewGlobalRef(owner); |
+ env->GetJavaVM(&vm_); |
+ } |
+ |
+ virtual void OnAppendChunkCompleted(URLRequestPeer* request) OVERRIDE { |
+ JNIEnv* env = GetEnv(vm_); |
+ env->CallVoidMethod(owner_, g_method_onAppendChunkCompleted); |
+ if (env->ExceptionOccurred()) { |
+ env->ExceptionDescribe(); |
+ env->ExceptionClear(); |
+ } |
+ } |
+ |
+ virtual void OnResponseStarted(URLRequestPeer* request) OVERRIDE { |
+ JNIEnv* env = GetEnv(vm_); |
+ env->CallVoidMethod(owner_, g_method_onResponseStarted); |
+ if (env->ExceptionOccurred()) { |
+ env->ExceptionDescribe(); |
+ env->ExceptionClear(); |
+ } |
+ } |
+ |
+ virtual void OnBytesRead(URLRequestPeer* request) OVERRIDE { |
+ int bytes_read = request->bytes_read(); |
+ if (bytes_read != 0) { |
+ JNIEnv* env = GetEnv(vm_); |
+ jobject bytebuf = env->NewDirectByteBuffer(request->Data(), bytes_read); |
+ env->CallVoidMethod(owner_, g_method_onReadBytes, bytebuf); |
+ env->DeleteLocalRef(bytebuf); |
+ if (env->ExceptionOccurred()) { |
+ env->ExceptionDescribe(); |
+ env->ExceptionClear(); |
+ } |
+ } |
+ } |
+ |
+ virtual void OnRequestFinished(URLRequestPeer* request) OVERRIDE { |
+ JNIEnv* env = GetEnv(vm_); |
+ env->CallVoidMethod(owner_, g_method_finish); |
+ if (env->ExceptionOccurred()) { |
+ env->ExceptionDescribe(); |
+ env->ExceptionClear(); |
+ } |
+ } |
+ |
+ protected: |
+ virtual ~JniURLRequestPeerDelegate() { GetEnv(vm_)->DeleteGlobalRef(owner_); } |
+ |
+ private: |
+ jobject owner_; |
+ JavaVM* vm_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(JniURLRequestPeerDelegate); |
+}; |
+ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeInit(JNIEnv* env, |
+ jobject object, |
+ jobject request_context, |
+ jstring url_string, |
+ jint priority) { |
+ URLRequestContextPeer* context = |
+ GetURLRequestContextPeer(env, request_context); |
+ DCHECK(context != NULL); |
+ |
+ const char* url_utf8 = env->GetStringUTFChars(url_string, NULL); |
+ |
+ DVLOG(context->logging_level()) |
+ << "New chromium network request. URL:" << url_utf8; |
+ |
+ GURL url(url_utf8); |
+ |
+ env->ReleaseStringUTFChars(url_string, url_utf8); |
+ |
+ URLRequestPeer* request = |
+ new URLRequestPeer(context, |
+ new JniURLRequestPeerDelegate(env, object), |
+ url, |
+ ConvertRequestPriority(priority)); |
+ |
+ SetNativeObject(env, object, request); |
+} |
+ |
+// synchronized |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeAddHeader(JNIEnv* env, |
+ jobject object, |
+ jstring name, |
+ jstring value) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ DCHECK(request != NULL); |
+ |
+ const char* name_utf8 = env->GetStringUTFChars(name, NULL); |
+ std::string name_string(name_utf8); |
+ env->ReleaseStringUTFChars(name, name_utf8); |
+ |
+ const char* value_utf8 = env->GetStringUTFChars(value, NULL); |
+ std::string value_string(value_utf8); |
+ env->ReleaseStringUTFChars(value, value_utf8); |
+ |
+ request->AddHeader(name_string, value_string); |
+} |
+ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeSetPostData(JNIEnv* env, |
+ jobject object, |
+ jstring content_type, |
+ jbyteArray content) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ SetPostContentType(env, request, content_type); |
+ |
+ if (content != NULL) { |
+ jsize size = env->GetArrayLength(content); |
+ if (size > 0) { |
+ jbyte* content_bytes = env->GetByteArrayElements(content, NULL); |
+ request->SetPostContent(reinterpret_cast<const char*>(content_bytes), |
+ size); |
+ env->ReleaseByteArrayElements(content, content_bytes, 0); |
+ } |
+ } |
+} |
+ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeBeginChunkedUpload( |
+ JNIEnv* env, |
+ jobject object, |
+ jstring content_type) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ SetPostContentType(env, request, content_type); |
+ |
+ request->EnableStreamingUpload(); |
+} |
+ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeAppendChunk(JNIEnv* env, |
+ jobject object, |
+ jobject chunk_byte_buffer, |
+ jint chunk_size, |
+ jboolean is_last_chunk) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ CHECK(request != NULL); |
+ |
+ if (chunk_byte_buffer != NULL) { |
+ void* chunk = env->GetDirectBufferAddress(chunk_byte_buffer); |
+ request->AppendChunk( |
+ reinterpret_cast<const char*>(chunk), chunk_size, is_last_chunk); |
+ } |
+} |
+ |
+/* synchronized */ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeStart(JNIEnv* env, jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ if (request != NULL) { |
+ request->Start(); |
+ } |
+} |
+ |
+/* synchronized */ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeRecycle(JNIEnv* env, jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ if (request != NULL) { |
+ request->Destroy(); |
+ } |
+ |
+ SetNativeObject(env, object, NULL); |
+} |
+ |
+/* synchronized */ |
+JNIEXPORT void JNICALL |
+Java_org_chromium_net_UrlRequest_nativeCancel(JNIEnv* env, jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ if (request != NULL) { |
+ request->Cancel(); |
+ } |
+} |
+ |
+JNIEXPORT jint JNICALL |
+Java_org_chromium_net_UrlRequest_nativeGetErrorCode(JNIEnv* env, |
+ jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ int error_code = request->error_code(); |
+ switch (error_code) { |
+ // TODO(mef): Investigate returning success on positive values, too, as |
+ // they technically indicate success. |
+ case net::OK: |
+ return 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 ERROR_SUCCESS; |
+ |
+ case net::ERR_INVALID_URL: |
+ case net::ERR_DISALLOWED_URL_SCHEME: |
+ case net::ERR_UNKNOWN_URL_SCHEME: |
+ return ERROR_MALFORMED_URL; |
+ |
+ case net::ERR_CONNECTION_TIMED_OUT: |
+ return ERROR_CONNECTION_TIMED_OUT; |
+ |
+ case net::ERR_NAME_NOT_RESOLVED: |
+ return ERROR_UNKNOWN_HOST; |
+ } |
+ return ERROR_UNKNOWN; |
+} |
+ |
+JNIEXPORT jstring JNICALL |
+Java_org_chromium_net_UrlRequest_nativeGetErrorString(JNIEnv* env, |
+ jobject object) { |
+ int error_code = GetNativeObject(env, object)->error_code(); |
+ char buffer[200]; |
+ snprintf(buffer, |
+ sizeof(buffer), |
+ "System error: %s(%d)", |
+ net::ErrorToString(error_code), |
+ error_code); |
+ return env->NewStringUTF(buffer); |
+} |
+ |
+JNIEXPORT jint JNICALL |
+Java_org_chromium_net_UrlRequest_getHttpStatusCode(JNIEnv* env, |
+ jobject object) { |
+ return GetNativeObject(env, object)->http_status_code(); |
+} |
+ |
+JNIEXPORT jstring JNICALL |
+Java_org_chromium_net_UrlRequest_nativeGetContentType(JNIEnv* env, |
+ jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ if (request == NULL) { |
+ return NULL; |
+ } |
+ std::string type = request->content_type(); |
+ if (!type.empty()) { |
+ return env->NewStringUTF(type.c_str()); |
+ } else { |
+ return NULL; |
+ } |
+} |
+ |
+JNIEXPORT jlong JNICALL |
+Java_org_chromium_net_UrlRequest_nativeGetContentLength(JNIEnv* env, |
+ jobject object) { |
+ URLRequestPeer* request = GetNativeObject(env, object); |
+ if (request == NULL) { |
+ return 0; |
+ } |
+ return request->content_length(); |
+} |