| // 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 "base/win/shortcut.h" |
| |
| #include <stdint.h> |
| |
| #include <string> |
| |
| #include "base/files/file_path.h" |
| #include "base/files/file_util.h" |
| #include "base/files/scoped_temp_dir.h" |
| #include "base/macros.h" |
| #include "base/test/test_file_util.h" |
| #include "base/test/test_shortcut_win.h" |
| #include "base/win/scoped_com_initializer.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace base { |
| namespace win { |
| |
| namespace { |
| |
| static const char kFileContents[] = "This is a target."; |
| static const char kFileContents2[] = "This is another target."; |
| |
| class ShortcutTest : public testing::Test { |
| protected: |
| void SetUp() override { |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| ASSERT_TRUE(temp_dir_2_.CreateUniqueTempDir()); |
| |
| link_file_ = temp_dir_.GetPath().Append(L"My Link.lnk"); |
| |
| // Shortcut 1's properties |
| { |
| const FilePath target_file(temp_dir_.GetPath().Append(L"Target 1.txt")); |
| WriteFile(target_file, kFileContents, arraysize(kFileContents)); |
| |
| link_properties_.set_target(target_file); |
| link_properties_.set_working_dir(temp_dir_.GetPath()); |
| link_properties_.set_arguments(L"--magic --awesome"); |
| link_properties_.set_description(L"Chrome is awesome."); |
| link_properties_.set_icon(link_properties_.target, 4); |
| link_properties_.set_app_id(L"Chrome"); |
| link_properties_.set_dual_mode(false); |
| |
| // The CLSID below was randomly selected. |
| static constexpr CLSID toast_activator_clsid = { |
| 0x08d401c2, |
| 0x3f79, |
| 0x41d8, |
| {0x89, 0xd0, 0x99, 0x25, 0xee, 0x16, 0x28, 0x63}}; |
| link_properties_.set_toast_activator_clsid(toast_activator_clsid); |
| } |
| |
| // Shortcut 2's properties (all different from properties of shortcut 1). |
| { |
| const FilePath target_file_2(temp_dir_.GetPath().Append(L"Target 2.txt")); |
| WriteFile(target_file_2, kFileContents2, arraysize(kFileContents2)); |
| |
| FilePath icon_path_2; |
| CreateTemporaryFileInDir(temp_dir_.GetPath(), &icon_path_2); |
| |
| link_properties_2_.set_target(target_file_2); |
| link_properties_2_.set_working_dir(temp_dir_2_.GetPath()); |
| link_properties_2_.set_arguments(L"--super --crazy"); |
| link_properties_2_.set_description(L"The best in the west."); |
| link_properties_2_.set_icon(icon_path_2, 0); |
| link_properties_2_.set_app_id(L"Chrome.UserLevelCrazySuffix"); |
| link_properties_2_.set_dual_mode(true); |
| link_properties_2_.set_toast_activator_clsid(CLSID_NULL); |
| } |
| } |
| |
| ScopedCOMInitializer com_initializer_; |
| ScopedTempDir temp_dir_; |
| ScopedTempDir temp_dir_2_; |
| |
| // The link file to be created/updated in the shortcut tests below. |
| FilePath link_file_; |
| |
| // Properties for the created shortcut. |
| ShortcutProperties link_properties_; |
| |
| // Properties for the updated shortcut. |
| ShortcutProperties link_properties_2_; |
| }; |
| |
| } // namespace |
| |
| TEST_F(ShortcutTest, CreateAndResolveShortcutProperties) { |
| // Test all properties. |
| FilePath file_1(temp_dir_.GetPath().Append(L"Link1.lnk")); |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| file_1, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties properties_read_1; |
| ASSERT_TRUE(ResolveShortcutProperties( |
| file_1, ShortcutProperties::PROPERTIES_ALL, &properties_read_1)); |
| EXPECT_EQ(static_cast<unsigned>(ShortcutProperties::PROPERTIES_ALL), |
| properties_read_1.options); |
| ValidatePathsAreEqual(link_properties_.target, properties_read_1.target); |
| ValidatePathsAreEqual(link_properties_.working_dir, |
| properties_read_1.working_dir); |
| EXPECT_EQ(link_properties_.arguments, properties_read_1.arguments); |
| EXPECT_EQ(link_properties_.description, properties_read_1.description); |
| ValidatePathsAreEqual(link_properties_.icon, properties_read_1.icon); |
| EXPECT_EQ(link_properties_.icon_index, properties_read_1.icon_index); |
| EXPECT_EQ(link_properties_.app_id, properties_read_1.app_id); |
| EXPECT_EQ(link_properties_.dual_mode, properties_read_1.dual_mode); |
| EXPECT_EQ(link_properties_.toast_activator_clsid, |
| properties_read_1.toast_activator_clsid); |
| |
| // Test simple shortcut with no special properties set. |
| FilePath file_2(temp_dir_.GetPath().Append(L"Link2.lnk")); |
| ShortcutProperties only_target_properties; |
| only_target_properties.set_target(link_properties_.target); |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| file_2, only_target_properties, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties properties_read_2; |
| ASSERT_TRUE(ResolveShortcutProperties( |
| file_2, ShortcutProperties::PROPERTIES_ALL, &properties_read_2)); |
| EXPECT_EQ(static_cast<unsigned>(ShortcutProperties::PROPERTIES_ALL), |
| properties_read_2.options); |
| ValidatePathsAreEqual(only_target_properties.target, |
| properties_read_2.target); |
| ValidatePathsAreEqual(FilePath(), properties_read_2.working_dir); |
| EXPECT_EQ(L"", properties_read_2.arguments); |
| EXPECT_EQ(L"", properties_read_2.description); |
| ValidatePathsAreEqual(FilePath(), properties_read_2.icon); |
| EXPECT_EQ(0, properties_read_2.icon_index); |
| EXPECT_EQ(L"", properties_read_2.app_id); |
| EXPECT_FALSE(properties_read_2.dual_mode); |
| EXPECT_EQ(CLSID_NULL, properties_read_2.toast_activator_clsid); |
| } |
| |
| TEST_F(ShortcutTest, CreateAndResolveShortcut) { |
| ShortcutProperties only_target_properties; |
| only_target_properties.set_target(link_properties_.target); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, only_target_properties, SHORTCUT_CREATE_ALWAYS)); |
| |
| FilePath resolved_name; |
| EXPECT_TRUE(ResolveShortcut(link_file_, &resolved_name, NULL)); |
| |
| char read_contents[arraysize(kFileContents)]; |
| base::ReadFile(resolved_name, read_contents, arraysize(read_contents)); |
| EXPECT_STREQ(kFileContents, read_contents); |
| } |
| |
| TEST_F(ShortcutTest, ResolveShortcutWithArgs) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| FilePath resolved_name; |
| string16 args; |
| EXPECT_TRUE(ResolveShortcut(link_file_, &resolved_name, &args)); |
| |
| char read_contents[arraysize(kFileContents)]; |
| base::ReadFile(resolved_name, read_contents, arraysize(read_contents)); |
| EXPECT_STREQ(kFileContents, read_contents); |
| EXPECT_EQ(link_properties_.arguments, args); |
| } |
| |
| TEST_F(ShortcutTest, CreateShortcutWithOnlySomeProperties) { |
| ShortcutProperties target_and_args_properties; |
| target_and_args_properties.set_target(link_properties_.target); |
| target_and_args_properties.set_arguments(link_properties_.arguments); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, target_and_args_properties, |
| SHORTCUT_CREATE_ALWAYS)); |
| |
| ValidateShortcut(link_file_, target_and_args_properties); |
| } |
| |
| TEST_F(ShortcutTest, CreateShortcutVerifyProperties) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ValidateShortcut(link_file_, link_properties_); |
| } |
| |
| TEST_F(ShortcutTest, UpdateShortcutVerifyProperties) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_2_, SHORTCUT_UPDATE_EXISTING)); |
| |
| ValidateShortcut(link_file_, link_properties_2_); |
| } |
| |
| TEST_F(ShortcutTest, UpdateShortcutUpdateOnlyTargetAndResolve) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties update_only_target_properties; |
| update_only_target_properties.set_target(link_properties_2_.target); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, update_only_target_properties, |
| SHORTCUT_UPDATE_EXISTING)); |
| |
| ShortcutProperties expected_properties = link_properties_; |
| expected_properties.set_target(link_properties_2_.target); |
| ValidateShortcut(link_file_, expected_properties); |
| |
| FilePath resolved_name; |
| EXPECT_TRUE(ResolveShortcut(link_file_, &resolved_name, NULL)); |
| |
| char read_contents[arraysize(kFileContents2)]; |
| base::ReadFile(resolved_name, read_contents, arraysize(read_contents)); |
| EXPECT_STREQ(kFileContents2, read_contents); |
| } |
| |
| TEST_F(ShortcutTest, UpdateShortcutMakeDualMode) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties make_dual_mode_properties; |
| make_dual_mode_properties.set_dual_mode(true); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, make_dual_mode_properties, |
| SHORTCUT_UPDATE_EXISTING)); |
| |
| ShortcutProperties expected_properties = link_properties_; |
| expected_properties.set_dual_mode(true); |
| ValidateShortcut(link_file_, expected_properties); |
| } |
| |
| TEST_F(ShortcutTest, UpdateShortcutRemoveDualMode) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_2_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties remove_dual_mode_properties; |
| remove_dual_mode_properties.set_dual_mode(false); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, remove_dual_mode_properties, |
| SHORTCUT_UPDATE_EXISTING)); |
| |
| ShortcutProperties expected_properties = link_properties_2_; |
| expected_properties.set_dual_mode(false); |
| ValidateShortcut(link_file_, expected_properties); |
| } |
| |
| TEST_F(ShortcutTest, UpdateShortcutClearArguments) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties clear_arguments_properties; |
| clear_arguments_properties.set_arguments(string16()); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, clear_arguments_properties, |
| SHORTCUT_UPDATE_EXISTING)); |
| |
| ShortcutProperties expected_properties = link_properties_; |
| expected_properties.set_arguments(string16()); |
| ValidateShortcut(link_file_, expected_properties); |
| } |
| |
| TEST_F(ShortcutTest, FailUpdateShortcutThatDoesNotExist) { |
| ASSERT_FALSE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_UPDATE_EXISTING)); |
| ASSERT_FALSE(PathExists(link_file_)); |
| } |
| |
| TEST_F(ShortcutTest, ReplaceShortcutAllProperties) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_2_, SHORTCUT_REPLACE_EXISTING)); |
| |
| ValidateShortcut(link_file_, link_properties_2_); |
| } |
| |
| TEST_F(ShortcutTest, ReplaceShortcutSomeProperties) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| ShortcutProperties new_properties; |
| new_properties.set_target(link_properties_2_.target); |
| new_properties.set_arguments(link_properties_2_.arguments); |
| new_properties.set_description(link_properties_2_.description); |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, new_properties, SHORTCUT_REPLACE_EXISTING)); |
| |
| // Expect only properties in |new_properties| to be set, all other properties |
| // should have been overwritten. |
| ShortcutProperties expected_properties(new_properties); |
| expected_properties.set_working_dir(FilePath()); |
| expected_properties.set_icon(FilePath(), 0); |
| expected_properties.set_app_id(string16()); |
| expected_properties.set_dual_mode(false); |
| ValidateShortcut(link_file_, expected_properties); |
| } |
| |
| TEST_F(ShortcutTest, FailReplaceShortcutThatDoesNotExist) { |
| ASSERT_FALSE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_REPLACE_EXISTING)); |
| ASSERT_FALSE(PathExists(link_file_)); |
| } |
| |
| // Test that the old arguments remain on the replaced shortcut when not |
| // otherwise specified. |
| TEST_F(ShortcutTest, ReplaceShortcutKeepOldArguments) { |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_, SHORTCUT_CREATE_ALWAYS)); |
| |
| // Do not explicitly set the arguments. |
| link_properties_2_.options &= |
| ~ShortcutProperties::PROPERTIES_ARGUMENTS; |
| ASSERT_TRUE(CreateOrUpdateShortcutLink( |
| link_file_, link_properties_2_, SHORTCUT_REPLACE_EXISTING)); |
| |
| ShortcutProperties expected_properties(link_properties_2_); |
| expected_properties.set_arguments(link_properties_.arguments); |
| ValidateShortcut(link_file_, expected_properties); |
| } |
| |
| } // namespace win |
| } // namespace base |