Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 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 | 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 #include "base/file_util.h" | 5 #include "base/file_util.h" |
| 6 | 6 |
| 7 #if defined(OS_WIN) | 7 #if defined(OS_WIN) |
| 8 #include <io.h> | 8 #include <io.h> |
| 9 #endif | 9 #endif |
| 10 #include <stdio.h> | 10 #include <stdio.h> |
| 11 | 11 |
| 12 #include <fstream> | 12 #include <fstream> |
| 13 | 13 |
| 14 #include "base/file_path.h" | 14 #include "base/file_path.h" |
| 15 #include "base/logging.h" | 15 #include "base/logging.h" |
| 16 #include "base/stringprintf.h" | |
| 16 #include "base/string_piece.h" | 17 #include "base/string_piece.h" |
| 17 #include "base/string_util.h" | 18 #include "base/string_util.h" |
| 18 #include "base/utf_string_conversions.h" | 19 #include "base/utf_string_conversions.h" |
| 19 | 20 |
| 20 namespace { | 21 namespace { |
| 21 | 22 |
| 22 const FilePath::CharType kExtensionSeparator = FILE_PATH_LITERAL('.'); | 23 const FilePath::CharType kExtensionSeparator = FILE_PATH_LITERAL('.'); |
| 23 | 24 |
| 25 // The maximum number of 'uniquified' files we will try to create. | |
| 26 // This is used when the filename we're trying to download is already in use, | |
| 27 // so we create a new unique filename by appending " (nnn)" before the | |
| 28 // extension, where 1 <= nnn <= kMaxUniqueFiles. | |
| 29 // Also used by code that cleans up said files. | |
| 30 static const int kMaxUniqueFiles = 100; | |
| 31 | |
| 24 } // namespace | 32 } // namespace |
| 25 | 33 |
| 26 namespace file_util { | 34 namespace file_util { |
| 27 | 35 |
| 28 bool EndsWithSeparator(const FilePath& path) { | 36 bool EndsWithSeparator(const FilePath& path) { |
| 29 FilePath::StringType value = path.value(); | 37 FilePath::StringType value = path.value(); |
| 30 if (value.empty()) | 38 if (value.empty()) |
| 31 return false; | 39 return false; |
| 32 | 40 |
| 33 return FilePath::IsSeparator(value[value.size() - 1]); | 41 return FilePath::IsSeparator(value[value.size() - 1]); |
| (...skipping 196 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 230 if (_chsize(fd, current_offset) != 0) | 238 if (_chsize(fd, current_offset) != 0) |
| 231 return false; | 239 return false; |
| 232 #else | 240 #else |
| 233 int fd = fileno(file); | 241 int fd = fileno(file); |
| 234 if (ftruncate(fd, current_offset) != 0) | 242 if (ftruncate(fd, current_offset) != 0) |
| 235 return false; | 243 return false; |
| 236 #endif | 244 #endif |
| 237 return true; | 245 return true; |
| 238 } | 246 } |
| 239 | 247 |
| 248 int GetUniquePathNumber( | |
|
Evan Martin
2012/01/31 21:23:44
Should this be in the anon namespace?
jam
2012/01/31 21:54:30
no it's in the header, it's used by a few places
| |
| 249 const FilePath& path, | |
| 250 const FilePath::StringType& suffix) { | |
| 251 bool have_suffix = !suffix.empty(); | |
| 252 if (!PathExists(path) && | |
| 253 (!have_suffix || !PathExists(AppendSuffixToPath(path, suffix)))) { | |
| 254 return 0; | |
| 255 } | |
| 256 | |
| 257 FilePath new_path; | |
| 258 for (int count = 1; count <= kMaxUniqueFiles; ++count) { | |
| 259 new_path = FilePath(path); | |
| 260 AppendNumberToPath(&new_path, count); | |
| 261 | |
| 262 if (!PathExists(new_path) && | |
| 263 (!have_suffix || !PathExists(AppendSuffixToPath(new_path, suffix)))) { | |
| 264 return count; | |
| 265 } | |
| 266 } | |
| 267 | |
| 268 return -1; | |
| 269 } | |
| 270 | |
| 240 bool ContainsPath(const FilePath &parent, const FilePath& child) { | 271 bool ContainsPath(const FilePath &parent, const FilePath& child) { |
| 241 FilePath abs_parent = FilePath(parent); | 272 FilePath abs_parent = FilePath(parent); |
| 242 FilePath abs_child = FilePath(child); | 273 FilePath abs_child = FilePath(child); |
| 243 | 274 |
| 244 if (!file_util::AbsolutePath(&abs_parent) || | 275 if (!file_util::AbsolutePath(&abs_parent) || |
| 245 !file_util::AbsolutePath(&abs_child)) | 276 !file_util::AbsolutePath(&abs_child)) |
| 246 return false; | 277 return false; |
| 247 | 278 |
| 248 #if defined(OS_WIN) | 279 #if defined(OS_WIN) |
| 249 // file_util::AbsolutePath() does not flatten case on Windows, so we must do | 280 // file_util::AbsolutePath() does not flatten case on Windows, so we must do |
| 250 // a case-insensitive compare. | 281 // a case-insensitive compare. |
| 251 if (!StartsWith(abs_child.value(), abs_parent.value(), false)) | 282 if (!StartsWith(abs_child.value(), abs_parent.value(), false)) |
| 252 #else | 283 #else |
| 253 if (!StartsWithASCII(abs_child.value(), abs_parent.value(), true)) | 284 if (!StartsWithASCII(abs_child.value(), abs_parent.value(), true)) |
| 254 #endif | 285 #endif |
| 255 return false; | 286 return false; |
| 256 | 287 |
| 257 // file_util::AbsolutePath() normalizes '/' to '\' on Windows, so we only need | 288 // file_util::AbsolutePath() normalizes '/' to '\' on Windows, so we only need |
| 258 // to check kSeparators[0]. | 289 // to check kSeparators[0]. |
| 259 if (abs_child.value().length() <= abs_parent.value().length() || | 290 if (abs_child.value().length() <= abs_parent.value().length() || |
| 260 abs_child.value()[abs_parent.value().length()] != | 291 abs_child.value()[abs_parent.value().length()] != |
| 261 FilePath::kSeparators[0]) | 292 FilePath::kSeparators[0]) |
| 262 return false; | 293 return false; |
| 263 | 294 |
| 264 return true; | 295 return true; |
| 265 } | 296 } |
| 266 | 297 |
| 298 void AppendNumberToPath(FilePath* path, int number) { | |
| 299 *path = path->InsertBeforeExtensionASCII(StringPrintf(" (%d)", number)); | |
| 300 } | |
| 301 | |
| 302 FilePath AppendSuffixToPath( | |
| 303 const FilePath& path, | |
| 304 const FilePath::StringType& suffix) { | |
| 305 FilePath::StringType file_name; | |
| 306 base::SStringPrintf( | |
| 307 &file_name, PRFilePathLiteral PRFilePathLiteral, path.value().c_str(), | |
| 308 suffix.c_str()); | |
|
Evan Martin
2012/01/31 21:23:44
Simpler:
return FilePath(path.value() + suffix)
jam
2012/01/31 21:54:30
good point, done!
also same for AppendNumberToPat
| |
| 309 return FilePath(file_name); | |
| 310 } | |
| 311 | |
| 267 int64 ComputeDirectorySize(const FilePath& root_path) { | 312 int64 ComputeDirectorySize(const FilePath& root_path) { |
| 268 int64 running_size = 0; | 313 int64 running_size = 0; |
| 269 FileEnumerator file_iter(root_path, true, FileEnumerator::FILES); | 314 FileEnumerator file_iter(root_path, true, FileEnumerator::FILES); |
| 270 for (FilePath current = file_iter.Next(); !current.empty(); | 315 for (FilePath current = file_iter.Next(); !current.empty(); |
| 271 current = file_iter.Next()) { | 316 current = file_iter.Next()) { |
| 272 FileEnumerator::FindInfo info; | 317 FileEnumerator::FindInfo info; |
| 273 file_iter.GetFindInfo(&info); | 318 file_iter.GetFindInfo(&info); |
| 274 #if defined(OS_WIN) | 319 #if defined(OS_WIN) |
| 275 LARGE_INTEGER li = { info.nFileSizeLow, info.nFileSizeHigh }; | 320 LARGE_INTEGER li = { info.nFileSizeLow, info.nFileSizeHigh }; |
| 276 running_size += li.QuadPart; | 321 running_size += li.QuadPart; |
| (...skipping 116 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 393 // FileEnumerator | 438 // FileEnumerator |
| 394 // | 439 // |
| 395 // Note: the main logic is in file_util_<platform>.cc | 440 // Note: the main logic is in file_util_<platform>.cc |
| 396 | 441 |
| 397 bool FileEnumerator::ShouldSkip(const FilePath& path) { | 442 bool FileEnumerator::ShouldSkip(const FilePath& path) { |
| 398 FilePath::StringType basename = path.BaseName().value(); | 443 FilePath::StringType basename = path.BaseName().value(); |
| 399 return IsDot(path) || (IsDotDot(path) && !(INCLUDE_DOT_DOT & file_type_)); | 444 return IsDot(path) || (IsDotDot(path) && !(INCLUDE_DOT_DOT & file_type_)); |
| 400 } | 445 } |
| 401 | 446 |
| 402 } // namespace | 447 } // namespace |
| OLD | NEW |