• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/installer/util/self_cleaning_temp_dir.h"
6 
7 #include <windows.h>
8 
9 #include "base/files/file_util.h"
10 #include "base/logging.h"
11 #include "chrome/installer/util/delete_after_reboot_helper.h"
12 
13 namespace installer {
14 
15 // Populates |base_dir| with the topmost directory in the hierarchy of
16 // |temp_parent_dir| that does not exist.  If |temp_parent_dir| exists,
17 // |base_dir| is cleared.
18 // static
GetTopDirToCreate(const base::FilePath & temp_parent_dir,base::FilePath * base_dir)19 void SelfCleaningTempDir::GetTopDirToCreate(
20     const base::FilePath& temp_parent_dir,
21     base::FilePath* base_dir) {
22   DCHECK(base_dir);
23 
24   if (base::PathExists(temp_parent_dir)) {
25     // Empty base_dir means that we didn't create any extra directories.
26     base_dir->clear();
27   } else {
28     base::FilePath parent_dir(temp_parent_dir);
29     do {
30       *base_dir = parent_dir;
31       parent_dir = parent_dir.DirName();
32     } while (parent_dir != *base_dir && !base::PathExists(parent_dir));
33     LOG_IF(WARNING, !base::DirectoryExists(parent_dir))
34         << "A non-directory is at the base of the path leading to a desired "
35            "temp directory location: " << parent_dir.value();
36   }
37 }
38 
SelfCleaningTempDir()39 SelfCleaningTempDir::SelfCleaningTempDir() {
40 }
41 
~SelfCleaningTempDir()42 SelfCleaningTempDir::~SelfCleaningTempDir() {
43   if (!path().empty() && !Delete())
44     LOG(WARNING) << "Failed to clean temp dir in dtor " << path().value();
45 }
46 
Initialize(const base::FilePath & parent_dir,const StringType & temp_name)47 bool SelfCleaningTempDir::Initialize(const base::FilePath& parent_dir,
48                                      const StringType& temp_name) {
49   DCHECK(parent_dir.IsAbsolute());
50   DCHECK(!temp_name.empty());
51 
52   if (!path().empty()) {
53     LOG(DFATAL) << "Attempting to re-initialize a SelfSelfCleaningTempDir.";
54     return false;
55   }
56 
57   base::FilePath temp_dir(parent_dir.Append(temp_name));
58   base::FilePath base_dir;
59   GetTopDirToCreate(parent_dir, &base_dir);
60 
61   if (base::CreateDirectory(temp_dir)) {
62     base_dir_ = base_dir;
63     temp_dir_ = temp_dir;
64     return true;
65   }
66 
67   return false;
68 }
69 
Delete()70 bool SelfCleaningTempDir::Delete() {
71   if (path().empty()) {
72     LOG(DFATAL) << "Attempting to Delete an uninitialized SelfCleaningTempDir.";
73     return false;
74   }
75 
76   base::FilePath next_dir(path().DirName());
77   bool schedule_deletes = false;
78 
79   // First try to recursively delete the leaf directory managed by our
80   // base::ScopedTempDir.
81   if (!base::DeleteFile(path(), true)) {
82     // That failed, so schedule the temp dir and its contents for deletion after
83     // reboot.
84     LOG(WARNING) << "Failed to delete temporary directory " << path().value()
85                  << ". Scheduling for deletion at reboot.";
86     schedule_deletes = true;
87     if (!ScheduleDirectoryForDeletion(path()))
88       return false;  // Entirely unexpected failure (Schedule logs the reason).
89   }
90 
91   // Now delete or schedule all empty directories up to and including our
92   // base_dir_.  Any that can't be deleted are scheduled for deletion at reboot.
93   // This is safe since they'll only be deleted in that case if they're empty.
94   if (!base_dir_.empty()) {
95     do {
96       if (!schedule_deletes && !RemoveDirectory(next_dir.value().c_str())) {
97         PLOG_IF(WARNING, GetLastError() != ERROR_DIR_NOT_EMPTY)
98               << "Error removing directory " << next_dir.value().c_str();
99         schedule_deletes = true;
100       }
101       if (schedule_deletes) {
102         // Ignore the return code.  If we fail to schedule, go ahead and add the
103         // other parent directories anyway.
104         ScheduleFileSystemEntityForDeletion(next_dir);
105       }
106       if (next_dir == base_dir_)
107         break;  // We just processed the topmost directory we created.
108       next_dir = next_dir.DirName();
109     } while (true);
110   }
111 
112   base_dir_.clear();
113   temp_dir_.clear();
114 
115   return true;
116 }
117 
118 }  // namespace installer
119