chromium/chrome/browser/ash/fileapi/external_file_url_util.cc

// 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.

#include "chrome/browser/ash/fileapi/external_file_url_util.h"

#include <string>
#include <vector>

#include "base/strings/escape.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "chrome/browser/ash/file_manager/app_id.h"
#include "chrome/browser/ash/file_manager/fileapi_util.h"
#include "chrome/browser/ash/file_manager/path_util.h"
#include "chrome/browser/ash/fusebox/fusebox_server.h"
#include "chrome/browser/extensions/extension_util.h"
#include "content/public/browser/browser_context.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/common/url_constants.h"
#include "net/base/filename_util.h"
#include "storage/browser/file_system/file_system_context.h"
#include "storage/browser/file_system/file_system_url.h"

using content::BrowserThread;

namespace ash {

bool IsExternalFileURLType(storage::FileSystemType type, bool force) {
  return type == storage::kFileSystemTypeDeviceMediaAsFileStorage ||
         type == storage::kFileSystemTypeProvided ||
         type == storage::kFileSystemTypeArcContent ||
         type == storage::kFileSystemTypeArcDocumentsProvider || force;
}

GURL FileSystemURLToExternalFileURL(
    const storage::FileSystemURL& file_system_url,
    bool force) {
  if (file_system_url.mount_type() != storage::kFileSystemTypeExternal ||
      !IsExternalFileURLType(file_system_url.type(), force)) {
    return GURL();
  }

  return VirtualPathToExternalFileURL(file_system_url.virtual_path());
}

base::FilePath ExternalFileURLToVirtualPath(const GURL& url) {
  if (!url.is_valid() || url.scheme() != content::kExternalFileScheme)
    return base::FilePath();
  return base::FilePath::FromUTF8Unsafe(
      base::UnescapeBinaryURLComponent(url.path_piece()));
}

GURL VirtualPathToExternalFileURL(const base::FilePath& virtual_path) {
  return GURL(base::StringPrintf(
      "%s:%s", content::kExternalFileScheme,
      base::EscapePath(virtual_path.AsUTF8Unsafe()).c_str()));
}

GURL CreateExternalFileURLFromPath(content::BrowserContext* browser_context,
                                   const base::FilePath& path,
                                   bool force) {
  DCHECK_CURRENTLY_ON(BrowserThread::UI);

  GURL raw_file_system_url;
  if (!file_manager::util::ConvertAbsoluteFilePathToFileSystemUrl(
          browser_context, path, file_manager::util::GetFileManagerURL(),
          &raw_file_system_url)) {
    return GURL();
  }

  const storage::FileSystemURL file_system_url =
      file_manager::util::GetFileManagerFileSystemContext(browser_context)
          ->CrackURLInFirstPartyContext(raw_file_system_url);
  if (!file_system_url.is_valid())
    return GURL();

  return FileSystemURLToExternalFileURL(file_system_url, force);
}

GURL ExternalFileURLToFuseboxMonikerFileURL(
    content::BrowserContext* browser_context,
    const GURL& url,
    bool read_only,
    base::TimeDelta lifetime) {
  const base::FilePath virtual_path = ExternalFileURLToVirtualPath(url);

  const storage::FileSystemURL fs_url =
      file_manager::util::GetFileManagerFileSystemContext(browser_context)
          ->CreateCrackedFileSystemURL(
              blink::StorageKey::CreateFirstParty(
                  file_manager::util::GetFilesAppOrigin()),
              storage::kFileSystemTypeExternal, virtual_path);
  if (!fs_url.is_valid()) {
    return GURL();
  }

  fusebox::Server* fusebox_server = fusebox::Server::GetInstance();
  if (!fusebox_server) {
    return GURL();
  }

  fusebox::Moniker moniker = fusebox_server->CreateMoniker(fs_url, read_only);

  // Keep the Moniker alive for the lifetime. We could be cleverer about
  // scheduling the clean up, but "destroy after a fixed amount of time" is
  // simple and works well enough in practice.
  base::SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
      FROM_HERE,
      base::BindOnce(
          [](fusebox::Moniker moniker) {
            fusebox::Server* fusebox_server = fusebox::Server::GetInstance();
            if (fusebox_server) {
              fusebox_server->DestroyMoniker(moniker);
            }
          },
          moniker),
      lifetime);

  base::FilePath moniker_path(fusebox::MonikerMap::GetFilename(moniker));
  return net::FilePathToFileURL(
      moniker_path.AddExtension(virtual_path.Extension()));
}

}  // namespace ash