// Copyright (c) 2011 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. // This implements a browser-side endpoint for UI automation activity. // The client-side endpoint is implemented by AutomationProxy. // The entire lifetime of this object should be contained within that of // the BrowserProcess, and in particular the NotificationService that's // hung off of it. #ifndef CHROME_BROWSER_AUTOMATION_AUTOMATION_PROVIDER_H_ #define CHROME_BROWSER_AUTOMATION_AUTOMATION_PROVIDER_H_ #pragma once #include <map> #include <string> #include <vector> #include "base/basictypes.h" #include "base/compiler_specific.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/observer_list.h" #include "base/string16.h" #include "chrome/browser/autofill/field_types.h" #include "chrome/common/automation_constants.h" #include "chrome/common/content_settings.h" #include "content/browser/browser_thread.h" #include "content/browser/cancelable_request.h" #include "content/browser/tab_contents/navigation_entry.h" #include "content/common/notification_observer.h" #include "ipc/ipc_channel.h" #if defined(OS_WIN) #include "ui/gfx/native_widget_types.h" #include "views/events/event.h" #endif // defined(OS_WIN) class PopupMenuWaiter; class TabContents; struct AutomationMsg_Find_Params; struct Reposition_Params; struct ExternalTabSettings; namespace IPC { class ChannelProxy; } class AutofillProfile; class AutomationAutocompleteEditTracker; class AutomationBrowserTracker; class AutomationExtensionTracker; class AutomationResourceMessageFilter; class AutomationTabTracker; class AutomationWindowTracker; class Browser; class CreditCard; class DictionaryValue; class DownloadItem; class Extension; class ExtensionPortContainer; class ExtensionTestResultNotificationObserver; class ExternalTabContainer; class FilePath; class InitialLoadObserver; class ListValue; class LoginHandler; class MetricEventDurationObserver; class NavigationController; class NavigationControllerRestoredObserver; class Profile; class RenderViewHost; class TabContents; struct AutocompleteMatchData; namespace gfx { class Point; } class AutomationProvider : public IPC::Channel::Listener, public IPC::Message::Sender, public base::SupportsWeakPtr<AutomationProvider>, public base::RefCountedThreadSafe<AutomationProvider, BrowserThread::DeleteOnUIThread> { public: explicit AutomationProvider(Profile* profile); Profile* profile() const { return profile_; } // Initializes a channel for a connection to an AutomationProxy. // If channel_id starts with kNamedInterfacePrefix, it will act // as a server, create a named IPC socket with channel_id as its // path, and will listen on the socket for incoming connections. // If channel_id does not, it will act as a client and establish // a connection on its primary IPC channel. See ipc/ipc_channel_posix.cc // for more information about kPrimaryIPCChannel. bool InitializeChannel(const std::string& channel_id) WARN_UNUSED_RESULT; // Sets the number of tabs that we expect; when this number of tabs has // loaded, an AutomationMsg_InitialLoadsComplete message is sent. void SetExpectedTabCount(size_t expected_tabs); // Called when the inital set of tabs has finished loading. // Call SetExpectedTabCount(0) to set this to true immediately. void OnInitialTabLoadsComplete(); // Called when the ChromeOS network library has finished its first update. void OnNetworkLibraryInit(); // Get the index of a particular NavigationController object // in the given parent window. This method uses // TabStrip::GetIndexForNavigationController to get the index. int GetIndexForNavigationController(const NavigationController* controller, const Browser* parent) const; // Add or remove a non-owning reference to a tab's LoginHandler. This is for // when a login prompt is shown for HTTP/FTP authentication. // TODO(mpcomplete): The login handling is a fairly special purpose feature. // Eventually we'll probably want ways to interact with the ChromeView of the // login window in a generic manner, such that it can be used for anything, // not just logins. void AddLoginHandler(NavigationController* tab, LoginHandler* handler); void RemoveLoginHandler(NavigationController* tab); // IPC implementations virtual bool Send(IPC::Message* msg); virtual void OnChannelConnected(int pid); virtual bool OnMessageReceived(const IPC::Message& msg); virtual void OnChannelError(); IPC::Message* reply_message_release() { IPC::Message* reply_message = reply_message_; reply_message_ = NULL; return reply_message; } // Adds the extension passed in to the extension tracker, and returns // the associated handle. If the tracker already contains the extension, // the handle is simply returned. int AddExtension(const Extension* extension); #if defined(OS_WIN) // Adds the external tab passed in to the tab tracker. bool AddExternalTab(ExternalTabContainer* external_tab); #endif // Get the DictionaryValue equivalent for a download item. Caller owns the // DictionaryValue. DictionaryValue* GetDictionaryFromDownloadItem(const DownloadItem* download); protected: friend struct BrowserThread::DeleteOnThread<BrowserThread::UI>; friend class DeleteTask<AutomationProvider>; virtual ~AutomationProvider(); // Helper function to find the browser window that contains a given // NavigationController and activate that tab. // Returns the Browser if found. Browser* FindAndActivateTab(NavigationController* contents); // Convert a tab handle into a TabContents. If |tab| is non-NULL a pointer // to the tab is also returned. Returns NULL in case of failure or if the tab // is not of the TabContents type. TabContents* GetTabContentsForHandle(int handle, NavigationController** tab); // Returns the protocol version which typically is the module version. virtual std::string GetProtocolVersion(); // Returns the associated view for the tab handle passed in. // Returns NULL on failure. RenderViewHost* GetViewForTab(int tab_handle); // Called on IPC message deserialization failure. Prints an error message // and closes the IPC channel. void OnMessageDeserializationFailure(); scoped_ptr<AutomationAutocompleteEditTracker> autocomplete_edit_tracker_; scoped_ptr<AutomationBrowserTracker> browser_tracker_; scoped_ptr<InitialLoadObserver> initial_load_observer_; scoped_ptr<MetricEventDurationObserver> metric_event_duration_observer_; scoped_ptr<NavigationControllerRestoredObserver> restore_tracker_; scoped_ptr<AutomationTabTracker> tab_tracker_; scoped_ptr<AutomationWindowTracker> window_tracker_; typedef std::map<NavigationController*, LoginHandler*> LoginHandlerMap; LoginHandlerMap login_handler_map_; Profile* profile_; // A pointer to reply message used when we do asynchronous processing in the // message handler. // TODO(phajdan.jr): Remove |reply_message_|, it is error-prone. IPC::Message* reply_message_; // Consumer for asynchronous history queries. CancelableRequestConsumer consumer_; // Sends a find request for a given query. void SendFindRequest( TabContents* tab_contents, bool with_json, const string16& search_string, bool forward, bool match_case, bool find_next, IPC::Message* reply_message); scoped_refptr<AutomationResourceMessageFilter> automation_resource_message_filter_; // True iff we should open a new automation IPC channel if it closes. bool reinitialize_on_channel_error_; private: void OnUnhandledMessage(); // Clear and reinitialize the automation IPC channel. bool ReinitializeChannel(); // IPC Message callbacks. void WindowSimulateDrag(int handle, const std::vector<gfx::Point>& drag_path, int flags, bool press_escape_en_route, IPC::Message* reply_message); void HandleUnused(const IPC::Message& message, int handle); void SetFilteredInet(const IPC::Message& message, bool enabled); void GetFilteredInetHitCount(int* hit_count); void SetProxyConfig(const std::string& new_proxy_config); // Responds to the FindInPage request, retrieves the search query parameters, // launches an observer to listen for results and issues a StartFind request. void HandleFindRequest(int handle, const AutomationMsg_Find_Params& params, IPC::Message* reply_message); void OnSetPageFontSize(int tab_handle, int font_size); // See browsing_data_remover.h for explanation of bitmap fields. void RemoveBrowsingData(int remove_mask); // Notify the JavaScript engine in the render to change its parameters // while performing stress testing. See // |ViewHostMsg_JavaScriptStressTestControl_Commands| in render_messages.h // for information on the arguments. void JavaScriptStressTestControl(int handle, int cmd, int param); void InstallExtension(const FilePath& crx_path, IPC::Message* reply_message); void WaitForExtensionTestResult(IPC::Message* reply_message); void InstallExtensionAndGetHandle(const FilePath& crx_path, bool with_ui, IPC::Message* reply_message); void UninstallExtension(int extension_handle, bool* success); void ReloadExtension(int extension_handle, IPC::Message* reply_message); void EnableExtension(int extension_handle, IPC::Message* reply_message); void DisableExtension(int extension_handle, bool* success); void ExecuteExtensionActionInActiveTabAsync(int extension_handle, int browser_handle, IPC::Message* reply_message); void MoveExtensionBrowserAction(int extension_handle, int index, bool* success); void GetExtensionProperty(int extension_handle, AutomationMsg_ExtensionProperty type, bool* success, std::string* value); // Asynchronous request for printing the current tab. void PrintAsync(int tab_handle); // Uses the specified encoding to override the encoding of the page in the // specified tab. void OverrideEncoding(int tab_handle, const std::string& encoding_name, bool* success); // Selects all contents on the page. void SelectAll(int tab_handle); // Edit operations on the page. void Cut(int tab_handle); void Copy(int tab_handle); void Paste(int tab_handle); void ReloadAsync(int tab_handle); void StopAsync(int tab_handle); void SaveAsAsync(int tab_handle); // Returns the extension for the given handle. Returns NULL if there is // no extension for the handle. const Extension* GetExtension(int extension_handle); // Returns the extension for the given handle, if the handle is valid and // the associated extension is enabled. Returns NULL otherwise. const Extension* GetEnabledExtension(int extension_handle); // Returns the extension for the given handle, if the handle is valid and // the associated extension is disabled. Returns NULL otherwise. const Extension* GetDisabledExtension(int extension_handle); // Method called by the popup menu tracker when a popup menu is opened. void NotifyPopupMenuOpened(); #if defined(OS_WIN) // The functions in this block are for use with external tabs, so they are // Windows only. // The container of an externally hosted tab calls this to reflect any // accelerator keys that it did not process. This gives the tab a chance // to handle the keys void ProcessUnhandledAccelerator(const IPC::Message& message, int handle, const MSG& msg); void SetInitialFocus(const IPC::Message& message, int handle, bool reverse, bool restore_focus_to_view); void OnTabReposition(int tab_handle, const Reposition_Params& params); void OnForwardContextMenuCommandToChrome(int tab_handle, int command); void CreateExternalTab(const ExternalTabSettings& settings, gfx::NativeWindow* tab_container_window, gfx::NativeWindow* tab_window, int* tab_handle, int* session_id); void ConnectExternalTab(uint64 cookie, bool allow, gfx::NativeWindow parent_window, gfx::NativeWindow* tab_container_window, gfx::NativeWindow* tab_window, int* tab_handle, int* session_id); void NavigateInExternalTab( int handle, const GURL& url, const GURL& referrer, AutomationMsg_NavigationResponseValues* status); void NavigateExternalTabAtIndex( int handle, int index, AutomationMsg_NavigationResponseValues* status); // Handler for a message sent by the automation client. void OnMessageFromExternalHost(int handle, const std::string& message, const std::string& origin, const std::string& target); void OnBrowserMoved(int handle); void OnRunUnloadHandlers(int handle, IPC::Message* reply_message); void OnSetZoomLevel(int handle, int zoom_level); ExternalTabContainer* GetExternalTabForHandle(int handle); #endif // defined(OS_WIN) scoped_ptr<IPC::ChannelProxy> channel_; scoped_ptr<NotificationObserver> new_tab_ui_load_observer_; scoped_ptr<NotificationObserver> find_in_page_observer_; scoped_ptr<ExtensionTestResultNotificationObserver> extension_test_result_observer_; scoped_ptr<AutomationExtensionTracker> extension_tracker_; // True iff connected to an AutomationProxy. bool is_connected_; // True iff browser finished loading initial set of tabs. bool initial_tab_loads_complete_; // True iff the Chrome OS network library finished initialization. bool network_library_initialized_; // ID of automation channel. std::string channel_id_; DISALLOW_COPY_AND_ASSIGN(AutomationProvider); }; #endif // CHROME_BROWSER_AUTOMATION_AUTOMATION_PROVIDER_H_