Index: chrome/browser/chromeos/app_mode/kiosk_external_updater.cc |
diff --git a/chrome/browser/chromeos/app_mode/kiosk_external_updater.cc b/chrome/browser/chromeos/app_mode/kiosk_external_updater.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..dea8b8b5540230aada74112c503f1deb3ec93dcc |
--- /dev/null |
+++ b/chrome/browser/chromeos/app_mode/kiosk_external_updater.cc |
@@ -0,0 +1,517 @@ |
+// 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 "chrome/browser/chromeos/app_mode/kiosk_external_updater.h" |
+ |
+#include "base/bind.h" |
+#include "base/file_util.h" |
+#include "base/files/file_enumerator.h" |
+#include "base/json/json_file_value_serializer.h" |
+#include "base/location.h" |
+#include "base/logging.h" |
+#include "base/strings/utf_string_conversions.h" |
+#include "base/version.h" |
+#include "chrome/browser/chromeos/app_mode/kiosk_app_manager.h" |
+#include "chrome/browser/chromeos/ui/kiosk_external_update_notification.h" |
+#include "chrome/browser/extensions/sandboxed_unpacker.h" |
+#include "chrome/common/chrome_version_info.h" |
+#include "content/public/browser/browser_thread.h" |
+#include "extensions/common/extension.h" |
+#include "grit/chromium_strings.h" |
+#include "grit/generated_resources.h" |
+#include "ui/base/l10n/l10n_util.h" |
+#include "ui/base/resource/resource_bundle.h" |
+ |
+namespace chromeos { |
+ |
+namespace { |
+ |
+const char kExternalUpdateManifest[] = "external_update.json"; |
+const char kExternalCrx[] = "external_crx"; |
+const char kExternalVersion[] = "external_version"; |
+ |
+void ParseExternalUpdateManifest( |
+ const base::FilePath& external_update_dir, |
+ base::DictionaryValue* parsed_manifest, |
+ KioskExternalUpdater::ExternalUpdateErrorCode* error_code) { |
+ base::FilePath manifest = |
+ external_update_dir.AppendASCII(kExternalUpdateManifest); |
+ if (!base::PathExists(manifest)) { |
+ *error_code = KioskExternalUpdater::ERROR_NO_MANIFEST; |
+ return; |
+ } |
+ |
+ JSONFileValueSerializer serializer(manifest); |
+ std::string error_msg; |
+ base::Value* extensions = serializer.Deserialize(NULL, &error_msg); |
+ if (!extensions) { |
+ *error_code = KioskExternalUpdater::ERROR_INVALID_MANIFEST; |
+ return; |
+ } |
+ |
+ base::DictionaryValue* dict_value = NULL; |
+ if (!extensions->GetAsDictionary(&dict_value)) { |
+ *error_code = KioskExternalUpdater::ERROR_INVALID_MANIFEST; |
+ return; |
+ } |
+ |
+ parsed_manifest->Swap(dict_value); |
+ *error_code = KioskExternalUpdater::ERROR_NONE; |
+} |
+ |
+// Copies |external_crx_file| to |temp_crx_file|, and removes |temp_dir| |
+// created for unpacking |external_crx_file|. |
+void CopyExternalCrxAndDeleteTempDir(const base::FilePath& external_crx_file, |
+ const base::FilePath& temp_crx_file, |
+ const base::FilePath& temp_dir, |
+ bool* success) { |
+ base::DeleteFile(temp_dir, true); |
+ *success = base::CopyFile(external_crx_file, temp_crx_file); |
+} |
+ |
+// Returns true if |version_1| < |version_2|, and |
+// if |update_for_same_version| is true and |version_1| = |version_2|. |
+bool ShouldUpdateForHigherVersion(const std::string& version_1, |
+ const std::string& version_2, |
+ bool update_for_same_version) { |
+ const base::Version v1(version_1); |
+ const base::Version v2(version_2); |
+ if (!v1.IsValid() || !v2.IsValid()) |
+ return false; |
+ int compare_result = v1.CompareTo(v2); |
+ if (compare_result < 0) |
+ return true; |
+ else if (update_for_same_version && compare_result == 0) |
+ return true; |
+ else |
+ return false; |
+} |
+ |
+} // namespace |
+ |
+KioskExternalUpdater::ExternalUpdate::ExternalUpdate() { |
+} |
+ |
+KioskExternalUpdater::KioskExternalUpdater( |
+ const scoped_refptr<base::SequencedTaskRunner>& backend_task_runner, |
+ const base::FilePath& crx_cache_dir, |
+ const base::FilePath& crx_unpack_dir) |
+ : backend_task_runner_(backend_task_runner), |
+ crx_cache_dir_(crx_cache_dir), |
+ crx_unpack_dir_(crx_unpack_dir), |
+ weak_factory_(this) { |
+ // Subscribe to DiskMountManager. |
+ DCHECK(disks::DiskMountManager::GetInstance()); |
+ disks::DiskMountManager::GetInstance()->AddObserver(this); |
+} |
+ |
+KioskExternalUpdater::~KioskExternalUpdater() { |
+ if (disks::DiskMountManager::GetInstance()) |
+ disks::DiskMountManager::GetInstance()->RemoveObserver(this); |
+} |
+ |
+void KioskExternalUpdater::OnDiskEvent( |
+ disks::DiskMountManager::DiskEvent event, |
+ const disks::DiskMountManager::Disk* disk) { |
+} |
+ |
+void KioskExternalUpdater::OnDeviceEvent( |
+ disks::DiskMountManager::DeviceEvent event, |
+ const std::string& device_path) { |
+} |
+ |
+void KioskExternalUpdater::OnMountEvent( |
+ disks::DiskMountManager::MountEvent event, |
+ MountError error_code, |
+ const disks::DiskMountManager::MountPointInfo& mount_info) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ if (mount_info.mount_type != MOUNT_TYPE_DEVICE || |
+ error_code != MOUNT_ERROR_NONE) { |
+ return; |
+ } |
+ |
+ if (event == disks::DiskMountManager::MOUNTING) { |
+ // If multiple disks have been mounted, skip the rest of them if kiosk |
+ // update has already been found. |
+ if (!external_update_path_.empty()) { |
+ LOG(WARNING) << "External update path already found, skip " |
+ << mount_info.mount_path; |
+ return; |
+ } |
+ |
+ NotifyKioskUpdateProgress( |
+ ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_IN_PROGRESS)); |
+ |
+ base::DictionaryValue* parsed_manifest = new base::DictionaryValue(); |
+ ExternalUpdateErrorCode* parsing_error = new ExternalUpdateErrorCode; |
+ backend_task_runner_->PostTaskAndReply( |
+ FROM_HERE, |
+ base::Bind(&ParseExternalUpdateManifest, |
+ base::FilePath(mount_info.mount_path), |
+ parsed_manifest, |
+ parsing_error), |
+ base::Bind(&KioskExternalUpdater::ProcessParsedManifest, |
+ weak_factory_.GetWeakPtr(), |
+ base::Owned(parsing_error), |
+ base::FilePath(mount_info.mount_path), |
+ base::Owned(parsed_manifest))); |
+ } else { // unmounting a removable device. |
+ if (external_update_path_.value().empty()) { |
+ // Clear any previously displayed message. |
+ DismissKioskUpdateNotification(); |
+ } else if (external_update_path_.value() == mount_info.mount_path) { |
+ DismissKioskUpdateNotification(); |
+ if (IsExternalUpdatePending()) { |
+ LOG(ERROR) << "External kiosk update is not completed when the usb " |
+ "stick is unmoutned."; |
+ } |
+ external_updates_.clear(); |
+ external_update_path_.clear(); |
+ } |
+ } |
+} |
+ |
+void KioskExternalUpdater::OnFormatEvent( |
+ disks::DiskMountManager::FormatEvent event, |
+ FormatError error_code, |
+ const std::string& device_path) { |
+} |
+ |
+void KioskExternalUpdater::OnExtenalUpdateUnpackSuccess( |
+ const std::string& app_id, |
+ const std::string& version, |
+ const std::string& min_browser_version, |
+ const base::FilePath& temp_dir) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ |
+ // User might pull out the usb stick before updating is completed. |
+ if (CheckExternalUpdateInterrupted()) |
+ return; |
+ |
+ if (!ShouldDoExternalUpdate(app_id, version, min_browser_version)) { |
+ external_updates_[app_id].update_status = FAILED; |
+ MaybeValidateNextExternalUpdate(); |
+ return; |
+ } |
+ |
+ // User might pull out the usb stick before updating is completed. |
+ if (CheckExternalUpdateInterrupted()) |
+ return; |
+ |
+ base::FilePath external_crx_path = external_updates_[app_id].external_crx; |
+ base::FilePath temp_crx_path = |
+ crx_unpack_dir_.Append(external_crx_path.BaseName()); |
+ bool* success = new bool; |
+ backend_task_runner_->PostTaskAndReply( |
+ FROM_HERE, |
+ base::Bind(&CopyExternalCrxAndDeleteTempDir, |
+ external_crx_path, |
+ temp_crx_path, |
+ temp_dir, |
+ success), |
+ base::Bind(&KioskExternalUpdater::PutValidatedExtension, |
+ weak_factory_.GetWeakPtr(), |
+ base::Owned(success), |
+ app_id, |
+ temp_crx_path, |
+ version)); |
+} |
+ |
+void KioskExternalUpdater::OnExternalUpdateUnpackFailure( |
+ const std::string& app_id) { |
+ // User might pull out the usb stick before updating is completed. |
+ if (CheckExternalUpdateInterrupted()) |
+ return; |
+ |
+ external_updates_[app_id].update_status = FAILED; |
+ external_updates_[app_id].error = |
+ ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_BAD_CRX); |
+ MaybeValidateNextExternalUpdate(); |
+} |
+ |
+void KioskExternalUpdater::ProcessParsedManifest( |
+ ExternalUpdateErrorCode* parsing_error, |
+ const base::FilePath& external_update_dir, |
+ base::DictionaryValue* parsed_manifest) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ |
+ if (*parsing_error == ERROR_NO_MANIFEST) { |
+ NotifyKioskUpdateProgress( |
+ ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_NO_MANIFEST)); |
+ return; |
+ } else if (*parsing_error == ERROR_INVALID_MANIFEST) { |
+ NotifyKioskUpdateProgress( |
+ ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_INVALID_MANIFEST)); |
+ return; |
+ } |
+ |
+ external_update_path_ = external_update_dir; |
+ for (base::DictionaryValue::Iterator it(*parsed_manifest); !it.IsAtEnd(); |
+ it.Advance()) { |
+ std::string app_id = it.key(); |
+ std::string cached_version_str; |
+ base::FilePath cached_crx; |
+ if (!KioskAppManager::Get()->GetCachedCrx( |
+ app_id, &cached_crx, &cached_version_str)) { |
+ LOG(WARNING) << "Can't find app in existing cache " << app_id; |
+ continue; |
+ } |
+ |
+ const base::DictionaryValue* extension = NULL; |
+ if (!it.value().GetAsDictionary(&extension)) { |
+ LOG(ERROR) << "Found bad entry in manifest type " << it.value().GetType(); |
+ continue; |
+ } |
+ |
+ std::string external_crx_str; |
+ if (!extension->GetString(kExternalCrx, &external_crx_str)) { |
+ LOG(ERROR) << "Can't find external crx in manifest " << app_id; |
+ continue; |
+ } |
+ |
+ std::string external_version_str; |
+ if (extension->GetString(kExternalVersion, &external_version_str)) { |
+ if (!ShouldUpdateForHigherVersion( |
+ cached_version_str, external_version_str, false)) { |
+ LOG(WARNING) << "External app " << app_id |
+ << " is at the same or lower version comparing to " |
+ << " the existing one."; |
+ continue; |
+ } |
+ } |
+ |
+ ExternalUpdate update; |
+ KioskAppManager::App app; |
+ if (KioskAppManager::Get()->GetApp(app_id, &app)) { |
+ update.app_name = app.name; |
+ } else { |
+ NOTREACHED(); |
+ } |
+ update.external_crx = external_update_path_.AppendASCII(external_crx_str); |
+ update.update_status = PENDING; |
+ external_updates_[app_id] = update; |
+ } |
+ |
+ if (external_updates_.empty()) { |
+ NotifyKioskUpdateProgress( |
+ ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_NO_UPDATES)); |
+ return; |
+ } |
+ |
+ ValidateExternalUpdates(); |
+} |
+ |
+bool KioskExternalUpdater::CheckExternalUpdateInterrupted() { |
+ if (external_updates_.empty()) { |
+ // This could happen if user pulls out the usb stick before the updating |
+ // operation is completed. |
+ LOG(ERROR) << "external_updates_ has been cleared before external " |
+ << "updating completes."; |
+ return true; |
+ } |
+ |
+ return false; |
+} |
+ |
+void KioskExternalUpdater::ValidateExternalUpdates() { |
+ for (ExternalUpdateMap::iterator it = external_updates_.begin(); |
+ it != external_updates_.end(); |
+ ++it) { |
+ if (it->second.update_status == PENDING) { |
+ scoped_refptr<KioskExternalUpdateValidator> crx_validator = |
+ new KioskExternalUpdateValidator(backend_task_runner_, |
+ it->first, |
+ it->second.external_crx, |
+ crx_unpack_dir_, |
+ weak_factory_.GetWeakPtr()); |
+ crx_validator->Start(); |
+ break; |
+ } |
+ } |
+} |
+ |
+bool KioskExternalUpdater::IsExternalUpdatePending() { |
+ for (ExternalUpdateMap::iterator it = external_updates_.begin(); |
+ it != external_updates_.end(); |
+ ++it) { |
+ if (it->second.update_status == PENDING) { |
+ return true; |
+ } |
+ } |
+ return false; |
+} |
+ |
+bool KioskExternalUpdater::ShouldDoExternalUpdate( |
+ const std::string& app_id, |
+ const std::string& version, |
+ const std::string& min_browser_version) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ |
+ std::string existing_version_str; |
+ base::FilePath existing_path; |
+ bool cached = KioskAppManager::Get()->GetCachedCrx( |
+ app_id, &existing_path, &existing_version_str); |
+ DCHECK(cached); |
+ |
+ // Compare app version. |
+ ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); |
+ if (!ShouldUpdateForHigherVersion(existing_version_str, version, false)) { |
+ external_updates_[app_id].error = rb.GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_SAME_OR_LOWER_APP_VERSION); |
+ return false; |
+ } |
+ |
+ // Check minimum browser version. |
+ if (!min_browser_version.empty()) { |
+ chrome::VersionInfo current_version_info; |
+ if (!ShouldUpdateForHigherVersion( |
+ min_browser_version, current_version_info.Version(), true)) { |
+ external_updates_[app_id].error = l10n_util::GetStringFUTF16( |
+ IDS_KIOSK_EXTERNAL_UPDATE_REQUIRE_HIGHER_BROWSER_VERSION, |
+ base::UTF8ToUTF16(min_browser_version)); |
+ return false; |
+ } |
+ } |
+ |
+ return true; |
+} |
+ |
+void KioskExternalUpdater::PutValidatedExtension(bool* crx_copied, |
+ const std::string& app_id, |
+ const base::FilePath& crx_file, |
+ const std::string& version) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ if (CheckExternalUpdateInterrupted()) |
+ return; |
+ |
+ if (!*crx_copied) { |
+ LOG(ERROR) << "Cannot copy external crx file to " << crx_file.value(); |
+ external_updates_[app_id].update_status = FAILED; |
+ external_updates_[app_id].error = l10n_util::GetStringFUTF16( |
+ IDS_KIOSK_EXTERNAL_UPDATE_FAILED_COPY_CRX_TO_TEMP, |
+ base::UTF8ToUTF16(crx_file.value())); |
+ MaybeValidateNextExternalUpdate(); |
+ return; |
+ } |
+ |
+ chromeos::KioskAppManager::Get()->PutValidatedExternalExtension( |
+ app_id, |
+ crx_file, |
+ version, |
+ base::Bind(&KioskExternalUpdater::OnPutValidatedExtension, |
+ weak_factory_.GetWeakPtr())); |
+} |
+ |
+void KioskExternalUpdater::OnPutValidatedExtension(const std::string& app_id, |
+ bool success) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ if (CheckExternalUpdateInterrupted()) |
+ return; |
+ |
+ if (!success) { |
+ external_updates_[app_id].update_status = FAILED; |
+ external_updates_[app_id].error = l10n_util::GetStringFUTF16( |
+ IDS_KIOSK_EXTERNAL_UPDATE_CANNOT_INSTALL_IN_LOCAL_CACHE, |
+ base::UTF8ToUTF16(external_updates_[app_id].external_crx.value())); |
+ } else { |
+ external_updates_[app_id].update_status = SUCCESS; |
+ } |
+ |
+ // Validate the next pending external update. |
+ MaybeValidateNextExternalUpdate(); |
+} |
+ |
+void KioskExternalUpdater::MaybeValidateNextExternalUpdate() { |
+ if (IsExternalUpdatePending()) |
+ ValidateExternalUpdates(); |
+ else |
+ MayBeNotifyKioskAppUpdate(); |
+} |
+ |
+void KioskExternalUpdater::MayBeNotifyKioskAppUpdate() { |
+ if (IsExternalUpdatePending()) |
+ return; |
+ |
+ NotifyKioskUpdateProgress(GetUpdateReportMessage()); |
+ NotifyKioskAppUpdateAvailable(); |
+} |
+ |
+void KioskExternalUpdater::NotifyKioskAppUpdateAvailable() { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
+ for (ExternalUpdateMap::iterator it = external_updates_.begin(); |
+ it != external_updates_.end(); |
+ ++it) { |
+ if (it->second.update_status == SUCCESS) { |
+ KioskAppManager::Get()->OnKioskAppCacheUpdated(it->first); |
+ } |
+ } |
+} |
+ |
+void KioskExternalUpdater::NotifyKioskUpdateProgress( |
+ const base::string16& message) { |
+ if (!notification_) |
+ notification_.reset(new KioskExternalUpdateNotification(message)); |
+ else |
+ notification_->ShowMessage(message); |
+} |
+ |
+void KioskExternalUpdater::DismissKioskUpdateNotification() { |
+ if (notification_.get()) { |
+ notification_.reset(); |
+ } |
+} |
+ |
+base::string16 KioskExternalUpdater::GetUpdateReportMessage() { |
+ DCHECK(!IsExternalUpdatePending()); |
+ int updated = 0; |
+ int failed = 0; |
+ base::string16 updated_apps; |
+ base::string16 failed_apps; |
+ for (ExternalUpdateMap::iterator it = external_updates_.begin(); |
+ it != external_updates_.end(); |
+ ++it) { |
+ base::string16 app_name = base::UTF8ToUTF16(it->second.app_name); |
+ if (it->second.update_status == SUCCESS) { |
+ ++updated; |
+ if (updated_apps.empty()) |
+ updated_apps = app_name; |
+ else |
+ updated_apps = updated_apps + base::ASCIIToUTF16(", ") + app_name; |
+ } else { // FAILED |
+ ++failed; |
+ if (failed_apps.empty()) { |
+ failed_apps = app_name + base::ASCIIToUTF16(": ") + it->second.error; |
+ } else { |
+ failed_apps = failed_apps + base::ASCIIToUTF16("\n") + app_name + |
+ base::ASCIIToUTF16(": ") + it->second.error; |
+ } |
+ } |
+ } |
+ |
+ base::string16 message; |
+ message = ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_COMPLETE); |
+ base::string16 success_app_msg; |
+ if (updated) { |
+ success_app_msg = l10n_util::GetStringFUTF16( |
+ IDS_KIOSK_EXTERNAL_UPDATE_SUCCESSFUL_UPDATED_APPS, updated_apps); |
+ message = message + base::ASCIIToUTF16("\n") + success_app_msg; |
+ } |
+ |
+ base::string16 failed_app_msg; |
+ if (failed) { |
+ failed_app_msg = ui::ResourceBundle::GetSharedInstance().GetLocalizedString( |
+ IDS_KIOSK_EXTERNAL_UPDATE_FAILED_UPDATED_APPS) + |
+ base::ASCIIToUTF16("\n") + failed_apps; |
+ message = message + base::ASCIIToUTF16("\n") + failed_app_msg; |
+ } |
+ return message; |
+} |
+ |
+} // namespace chromeos |