// Copyright (c) 2012 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. // URL request job for reading from resources and assets. #include "chrome/browser/android/android_protocol_adapter.h" #include "base/android/jni_android.h" #include "base/android/jni_helper.h" #include "base/android/jni_string.h" #include "base/string_util.h" #include "chrome/browser/android/android_stream_reader_url_request_job.h" #include "chrome/common/url_constants.h" #include "googleurl/src/gurl.h" #include "jni/AndroidProtocolAdapter_jni.h" #include "net/base/io_buffer.h" #include "net/base/mime_util.h" #include "net/base/net_errors.h" #include "net/base/net_util.h" #include "net/http/http_util.h" #include "net/url_request/url_request_context.h" #include "net/url_request/url_request_context_getter.h" #include "net/url_request/url_request_job_factory.h" #include "net/url_request/url_request_job_manager.h" using base::android::AttachCurrentThread; using base::android::ClearException; using base::android::ConvertUTF8ToJavaString; using base::android::ScopedJavaGlobalRef; using base::android::ScopedJavaLocalRef; namespace { // Override resource context for reading resource and asset files. Used for // testing. JavaObjectWeakGlobalRef* g_resource_context = NULL; void ResetResourceContext(JavaObjectWeakGlobalRef* ref) { if (g_resource_context) delete g_resource_context; g_resource_context = ref; } class AndroidStreamReaderURLRequestJobDelegateImpl : public AndroidStreamReaderURLRequestJob::Delegate { public: AndroidStreamReaderURLRequestJobDelegateImpl(); virtual ScopedJavaLocalRef OpenInputStream( JNIEnv* env, net::URLRequest* request) OVERRIDE; virtual bool GetMimeType(JNIEnv* env, net::URLRequest* request, jobject stream, std::string* mime_type) OVERRIDE; virtual bool GetCharset(JNIEnv* env, net::URLRequest* request, jobject stream, std::string* charset) OVERRIDE; virtual ~AndroidStreamReaderURLRequestJobDelegateImpl(); }; class AssetFileProtocolInterceptor : public net::URLRequestJobFactory::Interceptor { public: AssetFileProtocolInterceptor(); virtual ~AssetFileProtocolInterceptor() OVERRIDE; virtual net::URLRequestJob* MaybeIntercept( net::URLRequest* request, net::NetworkDelegate* network_delegate) const OVERRIDE; virtual net::URLRequestJob* MaybeInterceptRedirect( const GURL& location, net::URLRequest* request, net::NetworkDelegate* network_delegate) const OVERRIDE; virtual net::URLRequestJob* MaybeInterceptResponse( net::URLRequest* request, net::NetworkDelegate* network_delegate) const OVERRIDE; private: // file:///android_asset/ const std::string asset_prefix_; // file:///android_res/ const std::string resource_prefix_; }; } // namespace static bool InitJNIBindings(JNIEnv* env) { return RegisterNativesImpl(env) && AndroidStreamReaderURLRequestJob::InitJNIBindings(env); } // static net::URLRequestJob* AndroidProtocolAdapter::Factory( net::URLRequest* request, net::NetworkDelegate* network_delegate, const std::string& scheme) { DCHECK(scheme == chrome::kContentScheme); return new AndroidStreamReaderURLRequestJob( request, network_delegate, scoped_ptr( new AndroidStreamReaderURLRequestJobDelegateImpl())); } static void AddFileSchemeInterceptorOnIOThread( net::URLRequestContextGetter* context_getter) { // The job factory takes ownership of the interceptor. const_cast( context_getter->GetURLRequestContext()->job_factory())->AddInterceptor( new AssetFileProtocolInterceptor()); } // static void AndroidProtocolAdapter::RegisterProtocols( JNIEnv* env, net::URLRequestContextGetter* context_getter) { DCHECK(env); if (!InitJNIBindings(env)) { // Must not fail. NOTREACHED(); } // Register content://. Note that even though a scheme is // registered here, it cannot be used by child processes until access to it is // granted via ChildProcessSecurityPolicy::GrantScheme(). This is done in // RenderViewHost. // // TODO(mnaganov): Convert into a ProtocolHandler. net::URLRequestJobManager* job_manager = net::URLRequestJobManager::GetInstance(); job_manager->RegisterProtocolFactory(chrome::kContentScheme, &AndroidProtocolAdapter::Factory); // Register a file: scheme interceptor for application assets. context_getter->GetNetworkTaskRunner()->PostTask( FROM_HERE, base::Bind(&AddFileSchemeInterceptorOnIOThread, make_scoped_refptr(context_getter))); // TODO(mnaganov): Add an interceptor for the incognito profile? } // Set a context object to be used for resolving resource queries. This can // be used to override the default application context and redirect all // resource queries to a specific context object, e.g., for the purposes of // testing. // // |context| should be a android.content.Context instance or NULL to enable // the use of the standard application context. static void SetResourceContextForTesting(JNIEnv* env, jclass /*clazz*/, jobject context) { if (context) { ResetResourceContext(new JavaObjectWeakGlobalRef(env, context)); } else { ResetResourceContext(NULL); } } static jstring GetAndroidAssetPath(JNIEnv* env, jclass /*clazz*/) { // OK to release, JNI binding. return ConvertUTF8ToJavaString(env, chrome::kAndroidAssetPath).Release(); } static jstring GetAndroidResourcePath(JNIEnv* env, jclass /*clazz*/) { // OK to release, JNI binding. return ConvertUTF8ToJavaString(env, chrome::kAndroidResourcePath).Release(); } static ScopedJavaLocalRef GetResourceContext(JNIEnv* env) { if (g_resource_context) return g_resource_context->get(env); ScopedJavaLocalRef context; // We have to reset as GetApplicationContext() returns a jobject with a // global ref. The constructor that takes a jobject would expect a local ref // and would assert. context.Reset(env, base::android::GetApplicationContext()); return context; } AndroidStreamReaderURLRequestJobDelegateImpl:: AndroidStreamReaderURLRequestJobDelegateImpl() { } AndroidStreamReaderURLRequestJobDelegateImpl:: ~AndroidStreamReaderURLRequestJobDelegateImpl() { } ScopedJavaLocalRef AndroidStreamReaderURLRequestJobDelegateImpl::OpenInputStream( JNIEnv* env, net::URLRequest* request) { DCHECK(request); DCHECK(env); // Open the input stream. ScopedJavaLocalRef url = ConvertUTF8ToJavaString(env, request->url().spec()); ScopedJavaLocalRef stream = Java_AndroidProtocolAdapter_open( env, GetResourceContext(env).obj(), url.obj()); // Check and clear pending exceptions. if (ClearException(env) || stream.is_null()) { DLOG(ERROR) << "Unable to open input stream for Android URL"; return ScopedJavaLocalRef(env, NULL); } return stream; } bool AndroidStreamReaderURLRequestJobDelegateImpl::GetMimeType( JNIEnv* env, net::URLRequest* request, jobject stream, std::string* mime_type) { DCHECK(env); DCHECK(request); DCHECK(mime_type); if (!stream) return false; // Query the mime type from the Java side. It is possible for the query to // fail, as the mime type cannot be determined for all supported schemes. ScopedJavaLocalRef url = ConvertUTF8ToJavaString(env, request->url().spec()); ScopedJavaLocalRef returned_type = Java_AndroidProtocolAdapter_getMimeType(env, GetResourceContext(env).obj(), stream, url.obj()); if (ClearException(env) || returned_type.is_null()) return false; *mime_type = base::android::ConvertJavaStringToUTF8(returned_type); return true; } bool AndroidStreamReaderURLRequestJobDelegateImpl::GetCharset( JNIEnv* env, net::URLRequest* request, jobject stream, std::string* charset) { // TODO: We should probably be getting this from the managed side. return false; } AssetFileProtocolInterceptor::AssetFileProtocolInterceptor() : asset_prefix_(std::string(chrome::kFileScheme) + std::string(content::kStandardSchemeSeparator) + chrome::kAndroidAssetPath), resource_prefix_(std::string(chrome::kFileScheme) + std::string(content::kStandardSchemeSeparator) + chrome::kAndroidResourcePath) { } AssetFileProtocolInterceptor::~AssetFileProtocolInterceptor() { } net::URLRequestJob* AssetFileProtocolInterceptor::MaybeIntercept( net::URLRequest* request, net::NetworkDelegate* network_delegate) const { if (!request->url().SchemeIsFile()) return NULL; const std::string& url = request->url().spec(); if (!StartsWithASCII(url, asset_prefix_, /*case_sensitive=*/ true) && !StartsWithASCII(url, resource_prefix_, /*case_sensitive=*/ true)) { return NULL; } return new AndroidStreamReaderURLRequestJob( request, network_delegate, scoped_ptr( new AndroidStreamReaderURLRequestJobDelegateImpl())); } net::URLRequestJob* AssetFileProtocolInterceptor::MaybeInterceptRedirect( const GURL& location, net::URLRequest* request, net::NetworkDelegate* network_delegate) const { return NULL; } net::URLRequestJob* AssetFileProtocolInterceptor::MaybeInterceptResponse( net::URLRequest* request, net::NetworkDelegate* network_delegate) const { return NULL; }