// Copyright (c) 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. #ifndef CONTENT_BROWSER_DEVTOOLS_DEVTOOLS_PROTOCOL_H_ #define CONTENT_BROWSER_DEVTOOLS_DEVTOOLS_PROTOCOL_H_ #include <map> #include <string> #include "base/basictypes.h" #include "base/callback.h" #include "base/compiler_specific.h" #include "base/memory/ref_counted.h" #include "base/values.h" #include "content/common/content_export.h" namespace content { // Utility classes for processing DevTools remote debugging messages. // https://developers.google.com/chrome-developer-tools/docs/debugger-protocol class DevToolsProtocol { public: typedef base::Callback<void(const std::string& message)> Notifier; class Response; class Message : public base::RefCountedThreadSafe<Message> { public: std::string domain() { return domain_; } std::string method() { return method_; } base::DictionaryValue* params() { return params_.get(); } virtual std::string Serialize() = 0; protected: friend class base::RefCountedThreadSafe<Message>; virtual ~Message(); Message(const std::string& method, base::DictionaryValue* params); std::string domain_; std::string method_; scoped_ptr<base::DictionaryValue> params_; private: DISALLOW_COPY_AND_ASSIGN(Message); }; class Command : public Message { public: int id() { return id_; } virtual std::string Serialize() OVERRIDE; // Creates success response. Takes ownership of |result|. scoped_refptr<Response> SuccessResponse(base::DictionaryValue* result); // Creates error response. scoped_refptr<Response> InternalErrorResponse(const std::string& message); // Creates error response. scoped_refptr<Response> InvalidParamResponse(const std::string& param); // Creates error response. scoped_refptr<Response> NoSuchMethodErrorResponse(); // Creates error response. scoped_refptr<Response> ServerErrorResponse(const std::string& message); // Creates async response promise. scoped_refptr<Response> AsyncResponsePromise(); protected: virtual ~Command(); private: friend class DevToolsProtocol; Command(int id, const std::string& method, base::DictionaryValue* params); int id_; DISALLOW_COPY_AND_ASSIGN(Command); }; class Response : public base::RefCountedThreadSafe<Response> { public: std::string Serialize(); bool is_async_promise() { return is_async_promise_; } private: friend class base::RefCountedThreadSafe<Response>; friend class Command; friend class DevToolsProtocol; virtual ~Response(); Response(int id, base::DictionaryValue* result); Response(int id, int error_code, const std::string& error_message); int id_; scoped_ptr<base::DictionaryValue> result_; int error_code_; std::string error_message_; bool is_async_promise_; DISALLOW_COPY_AND_ASSIGN(Response); }; class Notification : public Message { public: virtual std::string Serialize() OVERRIDE; private: friend class DevToolsProtocol; virtual ~Notification(); // Takes ownership of |params|. Notification(const std::string& method, base::DictionaryValue* params); DISALLOW_COPY_AND_ASSIGN(Notification); }; class CONTENT_EXPORT Handler { public: typedef base::Callback<scoped_refptr<DevToolsProtocol::Response>( scoped_refptr<DevToolsProtocol::Command> command)> CommandHandler; virtual ~Handler(); virtual scoped_refptr<DevToolsProtocol::Response> HandleCommand( scoped_refptr<DevToolsProtocol::Command> command); void SetNotifier(const Notifier& notifier); protected: Handler(); void RegisterCommandHandler(const std::string& command, const CommandHandler& handler); // Sends notification to the client. Takes ownership of |params|. void SendNotification(const std::string& method, base::DictionaryValue* params); void SendAsyncResponse(scoped_refptr<DevToolsProtocol::Response> response); // Sends message to client, the caller is presumed to properly // format the message. void SendRawMessage(const std::string& message); private: typedef std::map<std::string, CommandHandler> CommandHandlers; Notifier notifier_; CommandHandlers command_handlers_; DISALLOW_COPY_AND_ASSIGN(Handler); }; CONTENT_EXPORT static base::DictionaryValue* ParseMessage( const std::string& json, std::string* error_response); CONTENT_EXPORT static scoped_refptr<Command> ParseCommand( const std::string& json, std::string* error_response); CONTENT_EXPORT static scoped_refptr<Command> ParseCommand( base::DictionaryValue* command_dict, std::string* error_response); CONTENT_EXPORT static scoped_refptr<Command> CreateCommand( int id, const std::string& method, base::DictionaryValue* params); CONTENT_EXPORT static scoped_refptr<Response> ParseResponse( base::DictionaryValue* response_dict); static scoped_refptr<Notification> ParseNotification( const std::string& json); static scoped_refptr<Notification> CreateNotification( const std::string& method, base::DictionaryValue* params); DevToolsProtocol() {} ~DevToolsProtocol() {} }; } // namespace content #endif // CONTENT_BROWSER_DEVTOOLS_DEVTOOLS_PROTOCOL_H_