blob: 9e370d6dd84db4c34206d8304f595417e3ae7b0a [file] [log] [blame]
// Copyright (c) 2013 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.
// Browser test for basic Chrome OS file manager functionality:
// - The file list is updated when a file is added externally to the Downloads
// folder.
// - Selecting a file and copy-pasting it with the keyboard copies the file.
// - Selecting a file and pressing delete deletes it.
#include <deque>
#include <string>
#include "base/bind.h"
#include "base/callback.h"
#include "base/file_util.h"
#include "base/files/file_path.h"
#include "base/time/time.h"
#include "chrome/browser/chrome_notification_types.h"
#include "chrome/browser/chromeos/drive/drive_integration_service.h"
#include "chrome/browser/chromeos/drive/file_system_interface.h"
#include "chrome/browser/chromeos/extensions/file_manager/drive_test_util.h"
#include "chrome/browser/drive/fake_drive_service.h"
#include "chrome/browser/extensions/api/test/test_api.h"
#include "chrome/browser/extensions/component_loader.h"
#include "chrome/browser/extensions/extension_apitest.h"
#include "chrome/browser/extensions/extension_test_message_listener.h"
#include "chrome/browser/google_apis/gdata_wapi_parser.h"
#include "chrome/browser/google_apis/test_util.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/extensions/extension.h"
#include "chromeos/chromeos_switches.h"
#include "content/public/browser/browser_context.h"
#include "content/public/browser/notification_service.h"
#include "content/public/test/test_utils.h"
#include "net/test/embedded_test_server/embedded_test_server.h"
#include "webkit/browser/fileapi/external_mount_points.h"
namespace file_manager {
namespace {
enum EntryType {
FILE,
DIRECTORY,
};
enum SharedOption {
NONE,
SHARED,
};
enum GuestMode {
NOT_IN_GUEST_MODE,
IN_GUEST_MODE
};
// This global operator is used from Google Test to format error messages.
std::ostream& operator<<(std::ostream& os, const GuestMode& guest_mode) {
return os << (guest_mode == IN_GUEST_MODE ?
"IN_GUEST_MODE" : "NOT_IN_GUEST_MODE");
}
struct TestEntryInfo {
EntryType type;
const char* source_file_name; // Source file name to be used as a prototype.
const char* target_name; // Target file or directory name.
const char* mime_type;
SharedOption shared_option;
const char* last_modified_time_as_string;
};
TestEntryInfo kTestEntrySetCommon[] = {
{ FILE, "text.txt", "hello.txt", "text/plain", NONE, "4 Sep 1998 12:34:56" },
{ FILE, "image.png", "My Desktop Background.png", "text/plain", NONE,
"18 Jan 2038 01:02:03" },
{ FILE, "music.ogg", "Beautiful Song.ogg", "text/plain", NONE,
"12 Nov 2086 12:00:00" },
{ FILE, "video.ogv", "world.ogv", "text/plain", NONE,
"4 July 2012 10:35:00" },
{ DIRECTORY, "", "photos", NULL, NONE, "1 Jan 1980 23:59:59" },
{ DIRECTORY, "", ".warez", NULL, NONE, "26 Oct 1985 13:39" }
};
TestEntryInfo kTestEntrySetDriveOnly[] = {
{ FILE, "", "Test Document", "application/vnd.google-apps.document", NONE,
"10 Apr 2013 16:20:00" },
{ FILE, "", "Test Shared Document", "application/vnd.google-apps.document",
SHARED, "20 Mar 2013 22:40:00" }
};
// The local volume class for test.
// This class provides the operations for a test volume that simulates local
// drive.
class LocalTestVolume {
public:
// Adds this volume to the file system as a local volume. Returns true on
// success.
bool Mount(Profile* profile) {
const std::string kDownloads = "Downloads";
if (local_path_.empty()) {
if (!tmp_dir_.CreateUniqueTempDir())
return false;
local_path_ = tmp_dir_.path().Append(kDownloads);
}
fileapi::ExternalMountPoints* const mount_points =
content::BrowserContext::GetMountPoints(profile);
mount_points->RevokeFileSystem(kDownloads);
return mount_points->RegisterFileSystem(
kDownloads, fileapi::kFileSystemTypeNativeLocal, local_path_) &&
file_util::CreateDirectory(local_path_);
}
void CreateEntry(const TestEntryInfo& entry) {
base::FilePath target_path = local_path_.AppendASCII(entry.target_name);
switch (entry.type) {
case FILE: {
base::FilePath source_path =
google_apis::test_util::GetTestFilePath("chromeos/file_manager").
AppendASCII(entry.source_file_name);
ASSERT_TRUE(base::CopyFile(source_path, target_path))
<< "Copy from " << source_path.value()
<< " to " << target_path.value() << " failed.";
break;
}
case DIRECTORY:
ASSERT_TRUE(file_util::CreateDirectory(target_path)) <<
"Failed to create a directory: " << target_path.value();
break;
}
base::Time time;
ASSERT_TRUE(base::Time::FromString(entry.last_modified_time_as_string,
&time));
ASSERT_TRUE(file_util::SetLastModifiedTime(target_path, time));
}
private:
base::FilePath local_path_;
base::ScopedTempDir tmp_dir_;
};
// The drive volume class for test.
// This class provides the operations for a test volume that simulates Google
// drive.
class DriveTestVolume {
public:
DriveTestVolume() : fake_drive_service_(NULL),
integration_service_(NULL) {
}
// Send request to add this volume to the file system as Google drive.
// This method must be calld at SetUp method of FileManagerBrowserTestBase.
// Returns true on success.
bool SetUp() {
if (!test_cache_root_.CreateUniqueTempDir())
return false;
drive::DriveIntegrationServiceFactory::SetFactoryForTest(
base::Bind(&DriveTestVolume::CreateDriveIntegrationService,
base::Unretained(this)));
return true;
}
void CreateEntry(const TestEntryInfo& entry) {
switch (entry.type) {
case FILE:
CreateFile(entry.source_file_name,
entry.target_name,
entry.mime_type,
entry.shared_option == SHARED,
entry.last_modified_time_as_string);
break;
case DIRECTORY:
CreateDirectory(entry.target_name, entry.last_modified_time_as_string);
break;
}
}
// Creates an empty directory with the given |name| and |modification_time|.
void CreateDirectory(const std::string& name,
const std::string& modification_time) {
google_apis::GDataErrorCode error = google_apis::GDATA_OTHER_ERROR;
scoped_ptr<google_apis::ResourceEntry> resource_entry;
fake_drive_service_->AddNewDirectory(
fake_drive_service_->GetRootResourceId(),
name,
google_apis::test_util::CreateCopyResultCallback(&error,
&resource_entry));
base::MessageLoop::current()->RunUntilIdle();
ASSERT_TRUE(error == google_apis::HTTP_CREATED);
ASSERT_TRUE(resource_entry);
base::Time time;
ASSERT_TRUE(base::Time::FromString(modification_time.c_str(), &time));
fake_drive_service_->SetLastModifiedTime(
resource_entry->resource_id(),
time,
google_apis::test_util::CreateCopyResultCallback(&error,
&resource_entry));
base::MessageLoop::current()->RunUntilIdle();
ASSERT_TRUE(error == google_apis::HTTP_SUCCESS);
ASSERT_TRUE(resource_entry);
CheckForUpdates();
}
// Creates a test file with the given spec.
// Serves |test_file_name| file. Pass an empty string for an empty file.
void CreateFile(const std::string& source_file_name,
const std::string& target_file_name,
const std::string& mime_type,
bool shared_with_me,
const std::string& modification_time) {
google_apis::GDataErrorCode error = google_apis::GDATA_OTHER_ERROR;
std::string content_data;
if (!source_file_name.empty()) {
base::FilePath source_file_path =
google_apis::test_util::GetTestFilePath("chromeos/file_manager").
AppendASCII(source_file_name);
ASSERT_TRUE(file_util::ReadFileToString(source_file_path, &content_data));
}
scoped_ptr<google_apis::ResourceEntry> resource_entry;
fake_drive_service_->AddNewFile(
mime_type,
content_data,
fake_drive_service_->GetRootResourceId(),
target_file_name,
shared_with_me,
google_apis::test_util::CreateCopyResultCallback(&error,
&resource_entry));
base::MessageLoop::current()->RunUntilIdle();
ASSERT_EQ(google_apis::HTTP_CREATED, error);
ASSERT_TRUE(resource_entry);
base::Time time;
ASSERT_TRUE(base::Time::FromString(modification_time.c_str(), &time));
fake_drive_service_->SetLastModifiedTime(
resource_entry->resource_id(),
time,
google_apis::test_util::CreateCopyResultCallback(&error,
&resource_entry));
base::MessageLoop::current()->RunUntilIdle();
ASSERT_EQ(google_apis::HTTP_SUCCESS, error);
ASSERT_TRUE(resource_entry);
CheckForUpdates();
}
// Notifies FileSystem that the contents in FakeDriveService are
// changed, hence the new contents should be fetched.
void CheckForUpdates() {
if (integration_service_ && integration_service_->file_system()) {
integration_service_->file_system()->CheckForUpdates();
}
}
// Sets the url base for the test server to be used to generate share urls
// on the files and directories.
void ConfigureShareUrlBase(const GURL& share_url_base) {
fake_drive_service_->set_share_url_base(share_url_base);
}
drive::DriveIntegrationService* CreateDriveIntegrationService(
Profile* profile) {
fake_drive_service_ = new drive::FakeDriveService;
fake_drive_service_->LoadResourceListForWapi(
"gdata/empty_feed.json");
fake_drive_service_->LoadAccountMetadataForWapi(
"gdata/account_metadata.json");
fake_drive_service_->LoadAppListForDriveApi("drive/applist.json");
integration_service_ = new drive::DriveIntegrationService(
profile,
fake_drive_service_,
test_cache_root_.path(),
NULL);
return integration_service_;
}
private:
base::ScopedTempDir test_cache_root_;
drive::FakeDriveService* fake_drive_service_;
drive::DriveIntegrationService* integration_service_;
};
// Listener to obtain the test relative messages synchronously.
class FileManagerTestListener : public content::NotificationObserver {
public:
struct Message {
int type;
std::string message;
extensions::TestSendMessageFunction* function;
};
FileManagerTestListener() {
registrar_.Add(this,
chrome::NOTIFICATION_EXTENSION_TEST_PASSED,
content::NotificationService::AllSources());
registrar_.Add(this,
chrome::NOTIFICATION_EXTENSION_TEST_FAILED,
content::NotificationService::AllSources());
registrar_.Add(this,
chrome::NOTIFICATION_EXTENSION_TEST_MESSAGE,
content::NotificationService::AllSources());
}
Message GetNextMessage() {
if (messages_.empty())
content::RunMessageLoop();
const Message entry = messages_.front();
messages_.pop_front();
return entry;
}
virtual void Observe(int type,
const content::NotificationSource& source,
const content::NotificationDetails& details) OVERRIDE {
Message entry;
entry.type = type;
entry.message = type != chrome::NOTIFICATION_EXTENSION_TEST_PASSED ?
*content::Details<std::string>(details).ptr() :
std::string();
entry.function = type == chrome::NOTIFICATION_EXTENSION_TEST_MESSAGE ?
content::Source<extensions::TestSendMessageFunction>(source).ptr() :
NULL;
messages_.push_back(entry);
base::MessageLoopForUI::current()->Quit();
}
private:
std::deque<Message> messages_;
content::NotificationRegistrar registrar_;
};
// Parameter of FileManagerBrowserTest.
// The second value is the case name of JavaScript.
typedef std::tr1::tuple<GuestMode, const char*> TestParameter;
// The base test class.
class FileManagerBrowserTest :
public ExtensionApiTest,
public ::testing::WithParamInterface<TestParameter> {
protected:
FileManagerBrowserTest() :
local_volume_(new LocalTestVolume),
drive_volume_(std::tr1::get<0>(GetParam()) != IN_GUEST_MODE ?
new DriveTestVolume() : NULL) {}
virtual void SetUp() OVERRIDE {
// TODO(danakj): The GPU Video Decoder needs real GL bindings.
// crbug.com/269087
UseRealGLBindings();
ExtensionApiTest::SetUp();
}
virtual void SetUpInProcessBrowserTestFixture() OVERRIDE;
virtual void SetUpOnMainThread() OVERRIDE;
// Adds an incognito and guest-mode flags for tests in the guest mode.
virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE;
// Loads our testing extension and sends it a string identifying the current
// test.
void StartTest();
const scoped_ptr<LocalTestVolume> local_volume_;
const scoped_ptr<DriveTestVolume> drive_volume_;
};
void FileManagerBrowserTest::SetUpInProcessBrowserTestFixture() {
ExtensionApiTest::SetUpInProcessBrowserTestFixture();
extensions::ComponentLoader::EnableBackgroundExtensionsForTesting();
if (drive_volume_)
ASSERT_TRUE(drive_volume_->SetUp());
}
void FileManagerBrowserTest::SetUpOnMainThread() {
ExtensionApiTest::SetUpOnMainThread();
ASSERT_TRUE(local_volume_->Mount(browser()->profile()));
for (size_t i = 0; i < arraysize(kTestEntrySetCommon); ++i)
local_volume_->CreateEntry(kTestEntrySetCommon[i]);
if (drive_volume_) {
// Install the web server to serve the mocked share dialog.
ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady());
const GURL share_url_base(embedded_test_server()->GetURL(
"/chromeos/file_manager/share_dialog_mock/index.html"));
drive_volume_->ConfigureShareUrlBase(share_url_base);
for (size_t i = 0; i < arraysize(kTestEntrySetCommon); ++i)
drive_volume_->CreateEntry(kTestEntrySetCommon[i]);
// For testing Drive, create more entries with Drive specific attributes.
// TODO(haruki): Add a case for an entry cached by DriveCache.
for (size_t i = 0; i < arraysize(kTestEntrySetDriveOnly); ++i)
drive_volume_->CreateEntry(kTestEntrySetDriveOnly[i]);
test_util::WaitUntilDriveMountPointIsAdded(browser()->profile());
}
}
void FileManagerBrowserTest::SetUpCommandLine(CommandLine* command_line) {
if (std::tr1::get<0>(GetParam()) == IN_GUEST_MODE) {
command_line->AppendSwitch(chromeos::switches::kGuestSession);
command_line->AppendSwitchNative(chromeos::switches::kLoginUser, "");
command_line->AppendSwitch(switches::kIncognito);
}
ExtensionApiTest::SetUpCommandLine(command_line);
}
IN_PROC_BROWSER_TEST_P(FileManagerBrowserTest, Test) {
// Launch the extension.
base::FilePath path = test_data_dir_.AppendASCII("file_manager_browsertest");
const extensions::Extension* extension = LoadExtensionAsComponent(path);
ASSERT_TRUE(extension);
// Handle the messages from JavaScript.
// The while loop is break when the test is passed or failed.
FileManagerTestListener listener;
while (true) {
FileManagerTestListener::Message entry = listener.GetNextMessage();
if (entry.type == chrome::NOTIFICATION_EXTENSION_TEST_PASSED) {
// Test succeed.
break;
} else if (entry.type == chrome::NOTIFICATION_EXTENSION_TEST_FAILED) {
// Test failed.
ADD_FAILURE() << entry.message;
break;
} else if (entry.message == "getTestName") {
// Pass the test case name.
entry.function->Reply(std::tr1::get<1>(GetParam()));
} else if (entry.message == "isInGuestMode") {
// Obtains whther the test is in guest mode or not.
entry.function->Reply(std::tr1::get<0>(GetParam()) ? "true" : "false");
} else if (entry.message == "addEntry") {
// Add the extra entry.
const TestEntryInfo file = {
FILE,
"music.ogg", // Prototype file name.
"newly added file.ogg", // Target file name.
"audio/ogg",
NONE,
"4 Sep 1998 00:00:00"
};
if (drive_volume_)
drive_volume_->CreateEntry(file);
local_volume_->CreateEntry(file);
entry.function->Reply("onEntryAdded");
}
}
}
INSTANTIATE_TEST_CASE_P(
FileDisplay,
FileManagerBrowserTest,
::testing::Values(TestParameter(NOT_IN_GUEST_MODE, "fileDisplayDownloads"),
TestParameter(IN_GUEST_MODE, "fileDisplayDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "fileDisplayDrive")));
// TODO(mtomasz): Fix this test. crbug.com/252561
/*
INSTANTIATE_TEST_CASE_P(
OpenSpecialTypes,
FileManagerBrowserTest,
::testing::Values(TestParameter(IN_GUEST_MODE, "videoOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "videoOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "videoOpenDrive"),
TestParameter(IN_GUEST_MODE, "audioOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "audioOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "audioOpenDrive"),
TestParameter(IN_GUEST_MODE, "galleryOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"galleryOpenDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "galleryOpenDrive")));
*/
INSTANTIATE_TEST_CASE_P(
KeyboardOperations,
FileManagerBrowserTest,
::testing::Values(TestParameter(IN_GUEST_MODE, "keyboardDeleteDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"keyboardDeleteDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "keyboardDeleteDrive"),
TestParameter(IN_GUEST_MODE, "keyboardCopyDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "keyboardCopyDownloads"),
TestParameter(NOT_IN_GUEST_MODE, "keyboardCopyDrive")));
INSTANTIATE_TEST_CASE_P(
DriveSpecific,
FileManagerBrowserTest,
::testing::Values(TestParameter(NOT_IN_GUEST_MODE, "openSidebarRecent"),
TestParameter(NOT_IN_GUEST_MODE, "openSidebarOffline"),
TestParameter(NOT_IN_GUEST_MODE,
"openSidebarSharedWithMe"),
TestParameter(NOT_IN_GUEST_MODE, "autocomplete")));
INSTANTIATE_TEST_CASE_P(
Transfer,
FileManagerBrowserTest,
::testing::Values(TestParameter(NOT_IN_GUEST_MODE,
"transferFromDriveToDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromDownloadsToDrive"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromSharedToDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromSharedToDrive"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromRecentToDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromRecentToDrive"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromOfflineToDownloads"),
TestParameter(NOT_IN_GUEST_MODE,
"transferFromOfflineToDrive")));
INSTANTIATE_TEST_CASE_P(
HideSearchBox,
FileManagerBrowserTest,
::testing::Values(TestParameter(IN_GUEST_MODE, "hideSearchBox"),
TestParameter(NOT_IN_GUEST_MODE, "hideSearchBox")));
INSTANTIATE_TEST_CASE_P(
RestorePrefs,
FileManagerBrowserTest,
::testing::Values(TestParameter(IN_GUEST_MODE, "restoreSortColumn"),
TestParameter(NOT_IN_GUEST_MODE, "restoreSortColumn"),
TestParameter(IN_GUEST_MODE, "restoreCurrentView"),
TestParameter(NOT_IN_GUEST_MODE, "restoreCurrentView")));
INSTANTIATE_TEST_CASE_P(
ShareDialog,
FileManagerBrowserTest,
::testing::Values(TestParameter(NOT_IN_GUEST_MODE, "shareFile"),
TestParameter(NOT_IN_GUEST_MODE, "shareDirectory")));
INSTANTIATE_TEST_CASE_P(
restoreGeometry,
FileManagerBrowserTest,
::testing::Values(TestParameter(NOT_IN_GUEST_MODE, "restoreGeometry"),
TestParameter(IN_GUEST_MODE, "restoreGeometry")));
} // namespace
} // namespace file_manager