OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2016 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/delete_old_versions.h" | |
6 | |
7 #include <set> | |
8 | |
9 #include "base/files/file.h" | |
10 #include "base/files/file_enumerator.h" | |
11 #include "base/files/file_util.h" | |
12 #include "base/files/scoped_temp_dir.h" | |
13 #include "base/logging.h" | |
14 #include "base/path_service.h" | |
15 #include "base/strings/string16.h" | |
16 #include "base/strings/string_util.h" | |
17 #include "base/strings/utf_string_conversions.h" | |
18 #include "base/version.h" | |
19 #include "chrome/installer/test/alternate_version_generator.h" | |
20 #include "chrome/installer/util/util_constants.h" | |
21 #include "testing/gtest/include/gtest/gtest.h" | |
22 | |
23 namespace { | |
24 | |
25 const base::char16 kVersionA[] = L"47.0.0.0"; | |
26 const base::char16 kVersionB[] = L"48.0.0.0"; | |
27 | |
28 class DeleteOldVersionsTest : public testing::Test { | |
29 protected: | |
30 bool CreateInstallDir() { return install_dir_.CreateUniqueTempDir(); } | |
31 | |
32 // Creates an executable with |name| and |version| in |install_dir_|. | |
33 bool CreateExecutable(const base::string16& name, | |
34 const base::string16& version) { | |
35 base::FilePath current_exe_path; | |
36 return base::PathService::Get(base::FILE_EXE, ¤t_exe_path) && | |
37 upgrade_test::GenerateSpecificPEFileVersion( | |
38 current_exe_path, install_dir().Append(name), | |
39 base::Version(base::UTF16ToUTF8(version))); | |
40 } | |
41 | |
42 // Creates a version directory named |name| in |install_dir_|. | |
43 bool CreateVersionDirectory(const base::string16& name) { | |
44 const char kDummyContent[] = "dummy"; | |
grt (UTC plus 2)
2016/02/24 20:06:17
nit: static const char for string literal constant
fdoray
2016/02/25 18:26:48
Done.
| |
45 const base::FilePath version_dir_path(install_dir().Append(name)); | |
46 | |
47 return base::CreateDirectory(install_dir().Append(name)) && | |
48 base::CreateDirectory(version_dir_path.Append(L"Installer")) && | |
49 base::WriteFile(version_dir_path.Append(L"chrome.dll"), | |
50 kDummyContent, sizeof(kDummyContent)) && | |
51 base::WriteFile(version_dir_path.Append(L"nacl64.exe"), | |
52 kDummyContent, sizeof(kDummyContent)) && | |
53 base::WriteFile(version_dir_path.Append(L"icudtl.dat"), | |
54 kDummyContent, sizeof(kDummyContent)) && | |
55 base::WriteFile(version_dir_path.Append(L"Installer\\setup.exe"), | |
56 kDummyContent, sizeof(kDummyContent)); | |
57 } | |
58 | |
59 // Returns the relative paths of all files and directories in |install_dir_|. | |
60 using FilePathSet = std::set<base::FilePath>; | |
61 FilePathSet GetInstallDirContent() const { | |
62 std::set<base::FilePath> content; | |
63 base::FileEnumerator file_enumerator( | |
64 install_dir(), true, | |
65 base::FileEnumerator::FILES | base::FileEnumerator::DIRECTORIES); | |
66 for (base::FilePath path = file_enumerator.Next(); !path.empty(); | |
67 path = file_enumerator.Next()) { | |
68 DCHECK(base::StartsWith(path.value(), install_dir().value(), | |
69 base::CompareCase::SENSITIVE)); | |
70 content.insert(base::FilePath( | |
71 path.value().substr(install_dir().value().size() + 1))); | |
72 } | |
73 return content; | |
74 } | |
75 | |
76 // Adds to |file_path_set| all files and directories that are expected to be | |
77 // found in the version directory |version| before any attempt to delete it. | |
78 void AddVersionFiles(const base::string16& version, | |
79 FilePathSet* file_path_set) { | |
80 file_path_set->insert(base::FilePath(version)); | |
81 file_path_set->insert(base::FilePath(version).Append(L"chrome.dll")); | |
82 file_path_set->insert(base::FilePath(version).Append(L"nacl64.exe")); | |
83 file_path_set->insert(base::FilePath(version).Append(L"icudtl.dat")); | |
84 file_path_set->insert(base::FilePath(version).Append(L"Installer")); | |
85 file_path_set->insert( | |
86 base::FilePath(version).Append(L"Installer\\setup.exe")); | |
87 } | |
88 | |
89 base::FilePath install_dir() const { return install_dir_.path(); } | |
90 | |
91 private: | |
92 base::ScopedTempDir install_dir_; | |
93 }; | |
grt (UTC plus 2)
2016/02/24 20:06:17
nit: DISALLOW_COPY_AND_ASSIGN
fdoray
2016/02/25 18:26:48
Done.
| |
94 | |
95 } // namespace | |
96 | |
97 // DeleteOldVersions() should return true when there is nothing to delete. | |
98 TEST_F(DeleteOldVersionsTest, DeleteEmptyInstallDir) { | |
99 ASSERT_TRUE(CreateInstallDir()); | |
100 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
101 } | |
102 | |
103 // An old executable without a matching directory should be deleted. | |
104 TEST_F(DeleteOldVersionsTest, DeleteOldExecutableWithoutMatchingDirectory) { | |
105 ASSERT_TRUE(CreateInstallDir()); | |
106 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
107 | |
108 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
109 EXPECT_TRUE(GetInstallDirContent().empty()); | |
110 } | |
111 | |
112 // DeleteOldVersions() should return false when it fails to delete an old | |
113 // executable that is in use. | |
114 TEST_F(DeleteOldVersionsTest, | |
115 DeleteInUseOldExecutableWithoutMatchingDirectory) { | |
116 ASSERT_TRUE(CreateInstallDir()); | |
117 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
118 | |
119 base::File file_in_use(install_dir().Append(installer::kChromeOldExe), | |
120 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
121 ASSERT_TRUE(file_in_use.IsValid()); | |
122 | |
123 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
124 } | |
125 | |
126 // chrome.exe and new_chrome.exe should never be deleted. | |
127 TEST_F(DeleteOldVersionsTest, DeleteNewExecutablesWithoutMatchingDirectory) { | |
128 ASSERT_TRUE(CreateInstallDir()); | |
129 ASSERT_TRUE(CreateExecutable(installer::kChromeExe, kVersionA)); | |
130 ASSERT_TRUE(CreateExecutable(installer::kChromeNewExe, kVersionB)); | |
131 | |
132 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
133 FilePathSet expected_install_dir_content; | |
134 expected_install_dir_content.insert(base::FilePath(installer::kChromeExe)); | |
135 expected_install_dir_content.insert(base::FilePath(installer::kChromeNewExe)); | |
136 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
137 } | |
138 | |
139 // A directory without a matching executable should be deleted. | |
140 TEST_F(DeleteOldVersionsTest, DeleteDirectoryWithoutMatchingExecutable) { | |
141 ASSERT_TRUE(CreateInstallDir()); | |
142 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
143 | |
144 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
145 EXPECT_TRUE(GetInstallDirContent().empty()); | |
146 } | |
147 | |
148 // DeleteOldVersions() should return false when it fails to delete a file that | |
149 // is in use in a version directory. | |
150 TEST_F(DeleteOldVersionsTest, DeleteInUseDirectoryWithoutMatchingExecutable) { | |
151 ASSERT_TRUE(CreateInstallDir()); | |
152 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
153 | |
154 base::File file_in_use(install_dir().Append(kVersionA).Append(L"icudtl.dat"), | |
155 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
156 ASSERT_TRUE(file_in_use.IsValid()); | |
157 | |
158 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
159 } | |
160 | |
161 // A pair of matching old executable/version directory that is not in use should | |
162 // be deleted. | |
163 TEST_F(DeleteOldVersionsTest, DeleteOldExecutableWithMatchingDirectory) { | |
164 ASSERT_TRUE(CreateInstallDir()); | |
165 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
166 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
167 | |
168 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
169 EXPECT_TRUE(GetInstallDirContent().empty()); | |
170 } | |
171 | |
172 // chrome.exe, new_chrome.exe and their matching version directories should | |
173 // never be deleted. | |
174 TEST_F(DeleteOldVersionsTest, DeleteNewExecutablesWithMatchingDirectory) { | |
175 ASSERT_TRUE(CreateInstallDir()); | |
176 ASSERT_TRUE(CreateExecutable(installer::kChromeExe, kVersionA)); | |
177 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
178 ASSERT_TRUE(CreateExecutable(installer::kChromeNewExe, kVersionB)); | |
179 ASSERT_TRUE(CreateVersionDirectory(kVersionB)); | |
180 | |
181 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
182 | |
183 FilePathSet expected_install_dir_content; | |
184 expected_install_dir_content.insert(base::FilePath(installer::kChromeExe)); | |
185 AddVersionFiles(kVersionA, &expected_install_dir_content); | |
186 expected_install_dir_content.insert(base::FilePath(installer::kChromeNewExe)); | |
187 AddVersionFiles(kVersionB, &expected_install_dir_content); | |
188 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
189 } | |
190 | |
191 // chrome.exe, new_chrome.exe and their matching version directories should | |
192 // never be deleted, even when files named old_chrome*.exe have the same | |
193 // versions as chrome.exe/new_chrome.exe. The old_chrome*.exe files, however, | |
194 // should be deleted. | |
195 TEST_F(DeleteOldVersionsTest, | |
196 DeleteNewExecutablesWithMatchingDirectoryAndOldExecutables) { | |
197 ASSERT_TRUE(CreateInstallDir()); | |
198 ASSERT_TRUE(CreateExecutable(installer::kChromeExe, kVersionA)); | |
199 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
200 ASSERT_TRUE(CreateExecutable(installer::kChromeNewExe, kVersionB)); | |
201 ASSERT_TRUE(CreateVersionDirectory(kVersionB)); | |
202 ASSERT_TRUE(CreateExecutable(L"old_chrome.exe", kVersionA)); | |
203 ASSERT_TRUE(CreateExecutable(L"old_chrome2.exe", kVersionB)); | |
204 | |
205 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
206 | |
207 FilePathSet expected_install_dir_content; | |
208 expected_install_dir_content.insert(base::FilePath(installer::kChromeExe)); | |
209 AddVersionFiles(kVersionA, &expected_install_dir_content); | |
210 expected_install_dir_content.insert(base::FilePath(installer::kChromeNewExe)); | |
211 AddVersionFiles(kVersionB, &expected_install_dir_content); | |
212 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
213 } | |
214 | |
215 // No file should be deleted for a given version if the executable is in use. | |
216 TEST_F(DeleteOldVersionsTest, DeleteVersionWithExecutableInUse) { | |
217 ASSERT_TRUE(CreateInstallDir()); | |
218 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
219 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
220 | |
221 base::File file_in_use(install_dir().Append(installer::kChromeOldExe), | |
222 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
223 ASSERT_TRUE(file_in_use.IsValid()); | |
224 | |
225 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
226 | |
227 FilePathSet expected_install_dir_content; | |
228 expected_install_dir_content.insert(base::FilePath(installer::kChromeOldExe)); | |
229 AddVersionFiles(kVersionA, &expected_install_dir_content); | |
230 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
231 } | |
232 | |
233 // No file should be deleted for a given version if a .dll file in the version | |
234 // directory is in use. | |
235 TEST_F(DeleteOldVersionsTest, DeleteVersionWithVersionDirectoryDllInUse) { | |
236 ASSERT_TRUE(CreateInstallDir()); | |
237 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
238 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
239 | |
240 base::File file_in_use(install_dir().Append(kVersionA).Append(L"chrome.dll"), | |
241 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
242 ASSERT_TRUE(file_in_use.IsValid()); | |
243 | |
244 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
245 | |
246 FilePathSet expected_install_dir_content; | |
247 expected_install_dir_content.insert(base::FilePath(installer::kChromeOldExe)); | |
248 AddVersionFiles(kVersionA, &expected_install_dir_content); | |
249 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
250 } | |
251 | |
252 // No file should be deleted for a given version if a .exe file in the version | |
253 // directory is in use. | |
254 TEST_F(DeleteOldVersionsTest, DeleteVersionWithVersionDirectoryExeInUse) { | |
255 ASSERT_TRUE(CreateInstallDir()); | |
256 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
257 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
258 | |
259 base::File file_in_use( | |
260 install_dir().Append(kVersionA).Append(L"Installer\\setup.exe"), | |
261 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
262 ASSERT_TRUE(file_in_use.IsValid()); | |
263 | |
264 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
265 | |
266 FilePathSet expected_install_dir_content; | |
267 expected_install_dir_content.insert(base::FilePath(installer::kChromeOldExe)); | |
268 AddVersionFiles(kVersionA, &expected_install_dir_content); | |
269 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
270 } | |
271 | |
272 // DeleteOldVersions() should return false when it tries to delete all the files | |
273 // of a version but one of them can't be deleted because it is in use. | |
274 TEST_F(DeleteOldVersionsTest, DeleteVersionWithUnimportantFileInUse) { | |
275 ASSERT_TRUE(CreateInstallDir()); | |
276 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
277 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
278 | |
279 base::File file_in_use(install_dir().Append(kVersionA).Append(L"icudtl.dat"), | |
280 base::File::FLAG_OPEN | base::File::FLAG_READ); | |
281 ASSERT_TRUE(file_in_use.IsValid()); | |
282 | |
283 EXPECT_FALSE(DeleteOldVersions(install_dir())); | |
284 } | |
285 | |
286 // If an installation directory contains a file named chrome.exe with a matching | |
287 // directory v1 and a file name old_chrome.exe with a matching directory v2, | |
288 // old_chrome.exe and v2 should be deleted but chrome.exe and v1 shouldn't. | |
289 TEST_F(DeleteOldVersionsTest, TypicalAfterRenameState) { | |
290 ASSERT_TRUE(CreateInstallDir()); | |
291 ASSERT_TRUE(CreateExecutable(installer::kChromeOldExe, kVersionA)); | |
292 ASSERT_TRUE(CreateVersionDirectory(kVersionA)); | |
293 ASSERT_TRUE(CreateExecutable(installer::kChromeExe, kVersionB)); | |
294 ASSERT_TRUE(CreateVersionDirectory(kVersionB)); | |
295 | |
296 EXPECT_TRUE(DeleteOldVersions(install_dir())); | |
297 | |
298 FilePathSet expected_install_dir_content; | |
299 expected_install_dir_content.insert(base::FilePath(installer::kChromeExe)); | |
300 AddVersionFiles(kVersionB, &expected_install_dir_content); | |
301 EXPECT_EQ(expected_install_dir_content, GetInstallDirContent()); | |
302 } | |
OLD | NEW |