| // Copyright 2014 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifdef UNSAFE_BUFFERS_BUILD |
| // TODO(crbug.com/40285824): Remove this and convert code to safer constructs. |
| #pragma allow_unsafe_buffers |
| #endif |
| |
| #include "chrome/browser/ash/file_system_provider/mount_path_util.h" |
| |
| #include <stddef.h> |
| |
| #include <vector> |
| |
| #include "base/strings/string_util.h" |
| #include "base/strings/stringprintf.h" |
| #include "chrome/browser/ash/file_manager/path_util.h" |
| #include "chrome/browser/ash/file_system_provider/provided_file_system.h" |
| #include "chrome/browser/ash/file_system_provider/provided_file_system_info.h" |
| #include "chrome/browser/ash/file_system_provider/service.h" |
| #include "chrome/browser/ash/profiles/profile_helper.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/profiles/profile_manager.h" |
| #include "components/user_manager/user.h" |
| #include "components/user_manager/user_manager.h" |
| #include "content/public/browser/browser_thread.h" |
| |
| using content::BrowserThread; |
| |
| namespace ash::file_system_provider::util { |
| |
| namespace { |
| |
| // Root mount path for all of the provided file systems. |
| const base::FilePath::CharType kProvidedMountPointRoot[] = |
| FILE_PATH_LITERAL("/provided"); |
| |
| } // namespace |
| |
| // Escapes the file system id so it can be used as a file/directory name. |
| // This is based on net/base/escape.cc: net::(anonymous namespace)::Escape |
| std::string EscapeFileSystemId(const std::string& file_system_id) { |
| std::string escaped; |
| for (char c : file_system_id) { |
| if (c == '%' || c == '.' || c == '/') { |
| base::StringAppendF(&escaped, "%%%02X", c); |
| } else { |
| escaped.push_back(c); |
| } |
| } |
| return escaped; |
| } |
| |
| base::FilePath GetMountPath(Profile* profile, |
| const ProviderId& provider_id, |
| const std::string& file_system_id) { |
| const user_manager::User* const user = |
| user_manager::UserManager::IsInitialized() |
| ? ProfileHelper::Get()->GetUserByProfile( |
| profile->GetOriginalProfile()) |
| : nullptr; |
| const std::string safe_file_system_id = EscapeFileSystemId(file_system_id); |
| const std::string username_suffix = user ? user->username_hash() : ""; |
| return base::FilePath(kProvidedMountPointRoot) |
| .AppendASCII(provider_id.ToString() + ":" + safe_file_system_id + ":" + |
| username_suffix); |
| } |
| |
| bool IsFileSystemProviderLocalPath(const base::FilePath& local_path) { |
| std::vector<base::FilePath::StringType> components = |
| local_path.GetComponents(); |
| |
| if (components.size() < 3) |
| return false; |
| |
| if (components[0] != FILE_PATH_LITERAL("/")) |
| return false; |
| |
| if (components[1] != kProvidedMountPointRoot + 1 /* no leading slash */) |
| return false; |
| |
| return true; |
| } |
| |
| FileSystemURLParser::FileSystemURLParser(const storage::FileSystemURL& url) |
| : url_(url), file_system_(nullptr) {} |
| |
| FileSystemURLParser::~FileSystemURLParser() = default; |
| |
| bool FileSystemURLParser::Parse() { |
| DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| |
| switch (url_.type()) { |
| case storage::kFileSystemTypeFuseBox: |
| case storage::kFileSystemTypeProvided: |
| break; |
| default: |
| return false; |
| } |
| |
| std::string filesystem_id = url_.filesystem_id(); |
| base::FilePath path = url_.path(); |
| |
| // Convert fusebox URL to its backing (FSP) file system provider URL. |
| if (url_.type() == storage::kFileSystemTypeFuseBox) { |
| const size_t prefix_len = |
| strlen(file_manager::util::kFuseBoxMountNamePrefix) + |
| strlen(file_manager::util::kFuseBoxSubdirPrefixFSP); |
| const std::string& virtual_path = url_.virtual_path().value(); |
| if ((filesystem_id.size() < prefix_len) || |
| (virtual_path.size() < prefix_len)) { |
| return false; |
| } |
| filesystem_id = filesystem_id.substr(prefix_len); |
| path = base::FilePath::FromUTF8Unsafe(base::JoinString( |
| {kProvidedMountPointRoot, virtual_path.substr(prefix_len)}, "/")); |
| } |
| |
| // Find the service that handles the provider URL mount point. |
| for (Profile* profile : |
| g_browser_process->profile_manager()->GetLoadedProfiles()) { |
| Profile* original_profile = profile->GetOriginalProfile(); |
| |
| if (original_profile != profile || |
| !ProfileHelper::IsUserProfile(original_profile)) { |
| continue; |
| } |
| |
| Service* const service = Service::Get(original_profile); |
| if (!service) |
| continue; |
| |
| ProvidedFileSystemInterface* const file_system = |
| service->GetProvidedFileSystem(filesystem_id); |
| if (!file_system) |
| continue; |
| |
| // Strip the mount path name from the local path, to extract the file path |
| // within the provided file system. |
| file_system_ = file_system; |
| std::vector<base::FilePath::StringType> components = path.GetComponents(); |
| if (components.size() < 3) |
| return false; |
| |
| file_path_ = base::FilePath(FILE_PATH_LITERAL("/")); |
| for (size_t i = 3; i < components.size(); ++i) { |
| file_path_ = file_path_.Append(components[i]); |
| } |
| |
| return true; |
| } |
| |
| // Nothing has been found. |
| return false; |
| } |
| |
| LocalPathParser::LocalPathParser(Profile* profile, |
| const base::FilePath& local_path) |
| : profile_(profile), local_path_(local_path), file_system_(nullptr) {} |
| |
| LocalPathParser::~LocalPathParser() = default; |
| |
| bool LocalPathParser::Parse() { |
| DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| |
| if (!IsFileSystemProviderLocalPath(local_path_)) |
| return false; |
| |
| std::vector<base::FilePath::StringType> components = |
| local_path_.GetComponents(); |
| |
| if (components.size() < 3) |
| return false; |
| |
| const std::string mount_point_name = components[2]; |
| |
| Service* const service = Service::Get(profile_); |
| if (!service) |
| return false; |
| |
| ProvidedFileSystemInterface* const file_system = |
| service->GetProvidedFileSystem(mount_point_name); |
| if (!file_system) |
| return false; |
| |
| // Strip the mount point path from the virtual path, to extract the file |
| // path within the provided file system. |
| file_system_ = file_system; |
| file_path_ = base::FilePath(FILE_PATH_LITERAL("/")); |
| for (size_t i = 3; i < components.size(); ++i) { |
| file_path_ = file_path_.Append(components[i]); |
| } |
| |
| return true; |
| } |
| |
| } // namespace ash::file_system_provider::util |