Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(8)

Unified Diff: services/dart/content_handler_app.cc

Issue 2004403003: Build library for the Dart Content Handler. (Closed) Base URL: https://github.com/domokit/mojo.git@master
Patch Set: Created 4 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « services/dart/content_handler_app.h ('k') | services/dart/content_handler_main.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: services/dart/content_handler_app.cc
diff --git a/services/dart/content_handler_app.cc b/services/dart/content_handler_app.cc
new file mode 100644
index 0000000000000000000000000000000000000000..5db8dac401656daa714e191a97b698aa944b5598
--- /dev/null
+++ b/services/dart/content_handler_app.cc
@@ -0,0 +1,260 @@
+// 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 "services/dart/content_handler_app.h"
+
+#include "base/message_loop/message_loop.h"
+#include "base/run_loop.h"
+#include "base/strings/string_split.h"
+#include "base/synchronization/waitable_event.h"
+#include "base/threading/platform_thread.h"
+#include "base/trace_event/trace_event.h"
+#include "mojo/dart/embedder/dart_controller.h"
+#include "mojo/public/cpp/application/application_impl.h"
+#include "mojo/public/cpp/application/connect.h"
+#include "mojo/public/cpp/bindings/binding.h"
+#include "mojo/public/cpp/bindings/interface_request.h"
+#include "mojo/services/tracing/interfaces/tracing.mojom.h"
+#include "services/dart/dart_app.h"
+#include "url/gurl.h"
+
+namespace dart {
+
+// Flags for the content handler:
+const char kDartTimeline[] = "--dart-timeline";
+const char kDisableObservatory[] = "--disable-observatory";
+const char kEnableStrictMode[] = "--enable-strict-mode";
+const char kRunOnMessageLoop[] = "--run-on-message-loop";
+const char kTraceStartup[] = "--trace-startup";
+// Flags forwarded to the Dart VM:
+const char kCompleteTimeline[] = "--complete-timeline";
+const char kPauseIsolatesOnStart[] = "--pause-isolates-on-start";
+const char kPauseIsolatesOnExit[] = "--pause-isolates-on-exit";
+
+static bool IsDartZip(std::string url) {
+ // If the url doesn't end with ".dart" we assume it is a zipped up
+ // dart application.
+ return !base::EndsWith(url, ".dart", base::CompareCase::INSENSITIVE_ASCII);
+}
+
+// Returns true if |requestedUrl| has a boolean query parameter named |param|.
+static bool HasBoolQueryParam(const std::string& requestedUrl,
+ const std::string& param) {
+ std::string param_true = param + "=true";
+ std::string param_false = param + "=false";
+
+ GURL url(requestedUrl);
+ if (url.has_query()) {
+ std::vector<std::string> query_parameters = base::SplitString(
+ url.query(), "&", base::KEEP_WHITESPACE, base::SPLIT_WANT_NONEMPTY);
+ bool has_true = std::find(query_parameters.begin(), query_parameters.end(),
+ param_true) != query_parameters.end();
+ bool has_false = std::find(query_parameters.begin(), query_parameters.end(),
+ param_false) != query_parameters.end();
+ return has_true || has_false;
+ }
+ return false;
+}
+
+// Returns the value of the boolean query parameter named |param|, or
+// |default_value| if it |param| is not present.
+static bool BoolQueryParamValue(const std::string& requestedUrl,
+ const std::string& param,
+ bool default_value) {
+ if (!HasBoolQueryParam(requestedUrl, param)) {
+ return default_value;
+ }
+ std::string param_true = param + "=true";
+ std::string param_false = param + "=false";
+ GURL url(requestedUrl);
+ DCHECK(url.has_query());
+ std::vector<std::string> query_parameters = base::SplitString(
+ url.query(), "&", base::KEEP_WHITESPACE, base::SPLIT_WANT_NONEMPTY);
+ bool has_true = std::find(query_parameters.begin(), query_parameters.end(),
+ param_true) != query_parameters.end();
+ bool has_false = std::find(query_parameters.begin(), query_parameters.end(),
+ param_false) != query_parameters.end();
+ if (has_true) {
+ return has_true;
+ }
+ if (has_false) {
+ return false;
+ }
+ return default_value;
+}
+
+static bool HasStrictQueryParam(const std::string& requestedUrl) {
+ return BoolQueryParamValue(requestedUrl, "strict", false);
+}
+
+DartContentHandler::DartContentHandler(DartContentHandlerApp* app, bool strict)
+ : app_(app), strict_(strict) {}
+
+void DartContentHandler::set_handler_task_runner(
+ scoped_refptr<base::SingleThreadTaskRunner> handler_task_runner) {
+ handler_task_runner_ = handler_task_runner;
+}
+
+DartContentHandlerApp::DartContentHandlerApp()
+ : content_handler_(this, false),
+ strict_content_handler_(this, true),
+ service_connector_(nullptr),
+ default_strict_(false),
+ run_on_message_loop_(false) {}
+
+DartContentHandlerApp::~DartContentHandlerApp() {
+ // Shutdown the controller.
+ mojo::dart::DartController::Shutdown();
+ delete service_connector_;
+}
+
+void DartContentHandlerApp::ExtractApplication(base::FilePath* application_dir,
+ mojo::URLResponsePtr response,
+ const base::Closure& callback) {
+ url_response_disk_cache_->UpdateAndGetExtracted(
+ response.Pass(),
+ [application_dir, callback](mojo::Array<uint8_t> application_dir_path,
+ mojo::Array<uint8_t> cache_path) {
+ if (application_dir_path.is_null()) {
+ *application_dir = base::FilePath();
+ } else {
+ *application_dir = base::FilePath(std::string(
+ reinterpret_cast<char*>(&application_dir_path.front()),
+ application_dir_path.size()));
+ }
+ callback.Run();
+ });
+}
+
+bool DartContentHandlerApp::run_on_message_loop() const {
+ return run_on_message_loop_;
+}
+
+void DartContentHandlerApp::Initialize(mojo::ApplicationImpl* app) {
+ // Tracing of content handler and controller.
+ tracing_.Initialize(app);
+ // Tracing of isolates and VM.
+ dart_tracing_.Initialize(app);
+
+ // TODO(qsr): This has no effect for now, as the tracing infrastructure
+ // doesn't allow to trace anything before the tracing app connects to the
+ // application.
+ TRACE_EVENT0("dart_content_handler", "DartContentHandler::Initialize");
+
+ default_strict_ = app->HasArg(kEnableStrictMode);
+ content_handler_.set_handler_task_runner(
+ base::MessageLoop::current()->task_runner());
+ strict_content_handler_.set_handler_task_runner(
+ base::MessageLoop::current()->task_runner());
+ mojo::ConnectToService(app->shell(), "mojo:url_response_disk_cache",
+ GetProxy(&url_response_disk_cache_));
+ service_connector_ = new ContentHandlerAppServiceConnector(app);
+
+ if (app->HasArg(kRunOnMessageLoop)) {
+ run_on_message_loop_ = true;
+ }
+
+ bool enable_observatory = true;
+ if (app->HasArg(kDisableObservatory)) {
+ enable_observatory = false;
+ }
+
+ bool enable_dart_timeline = false;
+ if (app->HasArg(kDartTimeline)) {
+ enable_dart_timeline = true;
+ }
+
+ std::vector<const char*> extra_args;
+
+ if (app->HasArg(kPauseIsolatesOnStart)) {
+ extra_args.push_back(kPauseIsolatesOnStart);
+ }
+
+ if (app->HasArg(kPauseIsolatesOnExit)) {
+ extra_args.push_back(kPauseIsolatesOnExit);
+ }
+
+ if (app->HasArg(kCompleteTimeline)) {
+ extra_args.push_back(kCompleteTimeline);
+ }
+
+ bool success = mojo::dart::DartController::Initialize(
+ service_connector_, default_strict_, enable_observatory,
+ enable_dart_timeline, extra_args.data(), extra_args.size());
+
+ if (app->HasArg(kTraceStartup)) {
+ DartTimelineController::EnableAll();
+ }
+ if (!success) {
+ LOG(ERROR) << "Dart VM Initialization failed";
+ }
+}
+
+bool DartContentHandlerApp::ConfigureIncomingConnection(
+ mojo::ServiceProviderImpl* service_provider_impl) {
+ bool strict = HasStrictQueryParam(
+ service_provider_impl->connection_context().connection_url);
+ if (default_strict_ || strict) {
+ service_provider_impl->AddService<mojo::ContentHandler>(
+ mojo::ContentHandlerFactory::GetInterfaceRequestHandler(
+ &strict_content_handler_));
+ } else {
+ service_provider_impl->AddService<mojo::ContentHandler>(
+ mojo::ContentHandlerFactory::GetInterfaceRequestHandler(
+ &content_handler_));
+ }
+ return true;
+}
+
+scoped_ptr<mojo::ContentHandlerFactory::HandledApplicationHolder>
+DartContentHandler::CreateApplication(
+ mojo::InterfaceRequest<mojo::Application> application_request,
+ mojo::URLResponsePtr response) {
+ base::trace_event::TraceLog::GetInstance()
+ ->SetCurrentThreadBlocksMessageLoop();
+
+ TRACE_EVENT1("dart_content_handler", "DartContentHandler::CreateApplication",
+ "url", response->url.get());
+
+ const bool run_on_message_loop = app_->run_on_message_loop();
+ base::FilePath application_dir;
+ std::string url = response->url.get();
+ const char* kPauseIsolatesOnStart = "pauseIsolatesOnStart";
+ const char* kPauseIsolatesOnExit = "pauseIsolatesOnExit";
+ const bool override_pause_isolates_flags =
+ HasBoolQueryParam(url, kPauseIsolatesOnStart) ||
+ HasBoolQueryParam(url, kPauseIsolatesOnExit);
+ const bool pause_isolates_on_start = BoolQueryParamValue(
+ url, kPauseIsolatesOnStart,
+ mojo::dart::DartControllerConfig::kDefaultPauseOnStart);
+ const bool pause_isolates_on_exit = BoolQueryParamValue(
+ url, kPauseIsolatesOnExit,
+ mojo::dart::DartControllerConfig::kDefaultPauseOnExit);
+ if (IsDartZip(response->url.get())) {
+ // Loading a zipped snapshot:
+ // 1) Extract the zip file.
+ // 2) Launch from temporary directory (|application_dir|).
+ handler_task_runner_->PostTask(
+ FROM_HERE,
+ base::Bind(
+ &DartContentHandlerApp::ExtractApplication, base::Unretained(app_),
+ base::Unretained(&application_dir), base::Passed(response.Pass()),
+ base::Bind(
+ base::IgnoreResult(&base::SingleThreadTaskRunner::PostTask),
+ base::MessageLoop::current()->task_runner(), FROM_HERE,
+ base::MessageLoop::QuitWhenIdleClosure())));
+ base::RunLoop().Run();
+ return make_scoped_ptr(
+ new DartApp(application_request.Pass(), url, application_dir, strict_,
+ run_on_message_loop, override_pause_isolates_flags,
+ pause_isolates_on_start, pause_isolates_on_exit));
+ } else {
+ // Loading a raw .dart file pointed at by |url|.
+ return make_scoped_ptr(
+ new DartApp(application_request.Pass(), url, strict_,
+ run_on_message_loop, override_pause_isolates_flags,
+ pause_isolates_on_start, pause_isolates_on_exit));
+ }
+}
+
+} // namespace dart
« no previous file with comments | « services/dart/content_handler_app.h ('k') | services/dart/content_handler_main.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698