// Copyright 2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/files/file_path.h" #include #include "base/logging.h" #include "base/stl_util.h" namespace base { #if defined(FILE_PATH_USES_WIN_SEPARATORS) const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("\\/"); #else // FILE_PATH_USES_WIN_SEPARATORS const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("/"); #endif // FILE_PATH_USES_WIN_SEPARATORS const FilePath::CharType FilePath::kCurrentDirectory[] = FILE_PATH_LITERAL("."); const FilePath::CharType FilePath::kParentDirectory[] = FILE_PATH_LITERAL(".."); const FilePath::CharType FilePath::kExtensionSeparator = FILE_PATH_LITERAL('.'); typedef FilePath::StringType StringType; namespace { const FilePath::CharType kStringTerminator = FILE_PATH_LITERAL('\0'); // If this FilePath contains a drive letter specification, returns the // position of the last character of the drive letter specification, // otherwise returns npos. This can only be true on Windows, when a pathname // begins with a letter followed by a colon. On other platforms, this always // returns npos. StringType::size_type FindDriveLetter(const StringType& path) { #if defined(FILE_PATH_USES_DRIVE_LETTERS) // This is dependent on an ASCII-based character set, but that's a // reasonable assumption. iswalpha can be too inclusive here. if (path.length() >= 2 && path[1] == L':' && ((path[0] >= L'A' && path[0] <= L'Z') || (path[0] >= L'a' && path[0] <= L'z'))) { return 1; } #endif // FILE_PATH_USES_DRIVE_LETTERS return StringType::npos; } #if defined(FILE_PATH_USES_DRIVE_LETTERS) bool EqualDriveLetterCaseInsensitive(const StringType& a, const StringType& b) { size_t a_letter_pos = FindDriveLetter(a); size_t b_letter_pos = FindDriveLetter(b); if (a_letter_pos == StringType::npos || b_letter_pos == StringType::npos) return a == b; if (::tolower(a[0]) != ::tolower(b[0])) return false; StringType a_rest(a.substr(a_letter_pos + 1)); StringType b_rest(b.substr(b_letter_pos + 1)); return a_rest == b_rest; } #endif // defined(FILE_PATH_USES_DRIVE_LETTERS) bool IsPathAbsolute(const StringType& path) { #if defined(FILE_PATH_USES_DRIVE_LETTERS) StringType::size_type letter = FindDriveLetter(path); if (letter != StringType::npos) { // Look for a separator right after the drive specification. return path.length() > letter + 1 && FilePath::IsSeparator(path[letter + 1]); } // Look for a pair of leading separators. return path.length() > 1 && FilePath::IsSeparator(path[0]) && FilePath::IsSeparator(path[1]); #else // FILE_PATH_USES_DRIVE_LETTERS // Look for a separator in the first position. return path.length() > 0 && FilePath::IsSeparator(path[0]); #endif // FILE_PATH_USES_DRIVE_LETTERS } } // namespace FilePath::FilePath() { } FilePath::FilePath(const FilePath& that) : path_(that.path_) { } FilePath::FilePath(const StringType& path) : path_(path) { StringType::size_type nul_pos = path_.find(kStringTerminator); if (nul_pos != StringType::npos) path_.erase(nul_pos, StringType::npos); } FilePath::~FilePath() { } FilePath& FilePath::operator=(const FilePath& that) { path_ = that.path_; return *this; } bool FilePath::operator==(const FilePath& that) const { #if defined(FILE_PATH_USES_DRIVE_LETTERS) return EqualDriveLetterCaseInsensitive(this->path_, that.path_); #else // defined(FILE_PATH_USES_DRIVE_LETTERS) return path_ == that.path_; #endif // defined(FILE_PATH_USES_DRIVE_LETTERS) } bool FilePath::operator!=(const FilePath& that) const { #if defined(FILE_PATH_USES_DRIVE_LETTERS) return !EqualDriveLetterCaseInsensitive(this->path_, that.path_); #else // defined(FILE_PATH_USES_DRIVE_LETTERS) return path_ != that.path_; #endif // defined(FILE_PATH_USES_DRIVE_LETTERS) } // static bool FilePath::IsSeparator(CharType character) { for (size_t i = 0; i < size(FilePath::kSeparators) - 1; ++i) { if (character == kSeparators[i]) { return true; } } return false; } // libgen's dirname and basename aren't guaranteed to be thread-safe and aren't // guaranteed to not modify their input strings, and in fact are implemented // differently in this regard on different platforms. Don't use them, but // adhere to their behavior. FilePath FilePath::DirName() const { FilePath new_path(path_); new_path.StripTrailingSeparatorsInternal(); // The drive letter, if any, always needs to remain in the output. If there // is no drive letter, as will always be the case on platforms which do not // support drive letters, letter will be npos, or -1, so the comparisons and // resizes below using letter will still be valid. StringType::size_type letter = FindDriveLetter(new_path.path_); StringType::size_type last_separator = new_path.path_.find_last_of(kSeparators, StringType::npos, size(kSeparators) - 1); if (last_separator == StringType::npos) { // path_ is in the current directory. new_path.path_.resize(letter + 1); } else if (last_separator == letter + 1) { // path_ is in the root directory. new_path.path_.resize(letter + 2); } else if (last_separator == letter + 2 && IsSeparator(new_path.path_[letter + 1])) { // path_ is in "//" (possibly with a drive letter); leave the double // separator intact indicating alternate root. new_path.path_.resize(letter + 3); } else if (last_separator != 0) { // path_ is somewhere else, trim the basename. new_path.path_.resize(last_separator); } new_path.StripTrailingSeparatorsInternal(); if (!new_path.path_.length()) new_path.path_ = kCurrentDirectory; return new_path; } FilePath FilePath::BaseName() const { FilePath new_path(path_); new_path.StripTrailingSeparatorsInternal(); // The drive letter, if any, is always stripped. StringType::size_type letter = FindDriveLetter(new_path.path_); if (letter != StringType::npos) { new_path.path_.erase(0, letter + 1); } // Keep everything after the final separator, but if the pathname is only // one character and it's a separator, leave it alone. StringType::size_type last_separator = new_path.path_.find_last_of(kSeparators, StringType::npos, size(kSeparators) - 1); if (last_separator != StringType::npos && last_separator < new_path.path_.length() - 1) { new_path.path_.erase(0, last_separator + 1); } return new_path; } StringType FilePath::FinalExtension() const { StringType base(BaseName().value()); // Special case "." and ".." if (base == FilePath::kCurrentDirectory || base == FilePath::kParentDirectory) return StringType(); const StringType::size_type dot = base.rfind(FilePath::kExtensionSeparator); if (dot == StringType::npos) return StringType(); return base.substr(dot, StringType::npos); } FilePath FilePath::RemoveFinalExtension() const { StringType extension = FinalExtension(); if (FinalExtension().empty()) return *this; return FilePath(path_.substr(0, path_.size() - extension.size())); } FilePath FilePath::Append(const StringType& component) const { const StringType* appended = &component; StringType without_nuls; StringType::size_type nul_pos = component.find(kStringTerminator); if (nul_pos != StringType::npos) { without_nuls = component.substr(0, nul_pos); appended = &without_nuls; } DCHECK(!IsPathAbsolute(*appended)); if (path_.compare(kCurrentDirectory) == 0) { // Append normally doesn't do any normalization, but as a special case, // when appending to kCurrentDirectory, just return a new path for the // component argument. Appending component to kCurrentDirectory would // serve no purpose other than needlessly lengthening the path, and // it's likely in practice to wind up with FilePath objects containing // only kCurrentDirectory when calling DirName on a single relative path // component. return FilePath(*appended); } FilePath new_path(path_); new_path.StripTrailingSeparatorsInternal(); // Don't append a separator if the path is empty (indicating the current // directory) or if the path component is empty (indicating nothing to // append). if (appended->length() > 0 && new_path.path_.length() > 0) { // Don't append a separator if the path still ends with a trailing // separator after stripping (indicating the root directory). if (!IsSeparator(new_path.path_[new_path.path_.length() - 1])) { // Don't append a separator if the path is just a drive letter. if (FindDriveLetter(new_path.path_) + 1 != new_path.path_.length()) { new_path.path_.append(1, kSeparators[0]); } } } new_path.path_.append(*appended); return new_path; } FilePath FilePath::Append(const FilePath& component) const { return Append(component.value()); } bool FilePath::IsAbsolute() const { return IsPathAbsolute(path_); } void FilePath::StripTrailingSeparatorsInternal() { // If there is no drive letter, start will be 1, which will prevent stripping // the leading separator if there is only one separator. If there is a drive // letter, start will be set appropriately to prevent stripping the first // separator following the drive letter, if a separator immediately follows // the drive letter. StringType::size_type start = FindDriveLetter(path_) + 2; StringType::size_type last_stripped = StringType::npos; for (StringType::size_type pos = path_.length(); pos > start && IsSeparator(path_[pos - 1]); --pos) { // If the string only has two separators and they're at the beginning, // don't strip them, unless the string began with more than two separators. if (pos != start + 1 || last_stripped == start + 2 || !IsSeparator(path_[start - 1])) { path_.resize(pos - 1); last_stripped = pos; } } } } // namespace base void PrintTo(const base::FilePath& path, std::ostream* out) { *out << path.value().c_str(); }