| // Copyright (c) 2012 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 "ui/base/dragdrop/os_exchange_data.h" |
| |
| #include <memory> |
| |
| #include "base/bind.h" |
| #include "base/callback.h" |
| #include "base/files/file_util.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/test/task_environment.h" |
| #include "base/win/scoped_hglobal.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "testing/platform_test.h" |
| #include "ui/base/clipboard/clipboard_format_type.h" |
| #include "ui/base/dragdrop/file_info.h" |
| #include "ui/base/dragdrop/os_exchange_data_provider_win.h" |
| #include "url/gurl.h" |
| |
| namespace ui { |
| |
| namespace { |
| const std::vector<DWORD> kStorageMediaTypesForVirtualFiles = { |
| TYMED_ISTORAGE, |
| TYMED_ISTREAM, |
| TYMED_HGLOBAL, |
| }; |
| |
| } // namespace |
| |
| class OSExchangeDataWinTest : public ::testing::Test { |
| public: |
| OSExchangeDataWinTest() |
| : task_environment_(base::test::TaskEnvironment::MainThreadType::UI) {} |
| |
| void OnGotVirtualFilesAsTempFiles( |
| const std::vector<std::pair<base::FilePath, base::FilePath>>& |
| filepaths_and_names) { |
| // Clear any previous results and cache a vector of FileInfo objects for |
| // verification. |
| retrieved_virtual_files_.clear(); |
| |
| for (const auto& filepath_and_name : filepaths_and_names) { |
| retrieved_virtual_files_.push_back( |
| FileInfo(filepath_and_name.first, filepath_and_name.second)); |
| } |
| } |
| |
| protected: |
| std::vector<FileInfo> retrieved_virtual_files_; |
| base::test::TaskEnvironment task_environment_; |
| }; |
| |
| // Test getting using the IDataObject COM API |
| TEST_F(OSExchangeDataWinTest, StringDataAccessViaCOM) { |
| OSExchangeData data; |
| std::wstring input = L"O hai googlz."; |
| data.SetString(input); |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| FORMATETC format_etc = |
| { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc)); |
| |
| STGMEDIUM medium; |
| EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium)); |
| std::wstring output = |
| base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get(); |
| EXPECT_EQ(input, output); |
| ReleaseStgMedium(&medium); |
| } |
| |
| // Test setting using the IDataObject COM API |
| TEST_F(OSExchangeDataWinTest, StringDataWritingViaCOM) { |
| OSExchangeData data; |
| std::wstring input = L"http://www.google.com/"; |
| |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| // Store data in the object using the COM SetData API. |
| CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL); |
| FORMATETC format_etc = |
| { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| STGMEDIUM medium; |
| medium.tymed = TYMED_HGLOBAL; |
| HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1)); |
| base::win::ScopedHGlobal<wchar_t*> global_lock(glob); |
| wchar_t* buffer_handle = global_lock.get(); |
| wcscpy_s(buffer_handle, input.size() + 1, input.c_str()); |
| medium.hGlobal = glob; |
| medium.pUnkForRelease = NULL; |
| EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE)); |
| |
| // Construct a new object with the old object so that we can use our access |
| // APIs. |
| OSExchangeData data2(data.provider().Clone()); |
| EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES)); |
| GURL url_from_data; |
| std::wstring title; |
| EXPECT_TRUE(data2.GetURLAndTitle( |
| OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title)); |
| GURL reference_url(input); |
| EXPECT_EQ(reference_url.spec(), url_from_data.spec()); |
| } |
| |
| // Verifies SetData invoked twice with the same data clobbers existing data. |
| TEST_F(OSExchangeDataWinTest, RemoveData) { |
| OSExchangeData data; |
| std::wstring input = L"http://www.google.com/"; |
| std::wstring input2 = L"http://www.google2.com/"; |
| |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| // Store data in the object using the COM SetData API. |
| CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL); |
| FORMATETC format_etc = |
| { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| STGMEDIUM medium; |
| medium.tymed = TYMED_HGLOBAL; |
| { |
| HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1)); |
| base::win::ScopedHGlobal<wchar_t*> global_lock(glob); |
| wchar_t* buffer_handle = global_lock.get(); |
| wcscpy_s(buffer_handle, input.size() + 1, input.c_str()); |
| medium.hGlobal = glob; |
| medium.pUnkForRelease = NULL; |
| EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE)); |
| } |
| // This should clobber the existing data. |
| { |
| HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input2.size() + 1)); |
| base::win::ScopedHGlobal<wchar_t*> global_lock(glob); |
| wchar_t* buffer_handle = global_lock.get(); |
| wcscpy_s(buffer_handle, input2.size() + 1, input2.c_str()); |
| medium.hGlobal = glob; |
| medium.pUnkForRelease = NULL; |
| EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE)); |
| } |
| EXPECT_EQ(1u, static_cast<DataObjectImpl*>(com_data.Get())->size()); |
| |
| // Construct a new object with the old object so that we can use our access |
| // APIs. |
| OSExchangeData data2(data.provider().Clone()); |
| EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES)); |
| GURL url_from_data; |
| std::wstring title; |
| EXPECT_TRUE(data2.GetURLAndTitle( |
| OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title)); |
| EXPECT_EQ(GURL(input2).spec(), url_from_data.spec()); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, URLDataAccessViaCOM) { |
| OSExchangeData data; |
| GURL url("http://www.google.com/"); |
| data.SetURL(url, L""); |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL); |
| FORMATETC format_etc = |
| { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc)); |
| |
| STGMEDIUM medium; |
| EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium)); |
| std::wstring output = |
| base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get(); |
| EXPECT_EQ(url.spec(), base::WideToUTF8(output)); |
| ReleaseStgMedium(&medium); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, MultipleFormatsViaCOM) { |
| OSExchangeData data; |
| std::string url_spec = "http://www.google.com/"; |
| GURL url(url_spec); |
| std::wstring text = L"O hai googlz."; |
| data.SetURL(url, L"Google"); |
| data.SetString(text); |
| |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL); |
| FORMATETC url_format_etc = |
| { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| EXPECT_EQ(S_OK, com_data->QueryGetData(&url_format_etc)); |
| FORMATETC text_format_etc = |
| { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| EXPECT_EQ(S_OK, com_data->QueryGetData(&text_format_etc)); |
| |
| STGMEDIUM medium; |
| EXPECT_EQ(S_OK, com_data->GetData(&url_format_etc, &medium)); |
| std::wstring output_url = |
| base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get(); |
| EXPECT_EQ(url.spec(), base::WideToUTF8(output_url)); |
| ReleaseStgMedium(&medium); |
| |
| // The text is supposed to be the raw text of the URL, _NOT_ the value of |
| // |text|! This is because the URL is added first and thus takes precedence! |
| EXPECT_EQ(S_OK, com_data->GetData(&text_format_etc, &medium)); |
| std::wstring output_text = |
| base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get(); |
| EXPECT_EQ(url_spec, base::WideToUTF8(output_text)); |
| ReleaseStgMedium(&medium); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, EnumerationViaCOM) { |
| OSExchangeData data; |
| data.SetURL(GURL("http://www.google.com/"), L""); |
| data.SetString(L"O hai googlz."); |
| |
| CLIPFORMAT cfstr_file_group_descriptor = |
| RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR); |
| CLIPFORMAT text_x_moz_url = RegisterClipboardFormat(L"text/x-moz-url"); |
| |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| Microsoft::WRL::ComPtr<IEnumFORMATETC> enumerator; |
| EXPECT_EQ(S_OK, com_data.Get()->EnumFormatEtc(DATADIR_GET, |
| enumerator.GetAddressOf())); |
| |
| // Test that we can get one item. |
| { |
| // Explictly don't reset the first time, to verify the creation state is |
| // OK. |
| ULONG retrieved = 0; |
| FORMATETC elements_array[1]; |
| EXPECT_EQ(S_OK, enumerator->Next(1, |
| reinterpret_cast<FORMATETC*>(&elements_array), &retrieved)); |
| EXPECT_EQ(1u, retrieved); |
| EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat); |
| } |
| |
| // Test that we can get one item with a NULL retrieved value. |
| { |
| EXPECT_EQ(S_OK, enumerator->Reset()); |
| FORMATETC elements_array[1]; |
| EXPECT_EQ(S_OK, enumerator->Next(1, |
| reinterpret_cast<FORMATETC*>(&elements_array), NULL)); |
| EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat); |
| } |
| |
| // Test that we can get two items. |
| { |
| EXPECT_EQ(S_OK, enumerator->Reset()); |
| ULONG retrieved = 0; |
| FORMATETC elements_array[2]; |
| EXPECT_EQ(S_OK, enumerator->Next(2, |
| reinterpret_cast<FORMATETC*>(&elements_array), &retrieved)); |
| EXPECT_EQ(2u, retrieved); |
| EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat); |
| EXPECT_EQ(cfstr_file_group_descriptor, elements_array[1].cfFormat); |
| } |
| |
| // Test that we can skip the first item. |
| { |
| EXPECT_EQ(S_OK, enumerator->Reset()); |
| EXPECT_EQ(S_OK, enumerator->Skip(1)); |
| ULONG retrieved = 0; |
| FORMATETC elements_array[1]; |
| EXPECT_EQ(S_OK, enumerator->Next(1, |
| reinterpret_cast<FORMATETC*>(&elements_array), &retrieved)); |
| EXPECT_EQ(1u, retrieved); |
| EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat); |
| } |
| |
| // Test that we can skip the first item, and create a clone that matches in |
| // this state, and modify the original without affecting the clone. |
| { |
| EXPECT_EQ(S_OK, enumerator->Reset()); |
| EXPECT_EQ(S_OK, enumerator->Skip(1)); |
| Microsoft::WRL::ComPtr<IEnumFORMATETC> cloned_enumerator; |
| EXPECT_EQ(S_OK, enumerator.Get()->Clone(cloned_enumerator.GetAddressOf())); |
| EXPECT_EQ(S_OK, enumerator.Get()->Reset()); |
| |
| { |
| ULONG retrieved = 0; |
| FORMATETC elements_array[1]; |
| EXPECT_EQ(S_OK, cloned_enumerator->Next(1, |
| reinterpret_cast<FORMATETC*>(&elements_array), &retrieved)); |
| EXPECT_EQ(1u, retrieved); |
| EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat); |
| } |
| |
| { |
| ULONG retrieved = 0; |
| FORMATETC elements_array[1]; |
| EXPECT_EQ(S_OK, enumerator->Next(1, |
| reinterpret_cast<FORMATETC*>(&elements_array), &retrieved)); |
| EXPECT_EQ(1u, retrieved); |
| EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat); |
| } |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, TestURLExchangeFormatsViaCOM) { |
| OSExchangeData data; |
| std::string url_spec = "http://www.google.com/"; |
| GURL url(url_spec); |
| std::wstring url_title = L"www.google.com"; |
| data.SetURL(url, url_title); |
| |
| // File contents access via COM |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| { |
| CLIPFORMAT cfstr_file_contents = |
| RegisterClipboardFormat(CFSTR_FILECONTENTS); |
| // format_etc.lindex value 0 used for file drop. |
| FORMATETC format_etc = {cfstr_file_contents, nullptr, DVASPECT_CONTENT, 0, |
| TYMED_HGLOBAL}; |
| EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc)); |
| |
| STGMEDIUM medium; |
| EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium)); |
| base::win::ScopedHGlobal<char*> glob(medium.hGlobal); |
| std::string output(glob.get(), glob.Size()); |
| std::string file_contents = "[InternetShortcut]\r\nURL="; |
| file_contents += url_spec; |
| file_contents += "\r\n"; |
| EXPECT_EQ(file_contents, output); |
| ReleaseStgMedium(&medium); |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, FileContents) { |
| OSExchangeData data; |
| std::string file_contents("data\0with\0nulls", 15); |
| data.SetFileContents(base::FilePath(L"filename.txt"), file_contents); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| base::FilePath filename; |
| std::string read_contents; |
| EXPECT_TRUE(copy.GetFileContents(&filename, &read_contents)); |
| EXPECT_EQ(L"filename.txt", filename.value()); |
| EXPECT_EQ(file_contents, read_contents); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFiles) { |
| const base::FilePath kPathPlaceholder(FILE_PATH_LITERAL("temp.tmp")); |
| |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {base::FilePath(FILE_PATH_LITERAL("filename.txt")), |
| std::string("just some data")}, |
| {base::FilePath(FILE_PATH_LITERAL("another filename.txt")), |
| std::string("just some data\0with\0nulls", 25)}, |
| {base::FilePath(FILE_PATH_LITERAL("and another filename.txt")), |
| std::string("just some more data")}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| std::vector<FileInfo> file_infos; |
| EXPECT_TRUE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < file_infos.size(); i++) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].first, file_infos[i].display_name); |
| EXPECT_EQ(kPathPlaceholder, file_infos[i].path); |
| } |
| |
| base::FilePath temp_dir; |
| EXPECT_TRUE(base::GetTempDir(&temp_dir)); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_TRUE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(kTestFilenamesAndContents.size(), |
| retrieved_virtual_files_.size()); |
| for (size_t i = 0; i < retrieved_virtual_files_.size(); i++) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].first, |
| retrieved_virtual_files_[i].display_name); |
| // Check if the temp files that back the virtual files are actually |
| // created in the temp directory. Need to compare long file paths here |
| // because GetTempDir can return a short ("8.3") path if the test is run |
| // under a username that is too long. |
| EXPECT_EQ( |
| base::MakeLongFilePath(temp_dir), |
| base::MakeLongFilePath(retrieved_virtual_files_[i].path.DirName())); |
| EXPECT_EQ(kTestFilenamesAndContents[i].first.Extension(), |
| retrieved_virtual_files_[i].path.Extension()); |
| std::string read_contents; |
| EXPECT_TRUE(base::ReadFileToString(retrieved_virtual_files_[i].path, |
| &read_contents)); |
| if (tymed != TYMED_ISTORAGE) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].second, read_contents); |
| } else { |
| // IStorage uses compound files, so temp files won't be flat text files. |
| // Just make sure the original contents appears in the compound files. |
| EXPECT_TRUE(read_contents.find(kTestFilenamesAndContents[i].second) != |
| std::string::npos); |
| } |
| } |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFilesRealFilesPreferred) { |
| // Verify that no virtual files retrieved if there is real file data. |
| const std::vector<FileInfo> kTestFilenames = { |
| {base::FilePath(FILE_PATH_LITERAL("C:\\tmp\\test_file1")), |
| base::FilePath()}, |
| {base::FilePath(FILE_PATH_LITERAL("C:\\tmp\\test_file2")), |
| base::FilePath()}, |
| }; |
| |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {base::FilePath(FILE_PATH_LITERAL("filename.txt")), |
| std::string("just some data")}, |
| {base::FilePath(FILE_PATH_LITERAL("another filename.txt")), |
| std::string("just some data\0with\0nulls", 25)}, |
| {base::FilePath(FILE_PATH_LITERAL("and another filename.txt")), |
| std::string("just some more data")}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.SetFilenames(kTestFilenames); |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| |
| std::vector<FileInfo> real_filenames; |
| EXPECT_TRUE(copy.GetFilenames(&real_filenames)); |
| EXPECT_EQ(kTestFilenames.size(), real_filenames.size()); |
| EXPECT_EQ(kTestFilenames, real_filenames); |
| |
| std::vector<FileInfo> file_infos; |
| EXPECT_FALSE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(static_cast<size_t>(0), file_infos.size()); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_FALSE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(static_cast<size_t>(0), retrieved_virtual_files_.size()); |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFilesDuplicateNames) { |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {base::FilePath(FILE_PATH_LITERAL("A (1) (2).txt")), |
| std::string("just some data")}, |
| {base::FilePath(FILE_PATH_LITERAL("A.txt")), |
| std::string("just some more data")}, |
| {base::FilePath(FILE_PATH_LITERAL("A (1).txt")), |
| std::string("just some more more data")}, |
| {base::FilePath(FILE_PATH_LITERAL("A.txt")), |
| std::string("just some more more more data")}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| std::vector<FileInfo> file_infos; |
| EXPECT_TRUE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < file_infos.size(); i++) { |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| file_infos[j].display_name.value(), |
| file_infos[i].display_name.value())); |
| } |
| } |
| |
| base::FilePath temp_dir; |
| EXPECT_TRUE(base::GetTempDir(&temp_dir)); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_TRUE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < retrieved_virtual_files_.size(); i++) { |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].display_name.value(), |
| retrieved_virtual_files_[i].display_name.value())); |
| } |
| |
| // Check that temp file path is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].path.value(), |
| retrieved_virtual_files_[i].path.value())); |
| } |
| |
| // Check if the temp files that back the virtual files are actually |
| // created in the temp directory. Need to compare long file paths here |
| // because GetTempDir can return a short ("8.3") path if the test is run |
| // under a username that is too long. |
| EXPECT_EQ( |
| base::MakeLongFilePath(temp_dir), |
| base::MakeLongFilePath(retrieved_virtual_files_[i].path.DirName())); |
| EXPECT_EQ(kTestFilenamesAndContents[i].first.Extension(), |
| retrieved_virtual_files_[i].path.Extension()); |
| std::string read_contents; |
| EXPECT_TRUE(base::ReadFileToString(retrieved_virtual_files_[i].path, |
| &read_contents)); |
| if (tymed != TYMED_ISTORAGE) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].second, read_contents); |
| } else { |
| // IStorage uses compound files, so temp files won't be flat text files. |
| // Just make sure the original contents appears in the compound files. |
| EXPECT_TRUE(read_contents.find(kTestFilenamesAndContents[i].second) != |
| std::string::npos); |
| } |
| } |
| } |
| } // namespace ui |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFilesDuplicateNamesCaseInsensitivity) { |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {base::FilePath(FILE_PATH_LITERAL("a.txt")), |
| std::string("just some data")}, |
| {base::FilePath(FILE_PATH_LITERAL("B.txt")), |
| std::string("just some more data")}, |
| {base::FilePath(FILE_PATH_LITERAL("A.txt")), |
| std::string("just some more more data")}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| std::vector<FileInfo> file_infos; |
| EXPECT_TRUE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < file_infos.size(); i++) { |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| file_infos[j].display_name.value(), |
| file_infos[i].display_name.value())); |
| } |
| } |
| |
| base::FilePath temp_dir; |
| EXPECT_TRUE(base::GetTempDir(&temp_dir)); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_TRUE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < retrieved_virtual_files_.size(); i++) { |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].display_name.value(), |
| retrieved_virtual_files_[i].display_name.value())); |
| } |
| |
| // Check that temp file path is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].path.value(), |
| retrieved_virtual_files_[i].path.value())); |
| } |
| |
| // Check if the temp files that back the virtual files are actually |
| // created in the temp directory. Need to compare long file paths here |
| // because GetTempDir can return a short ("8.3") path if the test is run |
| // under a username that is too long. |
| EXPECT_EQ( |
| base::MakeLongFilePath(temp_dir), |
| base::MakeLongFilePath(retrieved_virtual_files_[i].path.DirName())); |
| EXPECT_EQ(kTestFilenamesAndContents[i].first.Extension(), |
| retrieved_virtual_files_[i].path.Extension()); |
| std::string read_contents; |
| EXPECT_TRUE(base::ReadFileToString(retrieved_virtual_files_[i].path, |
| &read_contents)); |
| if (tymed != TYMED_ISTORAGE) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].second, read_contents); |
| } else { |
| // IStorage uses compound files, so temp files won't be flat text files. |
| // Just make sure the original contents appears in the compound files. |
| EXPECT_TRUE(read_contents.find(kTestFilenamesAndContents[i].second) != |
| std::string::npos); |
| } |
| } |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFilesInvalidAndDuplicateNames) { |
| const base::string16 kInvalidFileNameCharacters( |
| FILE_PATH_LITERAL("\\/:*?\"<>|")); |
| const base::string16 kInvalidFilePathCharacters( |
| FILE_PATH_LITERAL("/*?\"<>|")); |
| const base::FilePath kPathWithInvalidFileNameCharacters = |
| base::FilePath(kInvalidFileNameCharacters) |
| .AddExtension(FILE_PATH_LITERAL("txt")); |
| const base::FilePath kEmptyDisplayName(FILE_PATH_LITERAL("")); |
| const base::FilePath kMaxPathDisplayName = |
| base::FilePath(base::string16(MAX_PATH - 5, L'a')) |
| .AddExtension(FILE_PATH_LITERAL("txt")); |
| |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {kPathWithInvalidFileNameCharacters, std::string("just some data")}, |
| {kPathWithInvalidFileNameCharacters, |
| std::string("just some data\0with\0nulls", 25)}, |
| {// Test that still get a unique name if a previous uniquified |
| // name is a duplicate of this one. |
| kPathWithInvalidFileNameCharacters.InsertBeforeExtension( |
| FILE_PATH_LITERAL(" (1)")), |
| std::string("just some more data")}, |
| // Expect a default display name to be generated ("download" if it |
| // matters). |
| {kEmptyDisplayName, std::string("data for an empty display name")}, |
| {kEmptyDisplayName, |
| std::string("data for another empty display name")}, |
| // Expect good behavior if the display name length exceeds MAX_PATH. |
| {kMaxPathDisplayName, |
| std::string("data for a >MAX_PATH display name")}, |
| {kMaxPathDisplayName, |
| std::string("data for another >MAX_PATH display name")}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| std::vector<FileInfo> file_infos; |
| EXPECT_TRUE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < file_infos.size(); i++) { |
| // Check that display name does not contain invalid characters. |
| EXPECT_EQ(std::string::npos, |
| file_infos[i].display_name.value().find_first_of( |
| kInvalidFileNameCharacters)); |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| file_infos[j].display_name.value(), |
| file_infos[i].display_name.value())); |
| } |
| } |
| |
| base::FilePath temp_dir; |
| EXPECT_TRUE(base::GetTempDir(&temp_dir)); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_TRUE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < retrieved_virtual_files_.size(); i++) { |
| // Check that display name does not contain invalid characters. |
| EXPECT_EQ(std::string::npos, |
| retrieved_virtual_files_[i].display_name.value().find_first_of( |
| kInvalidFileNameCharacters)); |
| // Check that display name is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].display_name.value(), |
| retrieved_virtual_files_[i].display_name.value())); |
| } |
| // Check that temp file path does not contain invalid characters (except |
| // for separator). |
| EXPECT_EQ(std::string::npos, |
| retrieved_virtual_files_[i].path.value().find_first_of( |
| kInvalidFilePathCharacters)); |
| // Check that temp file path is unique. |
| for (size_t j = 0; j < i; j++) { |
| EXPECT_FALSE(base::FilePath::CompareEqualIgnoreCase( |
| retrieved_virtual_files_[j].path.value(), |
| retrieved_virtual_files_[i].path.value())); |
| } |
| |
| // Check if the temp files that back the virtual files are actually |
| // created in the temp directory. Need to compare long file paths here |
| // because GetTempDir can return a short ("8.3") path if the test is run |
| // under a username that is too long. |
| EXPECT_EQ( |
| base::MakeLongFilePath(temp_dir), |
| base::MakeLongFilePath(retrieved_virtual_files_[i].path.DirName())); |
| EXPECT_EQ(kTestFilenamesAndContents[i].first.Extension(), |
| retrieved_virtual_files_[i].path.Extension()); |
| std::string read_contents; |
| // Ability to read the contents implies a temp file was successfully |
| // created on the file system even though the original suggested display |
| // name had invalid filename characters. |
| EXPECT_TRUE(base::ReadFileToString(retrieved_virtual_files_[i].path, |
| &read_contents)); |
| if (tymed != TYMED_ISTORAGE) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].second, read_contents); |
| } else { |
| // IStorage uses compound files, so temp files won't be flat text files. |
| // Just make sure the original contents appears in the compound files. |
| EXPECT_TRUE(read_contents.find(kTestFilenamesAndContents[i].second) != |
| std::string::npos); |
| } |
| } |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, VirtualFilesEmptyContents) { |
| const std::vector<std::pair<base::FilePath, std::string>> |
| kTestFilenamesAndContents = { |
| {base::FilePath(FILE_PATH_LITERAL("file_with_no_contents.txt")), |
| std::string()}, |
| }; |
| |
| for (const auto& tymed : kStorageMediaTypesForVirtualFiles) { |
| OSExchangeData data; |
| data.provider().SetVirtualFileContentsForTesting(kTestFilenamesAndContents, |
| tymed); |
| |
| OSExchangeData copy(data.provider().Clone()); |
| std::vector<FileInfo> file_infos; |
| EXPECT_TRUE(copy.GetVirtualFilenames(&file_infos)); |
| EXPECT_EQ(kTestFilenamesAndContents.size(), file_infos.size()); |
| for (size_t i = 0; i < file_infos.size(); i++) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].first, file_infos[i].display_name); |
| } |
| |
| base::FilePath temp_dir; |
| EXPECT_TRUE(base::GetTempDir(&temp_dir)); |
| |
| // Callback for GetVirtualFilesAsTempFiles is executed when all virtual |
| // files are backed by temp files. |
| auto callback = |
| base::BindOnce(&OSExchangeDataWinTest::OnGotVirtualFilesAsTempFiles, |
| base::Unretained(this)); |
| |
| EXPECT_TRUE(copy.GetVirtualFilesAsTempFiles(std::move(callback))); |
| |
| // RunUntilIdle assures all async tasks are run. |
| task_environment_.RunUntilIdle(); |
| |
| EXPECT_EQ(kTestFilenamesAndContents.size(), |
| retrieved_virtual_files_.size()); |
| for (size_t i = 0; i < retrieved_virtual_files_.size(); i++) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].first, |
| retrieved_virtual_files_[i].display_name); |
| |
| // Check if the temp files that back the virtual files are actually |
| // created in the temp directory. Need to compare long file paths here |
| // because GetTempDir can return a short ("8.3") path if the test is run |
| // under a username that is too long. |
| EXPECT_EQ( |
| base::MakeLongFilePath(temp_dir), |
| base::MakeLongFilePath(retrieved_virtual_files_[i].path.DirName())); |
| EXPECT_EQ(kTestFilenamesAndContents[i].first.Extension(), |
| retrieved_virtual_files_[i].path.Extension()); |
| std::string read_contents; |
| EXPECT_TRUE(base::ReadFileToString(retrieved_virtual_files_[i].path, |
| &read_contents)); |
| // IStorage uses compound files, so temp files won't be flat text files. |
| // Just make sure the original contents appear in the compound files. |
| if (tymed != TYMED_ISTORAGE) { |
| EXPECT_EQ(kTestFilenamesAndContents[i].second, read_contents); |
| EXPECT_EQ(static_cast<size_t>(0), read_contents.length()); |
| } |
| } |
| } |
| } |
| |
| TEST_F(OSExchangeDataWinTest, CFHtml) { |
| OSExchangeData data; |
| GURL url("http://www.google.com/"); |
| std::wstring html( |
| L"<HTML>\n<BODY>\n" |
| L"<b>bold.</b> <i><b>This is bold italic.</b></i>\n" |
| L"</BODY>\n</HTML>"); |
| data.SetHtml(html, url); |
| |
| // Check the CF_HTML too. |
| std::string expected_cf_html( |
| "Version:0.9\r\nStartHTML:0000000139\r\nEndHTML:0000000288\r\n" |
| "StartFragment:0000000175\r\nEndFragment:0000000252\r\n" |
| "SourceURL:http://www.google.com/\r\n<html>\r\n<body>\r\n" |
| "<!--StartFragment-->"); |
| expected_cf_html += base::WideToUTF8(html); |
| expected_cf_html.append("<!--EndFragment-->\r\n</body>\r\n</html>"); |
| |
| FORMATETC format = ClipboardFormatType::GetHtmlType().ToFormatEtc(); |
| STGMEDIUM medium; |
| IDataObject* data_object = OSExchangeDataProviderWin::GetIDataObject(data); |
| EXPECT_EQ(S_OK, data_object->GetData(&format, &medium)); |
| base::win::ScopedHGlobal<char*> glob(medium.hGlobal); |
| std::string output(glob.get(), glob.Size()); |
| EXPECT_EQ(expected_cf_html, output); |
| ReleaseStgMedium(&medium); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, SetURLWithMaxPath) { |
| OSExchangeData data; |
| std::wstring long_title(L'a', MAX_PATH + 1); |
| data.SetURL(GURL("http://google.com"), long_title); |
| } |
| |
| TEST_F(OSExchangeDataWinTest, ProvideURLForPlainTextURL) { |
| OSExchangeData data; |
| data.SetString(L"http://google.com"); |
| |
| OSExchangeData data2(data.provider().Clone()); |
| ASSERT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES)); |
| GURL read_url; |
| std::wstring title; |
| EXPECT_TRUE(data2.GetURLAndTitle( |
| OSExchangeData::CONVERT_FILENAMES, &read_url, &title)); |
| EXPECT_EQ(GURL("http://google.com"), read_url); |
| } |
| |
| class MockDownloadFileProvider : public ui::DownloadFileProvider { |
| public: |
| MockDownloadFileProvider() = default; |
| ~MockDownloadFileProvider() override = default; |
| base::WeakPtr<MockDownloadFileProvider> GetWeakPtr() { |
| return weak_ptr_factory_.GetWeakPtr(); |
| } |
| |
| MOCK_METHOD1(Start, void(DownloadFileObserver* observer)); |
| MOCK_METHOD0(Wait, bool()); |
| MOCK_METHOD0(Stop, void()); |
| |
| private: |
| base::WeakPtrFactory<MockDownloadFileProvider> weak_ptr_factory_{this}; |
| }; |
| |
| // Verifies that DataObjectImpl::OnDownloadCompleted() doesn't delete |
| // the DownloadFileProvider instance. |
| TEST_F(OSExchangeDataWinTest, OnDownloadCompleted) { |
| OSExchangeData data; |
| Microsoft::WRL::ComPtr<IDataObject> com_data( |
| OSExchangeDataProviderWin::GetIDataObject(data)); |
| |
| OSExchangeDataProviderWin provider(com_data.Get()); |
| |
| auto download_file_provider = std::make_unique<MockDownloadFileProvider>(); |
| auto weak_ptr = download_file_provider->GetWeakPtr(); |
| OSExchangeData::DownloadFileInfo file_info( |
| base::FilePath(FILE_PATH_LITERAL("file_with_no_contents.txt")), |
| std::move(download_file_provider)); |
| provider.SetDownloadFileInfo(&file_info); |
| |
| OSExchangeDataProviderWin::GetDataObjectImpl(data)->OnDownloadCompleted( |
| base::FilePath()); |
| EXPECT_TRUE(weak_ptr); |
| } |
| |
| } // namespace ui |