// Copyright (c) 2015 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #ifndef ATOM_BROWSER_NET_JS_ASKER_H_ #define ATOM_BROWSER_NET_JS_ASKER_H_ #include "base/callback.h" #include "base/memory/weak_ptr.h" #include "base/values.h" #include "content/public/browser/browser_thread.h" #include "net/base/net_errors.h" #include "net/url_request/url_request_job.h" #include "v8/include/v8.h" namespace atom { using JavaScriptHandler = base::Callback)>; namespace internal { using ResponseCallback = base::Callback options)>; // Ask handler for options in UI thread. void AskForOptions(v8::Isolate* isolate, const JavaScriptHandler& handler, net::URLRequest* request, const ResponseCallback& callback); } // namespace internal template class JsAsker : public RequestJob { public: JsAsker(net::URLRequest* request, net::NetworkDelegate* network_delegate, v8::Isolate* isolate, const JavaScriptHandler& handler) : RequestJob(request, network_delegate), isolate_(isolate), handler_(handler), weak_factory_(this) {} // Subclass should do initailze work here. virtual void StartAsync(scoped_ptr options) { RequestJob::Start(); } private: // RequestJob: void Start() override { content::BrowserThread::PostTask( content::BrowserThread::UI, FROM_HERE, base::Bind(&internal::AskForOptions, isolate_, handler_, RequestJob::request(), base::Bind(&JsAsker::OnResponse, weak_factory_.GetWeakPtr()))); } // Called when the JS handler has sent the response, we need to decide whether // to start, or fail the job. void OnResponse(bool success, scoped_ptr options) { if (success && options) { StartAsync(options.Pass()); } else { RequestJob::NotifyStartError( net::URLRequestStatus(net::URLRequestStatus::FAILED, net::ERR_NOT_IMPLEMENTED)); } } v8::Isolate* isolate_; JavaScriptHandler handler_; base::WeakPtrFactory weak_factory_; DISALLOW_COPY_AND_ASSIGN(JsAsker); }; } // namespace atom #endif // ATOM_BROWSER_NET_JS_ASKER_H_