ipc/chromium/src/base/file_path.cc

Wed, 31 Dec 2014 06:09:35 +0100

author
Michael Schloh von Bennewitz <michael@schloh.com>
date
Wed, 31 Dec 2014 06:09:35 +0100
changeset 0
6474c204b198
permissions
-rw-r--r--

Cloned upstream origin tor-browser at tor-browser-31.3.0esr-4.5-1-build1
revision ID fc1c9ff7c1b2defdbc039f12214767608f46423f for hacking purpose.

     1 // Copyright (c) 2008 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.
     5 #include <fstream>
     7 #include "base/file_path.h"
     8 #include "base/logging.h"
    10 // These includes are just for the *Hack functions, and should be removed
    11 // when those functions are removed.
    12 #include "base/string_piece.h"
    13 #include "base/string_util.h"
    14 #include "base/sys_string_conversions.h"
    16 #if defined(FILE_PATH_USES_WIN_SEPARATORS)
    17 const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("\\/");
    18 #else  // FILE_PATH_USES_WIN_SEPARATORS
    19 const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("/");
    20 #endif  // FILE_PATH_USES_WIN_SEPARATORS
    22 const FilePath::CharType FilePath::kCurrentDirectory[] = FILE_PATH_LITERAL(".");
    23 const FilePath::CharType FilePath::kParentDirectory[] = FILE_PATH_LITERAL("..");
    25 const FilePath::CharType FilePath::kExtensionSeparator = FILE_PATH_LITERAL('.');
    28 namespace {
    30 // If this FilePath contains a drive letter specification, returns the
    31 // position of the last character of the drive letter specification,
    32 // otherwise returns npos.  This can only be true on Windows, when a pathname
    33 // begins with a letter followed by a colon.  On other platforms, this always
    34 // returns npos.
    35 FilePath::StringType::size_type FindDriveLetter(
    36     const FilePath::StringType& path) {
    37 #if defined(FILE_PATH_USES_DRIVE_LETTERS)
    38   // This is dependent on an ASCII-based character set, but that's a
    39   // reasonable assumption.  iswalpha can be too inclusive here.
    40   if (path.length() >= 2 && path[1] == L':' &&
    41       ((path[0] >= L'A' && path[0] <= L'Z') ||
    42        (path[0] >= L'a' && path[0] <= L'z'))) {
    43     return 1;
    44   }
    45 #endif  // FILE_PATH_USES_DRIVE_LETTERS
    46   return FilePath::StringType::npos;
    47 }
    49 bool IsPathAbsolute(const FilePath::StringType& path) {
    50 #if defined(FILE_PATH_USES_DRIVE_LETTERS)
    51   FilePath::StringType::size_type letter = FindDriveLetter(path);
    52   if (letter != FilePath::StringType::npos) {
    53     // Look for a separator right after the drive specification.
    54     return path.length() > letter + 1 &&
    55         FilePath::IsSeparator(path[letter + 1]);
    56   }
    57   // Look for a pair of leading separators.
    58   return path.length() > 1 &&
    59       FilePath::IsSeparator(path[0]) && FilePath::IsSeparator(path[1]);
    60 #else  // FILE_PATH_USES_DRIVE_LETTERS
    61   // Look for a separator in the first position.
    62   return path.length() > 0 && FilePath::IsSeparator(path[0]);
    63 #endif  // FILE_PATH_USES_DRIVE_LETTERS
    64 }
    66 }  // namespace
    68 bool FilePath::IsSeparator(CharType character) {
    69   for (size_t i = 0; i < arraysize(kSeparators) - 1; ++i) {
    70     if (character == kSeparators[i]) {
    71       return true;
    72     }
    73   }
    75   return false;
    76 }
    78 // libgen's dirname and basename aren't guaranteed to be thread-safe and aren't
    79 // guaranteed to not modify their input strings, and in fact are implemented
    80 // differently in this regard on different platforms.  Don't use them, but
    81 // adhere to their behavior.
    82 FilePath FilePath::DirName() const {
    83   FilePath new_path(path_);
    84   new_path.StripTrailingSeparatorsInternal();
    86   // The drive letter, if any, always needs to remain in the output.  If there
    87   // is no drive letter, as will always be the case on platforms which do not
    88   // support drive letters, letter will be npos, or -1, so the comparisons and
    89   // resizes below using letter will still be valid.
    90   StringType::size_type letter = FindDriveLetter(new_path.path_);
    92   StringType::size_type last_separator =
    93       new_path.path_.find_last_of(kSeparators, StringType::npos,
    94                                   arraysize(kSeparators) - 1);
    95   if (last_separator == StringType::npos) {
    96     // path_ is in the current directory.
    97     new_path.path_.resize(letter + 1);
    98   } else if (last_separator == letter + 1) {
    99     // path_ is in the root directory.
   100     new_path.path_.resize(letter + 2);
   101   } else if (last_separator == letter + 2 &&
   102              IsSeparator(new_path.path_[letter + 1])) {
   103     // path_ is in "//" (possibly with a drive letter); leave the double
   104     // separator intact indicating alternate root.
   105     new_path.path_.resize(letter + 3);
   106   } else if (last_separator != 0) {
   107     // path_ is somewhere else, trim the basename.
   108     new_path.path_.resize(last_separator);
   109   }
   111   new_path.StripTrailingSeparatorsInternal();
   112   if (!new_path.path_.length())
   113     new_path.path_ = kCurrentDirectory;
   115   return new_path;
   116 }
   118 FilePath FilePath::BaseName() const {
   119   FilePath new_path(path_);
   120   new_path.StripTrailingSeparatorsInternal();
   122   // The drive letter, if any, is always stripped.
   123   StringType::size_type letter = FindDriveLetter(new_path.path_);
   124   if (letter != StringType::npos) {
   125     new_path.path_.erase(0, letter + 1);
   126   }
   128   // Keep everything after the final separator, but if the pathname is only
   129   // one character and it's a separator, leave it alone.
   130   StringType::size_type last_separator =
   131       new_path.path_.find_last_of(kSeparators, StringType::npos,
   132                                   arraysize(kSeparators) - 1);
   133   if (last_separator != StringType::npos &&
   134       last_separator < new_path.path_.length() - 1) {
   135     new_path.path_.erase(0, last_separator + 1);
   136   }
   138   return new_path;
   139 }
   141 FilePath::StringType FilePath::Extension() const {
   142   // BaseName() calls StripTrailingSeparators, so cases like /foo.baz/// work.
   143   StringType base = BaseName().value();
   145   // Special case "." and ".."
   146   if (base == kCurrentDirectory || base == kParentDirectory)
   147     return StringType();
   149   const StringType::size_type last_dot = base.rfind(kExtensionSeparator);
   150   if (last_dot == StringType::npos)
   151     return StringType();
   152   return StringType(base, last_dot);
   153 }
   155 FilePath FilePath::RemoveExtension() const {
   156   StringType ext = Extension();
   157   // It's important to check Extension() since that verifies that the
   158   // kExtensionSeparator actually appeared in the last path component.
   159   if (ext.empty())
   160     return FilePath(path_);
   161   // Since Extension() verified that the extension is in fact in the last path
   162   // component, this substr will effectively strip trailing separators.
   163   const StringType::size_type last_dot = path_.rfind(kExtensionSeparator);
   164   return FilePath(path_.substr(0, last_dot));
   165 }
   167 FilePath FilePath::InsertBeforeExtension(const StringType& suffix) const {
   168   if (suffix.empty())
   169     return FilePath(path_);
   171   if (path_.empty())
   172     return FilePath();
   174   StringType base = BaseName().value();
   175   if (base.empty())
   176     return FilePath();
   177   if (*(base.end() - 1) == kExtensionSeparator) {
   178     // Special case "." and ".."
   179     if (base == kCurrentDirectory || base == kParentDirectory) {
   180       return FilePath();
   181     }
   182   }
   184   StringType ext = Extension();
   185   StringType ret = RemoveExtension().value();
   186   ret.append(suffix);
   187   ret.append(ext);
   188   return FilePath(ret);
   189 }
   191 FilePath FilePath::ReplaceExtension(const StringType& extension) const {
   192   if (path_.empty())
   193     return FilePath();
   195   StringType base = BaseName().value();
   196   if (base.empty())
   197     return FilePath();
   198   if (*(base.end() - 1) == kExtensionSeparator) {
   199     // Special case "." and ".."
   200     if (base == kCurrentDirectory || base == kParentDirectory) {
   201       return FilePath();
   202     }
   203   }
   205   FilePath no_ext = RemoveExtension();
   206   // If the new extension is "" or ".", then just remove the current extension.
   207   if (extension.empty() || extension == StringType(1, kExtensionSeparator))
   208     return no_ext;
   210   StringType str = no_ext.value();
   211   if (extension[0] != kExtensionSeparator)
   212     str.append(1, kExtensionSeparator);
   213   str.append(extension);
   214   return FilePath(str);
   215 }
   217 FilePath FilePath::Append(const StringType& component) const {
   218   DCHECK(!IsPathAbsolute(component));
   219   if (path_.compare(kCurrentDirectory) == 0) {
   220     // Append normally doesn't do any normalization, but as a special case,
   221     // when appending to kCurrentDirectory, just return a new path for the
   222     // component argument.  Appending component to kCurrentDirectory would
   223     // serve no purpose other than needlessly lengthening the path, and
   224     // it's likely in practice to wind up with FilePath objects containing
   225     // only kCurrentDirectory when calling DirName on a single relative path
   226     // component.
   227     return FilePath(component);
   228   }
   230   FilePath new_path(path_);
   231   new_path.StripTrailingSeparatorsInternal();
   233   // Don't append a separator if the path is empty (indicating the current
   234   // directory) or if the path component is empty (indicating nothing to
   235   // append).
   236   if (component.length() > 0 && new_path.path_.length() > 0) {
   238     // Don't append a separator if the path still ends with a trailing
   239     // separator after stripping (indicating the root directory).
   240     if (!IsSeparator(new_path.path_[new_path.path_.length() - 1])) {
   242       // Don't append a separator if the path is just a drive letter.
   243       if (FindDriveLetter(new_path.path_) + 1 != new_path.path_.length()) {
   244         new_path.path_.append(1, kSeparators[0]);
   245       }
   246     }
   247   }
   249   new_path.path_.append(component);
   250   return new_path;
   251 }
   253 FilePath FilePath::Append(const FilePath& component) const {
   254   return Append(component.value());
   255 }
   257 FilePath FilePath::AppendASCII(const std::string& component) const {
   258   DCHECK(IsStringASCII(component));
   259 #if defined(OS_WIN)
   260   return Append(ASCIIToWide(component));
   261 #elif defined(OS_POSIX)
   262   return Append(component);
   263 #endif
   264 }
   266 bool FilePath::IsAbsolute() const {
   267   return IsPathAbsolute(path_);
   268 }
   270 #if defined(OS_POSIX)
   271 // See file_path.h for a discussion of the encoding of paths on POSIX
   272 // platforms.  These *Hack() functions are not quite correct, but they're
   273 // only temporary while we fix the remainder of the code.
   274 // Remember to remove the #includes at the top when you remove these.
   276 // static
   277 FilePath FilePath::FromWStringHack(const std::wstring& wstring) {
   278   return FilePath(base::SysWideToNativeMB(wstring));
   279 }
   280 std::wstring FilePath::ToWStringHack() const {
   281   return base::SysNativeMBToWide(path_);
   282 }
   283 #elif defined(OS_WIN)
   284 // static
   285 FilePath FilePath::FromWStringHack(const std::wstring& wstring) {
   286   return FilePath(wstring);
   287 }
   288 std::wstring FilePath::ToWStringHack() const {
   289   return path_;
   290 }
   291 #endif
   293 void FilePath::OpenInputStream(std::ifstream& stream) const {
   294   stream.open(
   295 #ifndef __MINGW32__
   296               path_.c_str(),
   297 #else
   298               base::SysWideToNativeMB(path_).c_str(),
   299 #endif
   300               std::ios::in | std::ios::binary);
   301 }
   303 FilePath FilePath::StripTrailingSeparators() const {
   304   FilePath new_path(path_);
   305   new_path.StripTrailingSeparatorsInternal();
   307   return new_path;
   308 }
   310 void FilePath::StripTrailingSeparatorsInternal() {
   311   // If there is no drive letter, start will be 1, which will prevent stripping
   312   // the leading separator if there is only one separator.  If there is a drive
   313   // letter, start will be set appropriately to prevent stripping the first
   314   // separator following the drive letter, if a separator immediately follows
   315   // the drive letter.
   316   StringType::size_type start = FindDriveLetter(path_) + 2;
   318   StringType::size_type last_stripped = StringType::npos;
   319   for (StringType::size_type pos = path_.length();
   320        pos > start && IsSeparator(path_[pos - 1]);
   321        --pos) {
   322     // If the string only has two separators and they're at the beginning,
   323     // don't strip them, unless the string began with more than two separators.
   324     if (pos != start + 1 || last_stripped == start + 2 ||
   325         !IsSeparator(path_[start - 1])) {
   326       path_.resize(pos - 1);
   327       last_stripped = pos;
   328     }
   329   }
   330 }

mercurial