// Copyright (c) 2006-2008 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 "chrome/browser/google_update.h" #include #include #include "base/message_loop.h" #include "base/path_service.h" #include "base/task.h" #include "base/thread.h" #include "chrome/browser/browser_process.h" #include "chrome/installer/util/helper.h" #include "chrome/installer/util/google_update_constants.h" #include "google_update_idl_i.c" namespace { // Check if the currently running Chrome can be updated by Google Update by // checking if it is running from the standard location. Return true if running // from the standard location, otherwise return false. bool CanUpdateCurrentChrome() { std::wstring current_exe_path; if (PathService::Get(base::DIR_EXE, ¤t_exe_path)) { std::wstring standard_exe_path = installer::GetChromeInstallPath(false); std::transform(current_exe_path.begin(), current_exe_path.end(), current_exe_path.begin(), tolower); std::transform(standard_exe_path.begin(), standard_exe_path.end(), standard_exe_path.begin(), tolower); if (current_exe_path != standard_exe_path) { LOG(ERROR) << L"Google Update cannot update Chrome installed in a " << L"non-standard location: " << current_exe_path.c_str() << L". The standard location is: " << standard_exe_path.c_str() << L"."; return false; } } return true; } } // namespace //////////////////////////////////////////////////////////////////////////////// // // The GoogleUpdateJobObserver COM class is responsible for receiving status // reports from google Update. It keeps track of the progress as Google Update // notifies us and ends the message loop we are spinning in once Google Update // reports that it is done. // //////////////////////////////////////////////////////////////////////////////// class GoogleUpdateJobObserver : public CComObjectRootEx, public IJobObserver { public: BEGIN_COM_MAP(GoogleUpdateJobObserver) COM_INTERFACE_ENTRY(IJobObserver) END_COM_MAP() GoogleUpdateJobObserver() {} virtual ~GoogleUpdateJobObserver() {} // Notifications from Google Update: STDMETHOD(OnShow)() { return S_OK; } STDMETHOD(OnCheckingForUpdate)() { result_ = UPGRADE_CHECK_STARTED; return S_OK; } STDMETHOD(OnUpdateAvailable)(const TCHAR* version_string) { result_ = UPGRADE_IS_AVAILABLE; new_version_ = version_string; return S_OK; } STDMETHOD(OnWaitingToDownload)() { return S_OK; } STDMETHOD(OnDownloading)(int time_remaining_ms, int pos) { return S_OK; } STDMETHOD(OnWaitingToInstall)() { return S_OK; } STDMETHOD(OnInstalling)() { result_ = UPGRADE_STARTED; return S_OK; } STDMETHOD(OnPause)() { return S_OK; } STDMETHOD(OnComplete)(CompletionCodes code, const TCHAR* text) { switch (code) { case COMPLETION_CODE_SUCCESS_CLOSE_UI: case COMPLETION_CODE_SUCCESS: { if (result_ == UPGRADE_STARTED) result_ = UPGRADE_SUCCESSFUL; else if (result_ == UPGRADE_CHECK_STARTED) result_ = UPGRADE_ALREADY_UP_TO_DATE; break; } default: { NOTREACHED(); result_ = UPGRADE_ERROR; break; } } event_sink_ = NULL; // We no longer need to spin the message loop that we started spinning in // InitiateGoogleUpdateCheck. MessageLoop::current()->Quit(); return S_OK; } STDMETHOD(SetEventSink)(IProgressWndEvents* event_sink) { event_sink_ = event_sink; return S_OK; } // Returns the results of the update operation. STDMETHOD(GetResult)(GoogleUpdateUpgradeResult* result) { // Intermediary steps should never be reported to the client. DCHECK(result_ != UPGRADE_STARTED && result_ != UPGRADE_CHECK_STARTED); *result = result_; return S_OK; } // Returns which version Google Update found on the server (if a more // recent version was found). Otherwise, this will be blank. STDMETHOD(GetVersionInfo)(std::wstring* version_string) { *version_string = new_version_; return S_OK; } private: // The status/result of the Google Update operation. GoogleUpdateUpgradeResult result_; // The version string Google Update found. std::wstring new_version_; // Allows us control the upgrade process to a small degree. After OnComplete // has been called, this object can not be used. CComPtr event_sink_; }; // TODO(finnur): (http://b/1292883) The object is currently managing its own // lifetime, but should be converted to use RefCountedThreadSafe instead. template <> struct RunnableMethodTraits { static void RetainCallee(GoogleUpdate* obj) { } static void ReleaseCallee(GoogleUpdate* obj) { } }; //////////////////////////////////////////////////////////////////////////////// // GoogleUpdate, public: GoogleUpdate::GoogleUpdate() : listener_(NULL) { } GoogleUpdate::~GoogleUpdate() { } //////////////////////////////////////////////////////////////////////////////// // GoogleUpdate, ChromeViews::DialogDelegate implementation: void GoogleUpdate::CheckForUpdate(bool install_if_newer) { // We need to shunt this request over to InitiateGoogleUpdateCheck and have // it run in the file thread. MessageLoop* file_loop = g_browser_process->file_thread()->message_loop(); file_loop->PostTask(FROM_HERE, NewRunnableMethod(this, &GoogleUpdate::InitiateGoogleUpdateCheck, install_if_newer, MessageLoop::current())); } // Adds/removes a listener. Only one listener is maintained at the moment. void GoogleUpdate::AddStatusChangeListener( GoogleUpdateStatusListener* listener) { DCHECK(!listener_); listener_ = listener; } void GoogleUpdate::RemoveStatusChangeListener() { listener_ = NULL; } //////////////////////////////////////////////////////////////////////////////// // GoogleUpdate, private: bool GoogleUpdate::InitiateGoogleUpdateCheck(bool install_if_newer, MessageLoop* main_loop) { if (!CanUpdateCurrentChrome()) { main_loop->PostTask(FROM_HERE, NewRunnableMethod(this, &GoogleUpdate::ReportResults, UPGRADE_ERROR, CANNOT_UPGRADE_CHROME_IN_THIS_DIRECTORY)); return false; } CComObject* job_observer; HRESULT hr = CComObject::CreateInstance(&job_observer); if (hr != S_OK) { return ReportFailure(hr, GOOGLE_UPDATE_JOB_SERVER_CREATION_FAILED, main_loop); } CComPtr job_holder(job_observer); CComPtr on_demand; hr = on_demand.CoCreateInstance(CLSID_OnDemandClass); if (hr != S_OK) return ReportFailure(hr, GOOGLE_UPDATE_ONDEMAND_CLASS_NOT_FOUND, main_loop); if (!install_if_newer) hr = on_demand->CheckForUpdate(google_update::kChromeGuid, job_observer); else hr = on_demand->Update(google_update::kChromeGuid, job_observer); if (hr != S_OK) return ReportFailure(hr, GOOGLE_UPDATE_ONDEMAND_CLASS_REPORTED_ERROR, main_loop); // We need to spin the message loop while Google Update is running so that it // can report back to us through GoogleUpdateJobObserver. This message loop // will terminate once Google Update sends us the completion status // (success/error). See OnComplete(). MessageLoop::current()->Run(); GoogleUpdateUpgradeResult results; hr = job_observer->GetResult(&results); if (hr != S_OK) return ReportFailure(hr, GOOGLE_UPDATE_GET_RESULT_CALL_FAILED, main_loop); if (results == UPGRADE_ERROR) return ReportFailure(hr, GOOGLE_UPDATE_ERROR_UPDATING, main_loop); hr = job_observer->GetVersionInfo(&version_available_); if (hr != S_OK) return ReportFailure(hr, GOOGLE_UPDATE_GET_VERSION_INFO_FAILED, main_loop); main_loop->PostTask(FROM_HERE, NewRunnableMethod(this, &GoogleUpdate::ReportResults, results, GOOGLE_UPDATE_NO_ERROR)); job_holder = NULL; on_demand = NULL; return true; } void GoogleUpdate::ReportResults(GoogleUpdateUpgradeResult results, GoogleUpdateErrorCode error_code) { // If we get an error, then error code must not be blank, and vice versa. DCHECK(results == UPGRADE_ERROR ? error_code != GOOGLE_UPDATE_NO_ERROR : error_code == GOOGLE_UPDATE_NO_ERROR); if (listener_) listener_->OnReportResults(results, error_code, version_available_); delete this; } bool GoogleUpdate::ReportFailure(HRESULT hr, GoogleUpdateErrorCode error_code, MessageLoop* main_loop) { NOTREACHED() << "Communication with Google Update failed: " << hr << " error: " << error_code; main_loop->PostTask(FROM_HERE, NewRunnableMethod(this, &GoogleUpdate::ReportResults, UPGRADE_ERROR, error_code)); return false; }