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

Side by Side Diff: chrome/installer/setup/uninstall.cc

Issue 10665002: Implement installation of the Chrome App Host. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: A basic working app host installer/uninstaller. Created 8 years, 5 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 unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 // 4 //
5 // This file defines the methods useful for uninstalling Chrome. 5 // This file defines the methods useful for uninstalling Chrome.
6 6
7 #include "chrome/installer/setup/uninstall.h" 7 #include "chrome/installer/setup/uninstall.h"
8 8
9 #include <vector> 9 #include <vector>
10 10
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
42 42
43 // Build-time generated include file. 43 // Build-time generated include file.
44 #include "registered_dlls.h" // NOLINT 44 #include "registered_dlls.h" // NOLINT
45 45
46 using base::win::RegKey; 46 using base::win::RegKey;
47 using installer::InstallStatus; 47 using installer::InstallStatus;
48 using installer::MasterPreferences; 48 using installer::MasterPreferences;
49 49
50 namespace { 50 namespace {
51 51
52 // Avoid leaving behind a Temp dir. If one exists, ask SelfCleaningTempDir to
53 // clean it up for us. This may involve scheduling it for deletion after
54 // reboot. Don't report that a reboot is required in this case, however.
55 // TODO(erikwright): Shouldn't this still lead to
56 // ScheduleParentAndGrandparentForDeletion?
tommi (sloooow) - chröme 2012/07/12 08:11:31 if so, file a bug for tracking?
erikwright (departed) 2012/07/16 20:13:11 I don't know the answer. I'll ask Greg OOB.
57 void DeleteInstallTempDir(const FilePath& target_path) {
58 FilePath temp_path(target_path.DirName().Append(installer::kInstallTempDir));
59 if (file_util::DirectoryExists(temp_path)) {
60 installer::SelfCleaningTempDir temp_dir;
61 if (!temp_dir.Initialize(target_path.DirName(),
62 installer::kInstallTempDir) ||
63 !temp_dir.Delete()) {
64 LOG(ERROR) << "Failed to delete temp dir " << temp_path.value();
65 }
66 }
67 }
68
52 // Makes appropriate changes to the Google Update "ap" value in the registry. 69 // Makes appropriate changes to the Google Update "ap" value in the registry.
53 // Specifically, removes the flags associated with this product ("-chrome" or 70 // Specifically, removes the flags associated with this product ("-chrome" or
54 // "-chromeframe[-readymode]") from the "ap" values for all other 71 // "-chromeframe[-readymode]") from the "ap" values for all other
55 // installed products and for the multi-installer package. 72 // installed products and for the multi-installer package.
56 void ProcessGoogleUpdateItems( 73 void ProcessGoogleUpdateItems(
57 const installer::InstallationState& original_state, 74 const installer::InstallationState& original_state,
58 const installer::InstallerState& installer_state, 75 const installer::InstallerState& installer_state,
59 const installer::Product& product) { 76 const installer::Product& product) {
60 DCHECK(installer_state.is_multi_install()); 77 DCHECK(installer_state.is_multi_install());
61 const bool system_level = installer_state.system_install(); 78 const bool system_level = installer_state.system_install();
62 BrowserDistribution* distribution = product.distribution(); 79 BrowserDistribution* distribution = product.distribution();
63 const HKEY reg_root = installer_state.root_key(); 80 const HKEY reg_root = installer_state.root_key();
64 const installer::ProductState* product_state = 81 const installer::ProductState* product_state =
65 original_state.GetProductState(system_level, distribution->GetType()); 82 original_state.GetProductState(system_level, distribution->GetType());
66 DCHECK(product_state != NULL); 83 DCHECK(product_state != NULL);
67 installer::ChannelInfo channel_info; 84 installer::ChannelInfo channel_info;
68 85
69 // Remove product's flags from the channel value. 86 // Remove product's flags from the channel value.
70 channel_info.set_value(product_state->channel().value()); 87 channel_info.set_value(product_state->channel().value());
71 const bool modified = product.SetChannelFlags(false, &channel_info); 88 const bool modified = product.SetChannelFlags(false, &channel_info);
72 89
73 // Apply the new channel value to all other products and to the multi package. 90 // Apply the new channel value to all other products and to the multi package.
74 if (modified) { 91 if (modified) {
75 BrowserDistribution::Type other_dist_types[] = { 92 std::vector<BrowserDistribution::Type> other_dist_types;
76 (distribution->GetType() == BrowserDistribution::CHROME_BROWSER) ? 93 for (size_t i = 0; i < BrowserDistribution::kNumProductTypes; ++i) {
77 BrowserDistribution::CHROME_FRAME : 94 if (distribution->GetType() != BrowserDistribution::kProductTypes[i])
78 BrowserDistribution::CHROME_BROWSER, 95 other_dist_types.push_back(BrowserDistribution::kProductTypes[i]);
79 BrowserDistribution::CHROME_BINARIES 96 }
80 }; 97
81 scoped_ptr<WorkItemList> 98 scoped_ptr<WorkItemList>
82 update_list(WorkItem::CreateNoRollbackWorkItemList()); 99 update_list(WorkItem::CreateNoRollbackWorkItemList());
83 100
84 for (int i = 0; i < arraysize(other_dist_types); ++i) { 101 for (size_t i = 0; i < other_dist_types.size(); ++i) {
102 // TODO(erikwright): Who updates the ap value for the binaries?
85 BrowserDistribution::Type other_dist_type = other_dist_types[i]; 103 BrowserDistribution::Type other_dist_type = other_dist_types[i];
86 product_state = 104 product_state =
87 original_state.GetProductState(system_level, other_dist_type); 105 original_state.GetProductState(system_level, other_dist_type);
88 // Only modify other products if they're installed and multi. 106 // Only modify other products if they're installed and multi.
89 if (product_state != NULL && 107 if (product_state != NULL &&
90 product_state->is_multi_install() && 108 product_state->is_multi_install() &&
91 !product_state->channel().Equals(channel_info)) { 109 !product_state->channel().Equals(channel_info)) {
92 BrowserDistribution* other_dist = 110 BrowserDistribution* other_dist =
93 BrowserDistribution::GetSpecificDistribution(other_dist_type); 111 BrowserDistribution::GetSpecificDistribution(other_dist_type);
94 update_list->AddSetRegValueWorkItem(reg_root, other_dist->GetStateKey(), 112 update_list->AddSetRegValueWorkItem(reg_root, other_dist->GetStateKey(),
(...skipping 294 matching lines...) Expand 10 before | Expand all | Expand 10 after
389 // We cannot delete the file right away, but try to delete it some other 407 // We cannot delete the file right away, but try to delete it some other
390 // way. Either with the help of a different process or the system. 408 // way. Either with the help of a different process or the system.
391 if (ret && !file_util::DeleteAfterReboot(temp_file)) { 409 if (ret && !file_util::DeleteAfterReboot(temp_file)) {
392 static const uint32 kDeleteAfterMs = 10 * 1000; 410 static const uint32 kDeleteAfterMs = 10 * 1000;
393 installer::DeleteFileFromTempProcess(temp_file, kDeleteAfterMs); 411 installer::DeleteFileFromTempProcess(temp_file, kDeleteAfterMs);
394 } 412 }
395 } 413 }
396 return ret; 414 return ret;
397 } 415 }
398 416
399 DeleteResult DeleteFilesAndFolders(const InstallerState& installer_state, 417 DeleteResult DeleteAppHostFilesAndFolders(const InstallerState& installer_state,
400 const Version& installed_version) { 418 const Version& installed_version) {
401 const FilePath& target_path = installer_state.target_path(); 419 const FilePath& target_path = installer_state.target_path();
402 if (target_path.empty()) { 420 if (target_path.empty()) {
403 LOG(ERROR) << "DeleteFilesAndFolders: no installation destination path."; 421 LOG(ERROR) << "DeleteAppHostFilesAndFolders: no installation destination "
422 << "path.";
404 return DELETE_FAILED; // Nothing else we can do to uninstall, so we return. 423 return DELETE_FAILED; // Nothing else we can do to uninstall, so we return.
405 } 424 }
406 425
426 DeleteInstallTempDir(target_path);
427
407 DeleteResult result = DELETE_SUCCEEDED; 428 DeleteResult result = DELETE_SUCCEEDED;
408 429
409 // Avoid leaving behind a Temp dir. If one exists, ask SelfCleaningTempDir to 430 FilePath app_host_exe(target_path.Append(installer::kChromeAppHostExe));
410 // clean it up for us. This may involve scheduling it for deletion after 431 if (!file_util::Delete(app_host_exe, false)) {
411 // reboot. Don't report that a reboot is required in this case, however. 432 result = DELETE_FAILED;
412 FilePath temp_path(target_path.DirName().Append(kInstallTempDir)); 433 LOG(ERROR) << "Failed to delete path: " << app_host_exe.value();
413 if (file_util::DirectoryExists(temp_path)) { 434 } else {
414 installer::SelfCleaningTempDir temp_dir; 435 DeleteEmptyParentDir(target_path);
415 if (!temp_dir.Initialize(target_path.DirName(), kInstallTempDir) ||
416 !temp_dir.Delete()) {
417 LOG(ERROR) << "Failed to delete temp dir " << temp_path.value();
418 }
419 } 436 }
420 437
421 VLOG(1) << "Deleting install path " << target_path.value(); 438 return result;
422 if (!file_util::Delete(target_path, true)) { 439 }
423 LOG(ERROR) << "Failed to delete folder (1st try): " << target_path.value(); 440
424 if (installer_state.FindProduct(BrowserDistribution::CHROME_FRAME)) { 441 DeleteResult DeleteChromeFilesAndFolders(const InstallerState& installer_state,
425 // We don't try killing Chrome processes for Chrome Frame builds since 442 const Version& installed_version) {
426 // that is unlikely to help. Instead, schedule files for deletion and 443 const FilePath& target_path = installer_state.target_path();
427 // return a value that will trigger a reboot prompt. 444 if (target_path.empty()) {
428 ScheduleDirectoryForDeletion(target_path.value().c_str()); 445 LOG(ERROR) << "DeleteChromeFilesAndFolders: no installation destination "
429 result = DELETE_REQUIRES_REBOOT; 446 << "path.";
430 } else { 447 return DELETE_FAILED; // Nothing else we can do to uninstall, so we return.
431 // Try closing any running chrome processes and deleting files once again. 448 }
432 CloseAllChromeProcesses(); 449
433 if (!file_util::Delete(target_path, true)) { 450 DeleteInstallTempDir(target_path);
434 LOG(ERROR) << "Failed to delete folder (2nd try): " 451
435 << target_path.value(); 452 DeleteResult result = DELETE_SUCCEEDED;
436 result = DELETE_FAILED; 453
454 using file_util::FileEnumerator;
455 FileEnumerator file_enumerator(
456 target_path,
457 false,
458 static_cast<FileEnumerator::FileType>(FileEnumerator::FILES |
459 FileEnumerator::DIRECTORIES));
460 while (true) {
461 FilePath to_delete(file_enumerator.Next());
462 if (to_delete.empty())
463 break;
464 if (to_delete.BaseName().value() == installer::kChromeAppHostExe)
465 continue;
466
467 VLOG(1) << "Deleting install path " << target_path.value();
468 if (!file_util::Delete(to_delete, true)) {
469 LOG(ERROR) << "Failed to delete path (1st try): " << to_delete.value();
470 if (installer_state.FindProduct(BrowserDistribution::CHROME_FRAME)) {
471 // We don't try killing Chrome processes for Chrome Frame builds since
472 // that is unlikely to help. Instead, schedule files for deletion and
473 // return a value that will trigger a reboot prompt.
474 FileEnumerator::FindInfo find_info;
475 file_enumerator.GetFindInfo(&find_info);
476 if (FileEnumerator::IsDirectory(find_info))
477 ScheduleDirectoryForDeletion(to_delete.value().c_str());
478 else
479 ScheduleFileSystemEntityForDeletion(to_delete.value().c_str());
480 result = DELETE_REQUIRES_REBOOT;
481 } else {
482 // Try closing any running Chrome processes and deleting files once
483 // again.
484 CloseAllChromeProcesses();
485 if (!file_util::Delete(to_delete, true)) {
486 LOG(ERROR) << "Failed to delete path (2nd try): "
487 << to_delete.value();
488 result = DELETE_FAILED;
489 break;
490 }
437 } 491 }
438 } 492 }
439 } 493 }
440 494
441 if (result == DELETE_REQUIRES_REBOOT) { 495 if (result == DELETE_REQUIRES_REBOOT) {
442 // If we need a reboot to continue, schedule the parent directories for 496 // If we need a reboot to continue, schedule the parent directories for
443 // deletion unconditionally. If they are not empty, the session manager 497 // deletion unconditionally. If they are not empty, the session manager
444 // will not delete them on reboot. 498 // will not delete them on reboot.
445 ScheduleParentAndGrandparentForDeletion(target_path); 499 ScheduleParentAndGrandparentForDeletion(target_path);
446 } else { 500 } else {
(...skipping 43 matching lines...) Expand 10 before | Expand all | Expand 10 after
490 544
491 bool ShouldDeleteProfile(const InstallerState& installer_state, 545 bool ShouldDeleteProfile(const InstallerState& installer_state,
492 const CommandLine& cmd_line, InstallStatus status, 546 const CommandLine& cmd_line, InstallStatus status,
493 const Product& product) { 547 const Product& product) {
494 bool should_delete = false; 548 bool should_delete = false;
495 549
496 // Chrome Frame uninstallations always want to delete the profile (we have no 550 // Chrome Frame uninstallations always want to delete the profile (we have no
497 // UI to prompt otherwise and the profile stores no useful data anyway) 551 // UI to prompt otherwise and the profile stores no useful data anyway)
498 // unless they are managed by MSI. MSI uninstalls will explicitly include 552 // unless they are managed by MSI. MSI uninstalls will explicitly include
499 // the --delete-profile flag to distinguish them from MSI upgrades. 553 // the --delete-profile flag to distinguish them from MSI upgrades.
500 if (!product.is_chrome() && !installer_state.is_msi()) { 554 if (product.is_chrome_frame() && !installer_state.is_msi()) {
501 should_delete = true; 555 should_delete = true;
502 } else { 556 } else {
503 should_delete = 557 should_delete =
504 status == installer::UNINSTALL_DELETE_PROFILE || 558 status == installer::UNINSTALL_DELETE_PROFILE ||
505 cmd_line.HasSwitch(installer::switches::kDeleteProfile); 559 cmd_line.HasSwitch(installer::switches::kDeleteProfile);
506 } 560 }
507 561
508 return should_delete; 562 return should_delete;
509 } 563 }
510 564
(...skipping 322 matching lines...) Expand 10 before | Expand all | Expand 10 after
833 if (installer_state.system_install() || 887 if (installer_state.system_install() ||
834 (remove_all && 888 (remove_all &&
835 ShellUtil::QuickIsChromeRegisteredInHKLM( 889 ShellUtil::QuickIsChromeRegisteredInHKLM(
836 browser_dist, chrome_exe, suffix))) { 890 browser_dist, chrome_exe, suffix))) {
837 DeleteChromeRegistrationKeys(browser_dist, HKEY_LOCAL_MACHINE, suffix, 891 DeleteChromeRegistrationKeys(browser_dist, HKEY_LOCAL_MACHINE, suffix,
838 installer_state.target_path(), &ret); 892 installer_state.target_path(), &ret);
839 } 893 }
840 894
841 ProcessDelegateExecuteWorkItems(installer_state, product); 895 ProcessDelegateExecuteWorkItems(installer_state, product);
842 896
843 if (!is_chrome) { 897 if (product.is_chrome_frame()) {
844 ProcessChromeFrameWorkItems(original_state, installer_state, setup_path, 898 ProcessChromeFrameWorkItems(original_state, installer_state, setup_path,
845 product); 899 product);
846 } 900 }
847 901
848 if (installer_state.is_multi_install()) 902 if (installer_state.is_multi_install())
849 ProcessGoogleUpdateItems(original_state, installer_state, product); 903 ProcessGoogleUpdateItems(original_state, installer_state, product);
850 904
851 ProcessQuickEnableWorkItems(installer_state, original_state); 905 ProcessQuickEnableWorkItems(installer_state, original_state);
852 906
853 // Get the state of the installed product (if any) 907 // Get the state of the installed product (if any)
(...skipping 27 matching lines...) Expand all
881 935
882 AddRegisterComDllWorkItems(dll_folder, 936 AddRegisterComDllWorkItems(dll_folder,
883 com_dll_list, 937 com_dll_list,
884 installer_state.system_install(), 938 installer_state.system_install(),
885 false, // Unregister 939 false, // Unregister
886 true, // May fail 940 true, // May fail
887 unreg_work_item_list.get()); 941 unreg_work_item_list.get());
888 unreg_work_item_list->Do(); 942 unreg_work_item_list->Do();
889 } 943 }
890 944
891 if (!is_chrome) 945 if (product.is_chrome_frame())
892 ProcessIELowRightsPolicyWorkItems(installer_state); 946 ProcessIELowRightsPolicyWorkItems(installer_state);
893 } 947 }
894 948
895 // Close any Chrome Frame helper processes that may be running. 949 // Close any Chrome Frame helper processes that may be running.
896 if (product.is_chrome_frame()) { 950 if (product.is_chrome_frame()) {
897 VLOG(1) << "Closing the Chrome Frame helper process"; 951 VLOG(1) << "Closing the Chrome Frame helper process";
898 CloseChromeFrameHelperProcess(); 952 CloseChromeFrameHelperProcess();
899 } 953 }
900 954
901 if (product_state == NULL) 955 if (product_state == NULL)
902 return installer::UNINSTALL_SUCCESSFUL; 956 return installer::UNINSTALL_SUCCESSFUL;
903 957
904 // Finally delete all the files from Chrome folder after moving setup.exe 958 // Finally delete all the files from Chrome folder after moving setup.exe
905 // and the user's Local State to a temp location. 959 // and the user's Local State to a temp location.
906 bool delete_profile = ShouldDeleteProfile(installer_state, cmd_line, status, 960 bool delete_profile = ShouldDeleteProfile(installer_state, cmd_line, status,
907 product); 961 product);
908 ret = installer::UNINSTALL_SUCCESSFUL; 962 ret = installer::UNINSTALL_SUCCESSFUL;
909 963
910 // When deleting files, we must make sure that we're either a "single" 964 // When deleting files, we must make sure that we're either a "single"
911 // (aka non-multi) installation or, in the case of multi, that no other 965 // (aka non-multi) installation or we are the Chrome Binaries.
912 // "multi" products share the binaries we are about to delete.
913 966
914 bool can_delete_files = true; 967 FilePath backup_state_file(
915 if (installer_state.is_multi_install()) { 968 BackupLocalStateFile(GetLocalStateFolder(product)));
916 ProductState prod_state;
917 for (size_t i = 0; i < BrowserDistribution::kNumProductTypes; ++i) {
918 if (prod_state.Initialize(installer_state.system_install(),
919 BrowserDistribution::kProductTypes[i]) &&
920 prod_state.is_multi_install()) {
921 can_delete_files = false;
922 break;
923 }
924 }
925 LOG(INFO) << (can_delete_files ? "Shared binaries will be deleted." :
926 "Shared binaries still in use.");
927 if (can_delete_files) {
928 BrowserDistribution* multi_dist =
929 installer_state.multi_package_binaries_distribution();
930 InstallUtil::DeleteRegistryKey(reg_root, multi_dist->GetVersionKey());
931 }
932 }
933
934 FilePath backup_state_file(BackupLocalStateFile(
935 GetLocalStateFolder(product)));
936 969
937 DeleteResult delete_result = DELETE_SUCCEEDED; 970 DeleteResult delete_result = DELETE_SUCCEEDED;
938 if (can_delete_files) { 971
972 if (product.is_chrome_app_host()) {
973 DeleteAppHostFilesAndFolders(installer_state, product_state->version());
974 } else if (!installer_state.is_multi_install() ||
975 product.is_chrome_binaries()) {
976
939 // In order to be able to remove the folder in which we're running, we 977 // In order to be able to remove the folder in which we're running, we
940 // need to move setup.exe out of the install folder. 978 // need to move setup.exe out of the install folder.
941 // TODO(tommi): What if the temp folder is on a different volume? 979 // TODO(tommi): What if the temp folder is on a different volume?
942 MoveSetupOutOfInstallFolder(installer_state, setup_path, 980 MoveSetupOutOfInstallFolder(installer_state, setup_path,
943 product_state->version()); 981 product_state->version());
944 delete_result = DeleteFilesAndFolders(installer_state, 982 delete_result = DeleteChromeFilesAndFolders(installer_state,
945 product_state->version()); 983 product_state->version());
946 } 984 }
947 985
948 if (delete_profile) 986 if (delete_profile)
949 DeleteLocalState(product); 987 DeleteLocalState(product);
950 988
951 if (delete_result == DELETE_FAILED) { 989 if (delete_result == DELETE_FAILED) {
952 ret = installer::UNINSTALL_FAILED; 990 ret = installer::UNINSTALL_FAILED;
953 } else if (delete_result == DELETE_REQUIRES_REBOOT) { 991 } else if (delete_result == DELETE_REQUIRES_REBOOT) {
954 ret = installer::UNINSTALL_REQUIRES_REBOOT; 992 ret = installer::UNINSTALL_REQUIRES_REBOOT;
955 } 993 }
956 994
957 if (!force_uninstall) { 995 if (!force_uninstall) {
958 VLOG(1) << "Uninstallation complete. Launching Uninstall survey."; 996 VLOG(1) << "Uninstallation complete. Launching post-uninstall operations.";
959 browser_dist->DoPostUninstallOperations(product_state->version(), 997 browser_dist->DoPostUninstallOperations(product_state->version(),
960 backup_state_file, distribution_data); 998 backup_state_file, distribution_data);
961 } 999 }
962 1000
963 // Try and delete the preserved local state once the post-install 1001 // Try and delete the preserved local state once the post-install
964 // operations are complete. 1002 // operations are complete.
965 if (!backup_state_file.empty()) 1003 if (!backup_state_file.empty())
966 file_util::Delete(backup_state_file, false); 1004 file_util::Delete(backup_state_file, false);
967 1005
968 return ret; 1006 return ret;
969 } 1007 }
970 1008
971 } // namespace installer 1009 } // namespace installer
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698