Chromium Code Reviews| Index: components/cronet/android/test/upload_data_stream_handler.cc |
| diff --git a/components/cronet/android/test/upload_data_stream_handler.cc b/components/cronet/android/test/upload_data_stream_handler.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..509ecf7a8bcdae71d31ce05339ada1ee6eaba37c |
| --- /dev/null |
| +++ b/components/cronet/android/test/upload_data_stream_handler.cc |
| @@ -0,0 +1,270 @@ |
| +// 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 "upload_data_stream_handler.h" |
| + |
| +#include <string> |
| + |
| +#include "base/android/jni_android.h" |
| +#include "base/android/jni_string.h" |
| +#include "base/android/scoped_java_ref.h" |
| +#include "base/bind.h" |
| +#include "base/callback_helpers.h" |
| +#include "base/location.h" |
| +#include "base/logging.h" |
| +#include "base/macros.h" |
| +#include "base/memory/ref_counted.h" |
| +#include "base/memory/weak_ptr.h" |
| +#include "base/message_loop/message_loop_proxy.h" |
| +#include "components/cronet/android/cronet_upload_data_stream_adapter.h" |
| +#include "components/cronet/android/cronet_url_request_adapter.h" |
| +#include "jni/UploadDataStreamHandler_jni.h" |
| +#include "net/base/io_buffer.h" |
| +#include "net/base/net_errors.h" |
| +#include "net/base/test_completion_callback.h" |
| +#include "net/base/upload_data_stream.h" |
| + |
| +namespace cronet { |
| + |
| +static const size_t kReadBufferSize = 32768; |
| + |
| +bool UploadDataStreamHandlerRegisterJni(JNIEnv* env) { |
|
mmenke
2015/01/21 18:48:58
Declaration order should match definition order -
xunjieli
2015/01/21 23:49:24
Done.
|
| + return RegisterNativesImpl(env); |
| +}; |
| + |
| +class UploadDataStreamHandlerDelegate |
| + : public UploadDataStreamHandler::Delegate { |
| + public: |
| + UploadDataStreamHandlerDelegate(JNIEnv* env, |
| + jobject jupload_data_stream_handler) { |
| + jupload_data_stream_handler_.Reset(env, jupload_data_stream_handler); |
| + } |
| + |
| + ~UploadDataStreamHandlerDelegate() override {} |
| + |
| + void OnInitCalled(int res) { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onInitCalled( |
| + env, jupload_data_stream_handler_.obj(), res); |
| + } |
| + |
| + void OnInitCompleted(int res) { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onInitCompleted( |
| + env, jupload_data_stream_handler_.obj(), res); |
| + } |
| + |
| + void OnReadCompleted(int res) { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onReadCompleted( |
| + env, jupload_data_stream_handler_.obj(), res); |
| + } |
| + |
| + void OnResetCompleted() { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onResetCompleted( |
| + env, jupload_data_stream_handler_.obj()); |
| + } |
| + |
| + void OnCheckIfInitCallbackInvoked(bool res) { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onCheckIfInitCallbackInvoked( |
| + env, jupload_data_stream_handler_.obj(), res); |
| + } |
| + |
| + void OnCheckIfReadCallbackInvoked(bool res) { |
| + JNIEnv* env = base::android::AttachCurrentThread(); |
| + cronet::Java_UploadDataStreamHandler_onCheckIfReadCallbackInvoked( |
| + env, jupload_data_stream_handler_.obj(), res); |
| + } |
| + |
| + private: |
| + // Initialized on construction, effectively constant. |
| + base::android::ScopedJavaGlobalRef<jobject> jupload_data_stream_handler_; |
| + |
| + DISALLOW_COPY_AND_ASSIGN(UploadDataStreamHandlerDelegate); |
| +}; |
| + |
| +UploadDataStreamHandler::UploadDataStreamHandler( |
| + CronetUploadDataStreamAdapter* adapter, |
| + JNIEnv* env, |
| + jobject jupload_data_stream_handler) |
| + : init_complete_(false), |
| + read_complete_(false), |
| + bytes_read_(-1), |
| + adapter_(adapter), |
| + weak_factory_(this) { |
| + network_thread_ = new base::Thread("network"); |
|
mmenke
2015/01/21 18:48:58
Problem: Once we start mucking with weak referenc
xunjieli
2015/01/21 23:49:23
Got it. Thanks for the detailed explanation!
|
| + base::Thread::Options options; |
| + options.message_loop_type = base::MessageLoop::TYPE_IO; |
| + network_thread_->StartWithOptions(options); |
| + delegate_.reset( |
| + new UploadDataStreamHandlerDelegate(env, jupload_data_stream_handler)); |
| +} |
| + |
| +UploadDataStreamHandler::~UploadDataStreamHandler() { |
| +} |
| + |
| +// Called by the data provider. |
| +// On data provider's executor. |
|
mmenke
2015/01/21 18:48:58
Method level comments should go above the method d
xunjieli
2015/01/21 23:49:23
Done. Right, haven't cleaned this up. For my own r
|
| +void UploadDataStreamHandler::OnInitComplete(int res) { |
| + init_complete_ = true; |
| + delegate_->OnInitCompleted(res); |
| +} |
| + |
| +// Called by the data provider. |
| +// On data provider's executor. |
| +void UploadDataStreamHandler::OnReadComplete(int res) { |
| + read_complete_ = true; |
| + bytes_read_ = res; |
| + delegate_->OnReadCompleted(res); |
| +} |
| + |
| +void UploadDataStreamHandler::Init() { |
| + init_complete_ = false; |
|
mmenke
2015/01/21 18:48:58
This shouldn't be touched on the IO thread. May m
xunjieli
2015/01/21 23:49:23
Done. Agreed!
|
| + GetTaskRunner()->PostTask( |
| + FROM_HERE, base::Bind(&UploadDataStreamHandler::InitOnNetworkThread, |
| + weak_factory_.GetWeakPtr())); |
| +} |
| + |
| +void UploadDataStreamHandler::InitOnNetworkThread() { |
| + DCHECK(GetTaskRunner()->BelongsToCurrentThread()); |
| + int res = adapter_->Init(base::Bind(&UploadDataStreamHandler::OnInitComplete, |
| + weak_factory_.GetWeakPtr())); |
| + delegate_->OnInitCalled(res); |
| + if (res == net::OK) { |
| + delegate_->OnInitCompleted(res); |
| + } |
| +} |
| + |
| +void UploadDataStreamHandler::Read() { |
| + read_complete_ = false; |
|
mmenke
2015/01/21 18:48:58
This shouldn't be touched on the IO thread
xunjieli
2015/01/21 23:49:24
Done.
|
| + GetTaskRunner()->PostTask( |
| + FROM_HERE, base::Bind(&UploadDataStreamHandler::ReadOnNetworkThread, |
| + weak_factory_.GetWeakPtr())); |
|
mmenke
2015/01/21 18:48:58
The GetWeakPtr calls aren't safe. According to th
xunjieli
2015/01/21 23:49:23
Ahh I see. That's how it is used. Thanks!
|
| +} |
| + |
| +void UploadDataStreamHandler::ReadOnNetworkThread() { |
| + DCHECK(GetTaskRunner()->BelongsToCurrentThread()); |
| + if (!read_buffer_.get()) |
| + read_buffer_ = new net::IOBufferWithSize(kReadBufferSize); |
| + |
| + int res = adapter_->Read(read_buffer_.get(), kReadBufferSize, |
| + base::Bind(&UploadDataStreamHandler::OnReadComplete, |
| + weak_factory_.GetWeakPtr())); |
| + if (res == net::OK) { |
| + delegate_->OnReadCompleted(res); |
| + } |
| +} |
| + |
| +void UploadDataStreamHandler::Reset() { |
| + GetTaskRunner()->PostTask( |
| + FROM_HERE, base::Bind(&UploadDataStreamHandler::ResetOnNetworkThread, |
| + weak_factory_.GetWeakPtr())); |
| +} |
| + |
| +void UploadDataStreamHandler::ResetOnNetworkThread() { |
| + DCHECK(GetTaskRunner()->BelongsToCurrentThread()); |
| + read_buffer_ = NULL; |
| + bytes_read_ = 0; |
| + adapter_->Reset(); |
| + delegate_->OnResetCompleted(); |
| +} |
| + |
| +void UploadDataStreamHandler::CheckIfInitCallbackInvoked() { |
| + GetTaskRunner()->PostTask( |
| + FROM_HERE, base::Bind(&UploadDataStreamHandler::CheckIfInitCallbackInvokedOnNetworkThread, |
| + weak_factory_.GetWeakPtr())); |
| +} |
| + |
| +void UploadDataStreamHandler::CheckIfInitCallbackInvokedOnNetworkThread() { |
|
mmenke
2015/01/21 18:48:58
DCHECK(GetTaskRunner()->BelongsToCurrentThread());
xunjieli
2015/01/21 23:49:24
Done.
|
| + delegate_->OnCheckIfInitCallbackInvoked(init_complete_); |
| +} |
| + |
| +void UploadDataStreamHandler::CheckIfReadCallbackInvoked() { |
| + GetTaskRunner()->PostTask( |
| + FROM_HERE, base::Bind(&UploadDataStreamHandler::CheckIfReadCallbackInvokedOnNetworkThread, |
| + weak_factory_.GetWeakPtr())); |
| +} |
| + |
| +void UploadDataStreamHandler::CheckIfReadCallbackInvokedOnNetworkThread() { |
|
mmenke
2015/01/21 18:48:58
DCHECK(GetTaskRunner()->BelongsToCurrentThread());
xunjieli
2015/01/21 23:49:23
Done.
|
| + delegate_->OnCheckIfReadCallbackInvoked(read_complete_); |
| +} |
| + |
| +std::string UploadDataStreamHandler::Data() const { |
|
mmenke
2015/01/21 18:48:58
DCHECK(GetTaskRunner()->BelongsToCurrentThread());
xunjieli
2015/01/21 23:49:23
Done.
|
| + if (read_buffer_.get() && bytes_read_ > 0) |
| + return std::string(read_buffer_->data(), bytes_read_); |
| + return ""; |
| +} |
| + |
| +scoped_refptr<base::SingleThreadTaskRunner> |
| +UploadDataStreamHandler::GetTaskRunner() const { |
| + return network_thread_->task_runner(); |
| +} |
| + |
| +static void Init(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + handler->Init(); |
| +} |
| + |
| +static void Read(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + handler->Read(); |
| +} |
| + |
| +static void Reset(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + handler->Reset(); |
| +} |
| + |
| +static jstring GetData(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + return base::android::ConvertUTF8ToJavaString(env, handler->Data()).Release(); |
|
mmenke
2015/01/21 18:48:58
This doesn't seem threadsafe. Should probably p
xunjieli
2015/01/21 23:49:24
Done. You are right! this is not thread-safe.
|
| +} |
| + |
| +static void CheckIfInitCallbackInvoked(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + handler->CheckIfInitCallbackInvoked(); |
| +} |
| + |
| +static void CheckIfReadCallbackInvoked(JNIEnv* env, |
| + jclass jcaller, |
| + jlong jupload_data_stream_handler) { |
| + UploadDataStreamHandler* handler = |
| + reinterpret_cast<UploadDataStreamHandler*>(jupload_data_stream_handler); |
| + DCHECK(handler); |
| + handler->CheckIfReadCallbackInvoked(); |
| +} |
| + |
| +static jlong CreateUploadDataStreamHandler(JNIEnv* env, |
| + jobject jupload_data_stream_handler, |
| + jlong jupload_data_stream_adapter) { |
| + CronetUploadDataStreamAdapter* adapter = |
| + reinterpret_cast<CronetUploadDataStreamAdapter*>( |
| + jupload_data_stream_adapter); |
| + UploadDataStreamHandler* handler = |
| + new UploadDataStreamHandler(adapter, env, jupload_data_stream_handler); |
| + return reinterpret_cast<jlong>(handler); |
| +} |
| +} // namespace cronet |