| // 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. |
| |
| #import <Cocoa/Cocoa.h> |
| #include <stddef.h> |
| #include <stdint.h> |
| |
| #include "base/mac/mac_util.h" |
| |
| #include "base/files/file_path.h" |
| #include "base/files/file_util.h" |
| #include "base/files/scoped_temp_dir.h" |
| #include "base/mac/foundation_util.h" |
| #include "base/mac/scoped_cftyperef.h" |
| #include "base/mac/scoped_nsobject.h" |
| #include "base/macros.h" |
| #include "base/sys_info.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "testing/platform_test.h" |
| |
| #include <errno.h> |
| #include <sys/xattr.h> |
| |
| namespace base { |
| namespace mac { |
| |
| namespace { |
| |
| typedef PlatformTest MacUtilTest; |
| |
| TEST_F(MacUtilTest, GetUserDirectoryTest) { |
| // Try a few keys, make sure they come back with non-empty paths. |
| FilePath caches_dir; |
| EXPECT_TRUE(GetUserDirectory(NSCachesDirectory, &caches_dir)); |
| EXPECT_FALSE(caches_dir.empty()); |
| |
| FilePath application_support_dir; |
| EXPECT_TRUE(GetUserDirectory(NSApplicationSupportDirectory, |
| &application_support_dir)); |
| EXPECT_FALSE(application_support_dir.empty()); |
| |
| FilePath library_dir; |
| EXPECT_TRUE(GetUserDirectory(NSLibraryDirectory, &library_dir)); |
| EXPECT_FALSE(library_dir.empty()); |
| } |
| |
| TEST_F(MacUtilTest, TestLibraryPath) { |
| FilePath library_dir = GetUserLibraryPath(); |
| // Make sure the string isn't empty. |
| EXPECT_FALSE(library_dir.value().empty()); |
| } |
| |
| TEST_F(MacUtilTest, TestGetAppBundlePath) { |
| FilePath out; |
| |
| // Make sure it doesn't crash. |
| out = GetAppBundlePath(FilePath()); |
| EXPECT_TRUE(out.empty()); |
| |
| // Some more invalid inputs. |
| const char* const invalid_inputs[] = { |
| "/", "/foo", "foo", "/foo/bar.", "foo/bar.", "/foo/bar./bazquux", |
| "foo/bar./bazquux", "foo/.app", "//foo", |
| }; |
| for (size_t i = 0; i < arraysize(invalid_inputs); i++) { |
| out = GetAppBundlePath(FilePath(invalid_inputs[i])); |
| EXPECT_TRUE(out.empty()) << "loop: " << i; |
| } |
| |
| // Some valid inputs; this and |expected_outputs| should be in sync. |
| struct { |
| const char *in; |
| const char *expected_out; |
| } valid_inputs[] = { |
| { "FooBar.app/", "FooBar.app" }, |
| { "/FooBar.app", "/FooBar.app" }, |
| { "/FooBar.app/", "/FooBar.app" }, |
| { "//FooBar.app", "//FooBar.app" }, |
| { "/Foo/Bar.app", "/Foo/Bar.app" }, |
| { "/Foo/Bar.app/", "/Foo/Bar.app" }, |
| { "/F/B.app", "/F/B.app" }, |
| { "/F/B.app/", "/F/B.app" }, |
| { "/Foo/Bar.app/baz", "/Foo/Bar.app" }, |
| { "/Foo/Bar.app/baz/", "/Foo/Bar.app" }, |
| { "/Foo/Bar.app/baz/quux.app/quuux", "/Foo/Bar.app" }, |
| { "/Applications/Google Foo.app/bar/Foo Helper.app/quux/Foo Helper", |
| "/Applications/Google Foo.app" }, |
| }; |
| for (size_t i = 0; i < arraysize(valid_inputs); i++) { |
| out = GetAppBundlePath(FilePath(valid_inputs[i].in)); |
| EXPECT_FALSE(out.empty()) << "loop: " << i; |
| EXPECT_STREQ(valid_inputs[i].expected_out, |
| out.value().c_str()) << "loop: " << i; |
| } |
| } |
| |
| // http://crbug.com/425745 |
| TEST_F(MacUtilTest, DISABLED_TestExcludeFileFromBackups) { |
| // The file must already exist in order to set its exclusion property. |
| ScopedTempDir temp_dir_; |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| FilePath dummy_file_path = temp_dir_.GetPath().Append("DummyFile"); |
| const char dummy_data[] = "All your base are belong to us!"; |
| // Dump something real into the file. |
| ASSERT_EQ(static_cast<int>(arraysize(dummy_data)), |
| WriteFile(dummy_file_path, dummy_data, arraysize(dummy_data))); |
| NSString* fileURLString = |
| [NSString stringWithUTF8String:dummy_file_path.value().c_str()]; |
| NSURL* fileURL = [NSURL URLWithString:fileURLString]; |
| // Initial state should be non-excluded. |
| EXPECT_FALSE(CSBackupIsItemExcluded(base::mac::NSToCFCast(fileURL), NULL)); |
| // Exclude the file. |
| EXPECT_TRUE(SetFileBackupExclusion(dummy_file_path)); |
| // SetFileBackupExclusion never excludes by path. |
| Boolean excluded_by_path = FALSE; |
| Boolean excluded = |
| CSBackupIsItemExcluded(base::mac::NSToCFCast(fileURL), &excluded_by_path); |
| EXPECT_TRUE(excluded); |
| EXPECT_FALSE(excluded_by_path); |
| } |
| |
| TEST_F(MacUtilTest, NSObjectRetainRelease) { |
| base::scoped_nsobject<NSArray> array( |
| [[NSArray alloc] initWithObjects:@"foo", nil]); |
| EXPECT_EQ(1U, [array retainCount]); |
| |
| NSObjectRetain(array); |
| EXPECT_EQ(2U, [array retainCount]); |
| |
| NSObjectRelease(array); |
| EXPECT_EQ(1U, [array retainCount]); |
| } |
| |
| TEST_F(MacUtilTest, IsOSEllipsis) { |
| int32_t major, minor, bugfix; |
| base::SysInfo::OperatingSystemVersionNumbers(&major, &minor, &bugfix); |
| |
| // The patterns here are: |
| // - FALSE/FALSE/TRUE (it is not the earlier version, it is not "at most" the |
| // earlier version, it is "at least" the earlier version) |
| // - TRUE/TRUE/TRUE (it is the same version, it is "at most" the same version, |
| // it is "at least" the same version) |
| // - FALSE/TRUE/FALSE (it is not the later version, it is "at most" the later |
| // version, it is not "at least" the later version) |
| |
| // TODO(avi): Is there a better way to test this? Maybe with macros? Are |
| // macros a better way to test this? |
| |
| if (major == 10) { |
| if (minor == 9) { |
| EXPECT_TRUE(IsOS10_9()); |
| EXPECT_TRUE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_FALSE(IsOS10_10()); |
| EXPECT_TRUE(IsAtMostOS10_10()); |
| EXPECT_FALSE(IsAtLeastOS10_10()); |
| |
| EXPECT_FALSE(IsOS10_11()); |
| EXPECT_TRUE(IsAtMostOS10_11()); |
| EXPECT_FALSE(IsAtLeastOS10_11()); |
| |
| EXPECT_FALSE(IsOS10_12()); |
| EXPECT_TRUE(IsAtMostOS10_12()); |
| EXPECT_FALSE(IsAtLeastOS10_12()); |
| |
| EXPECT_FALSE(IsOS10_13()); |
| EXPECT_TRUE(IsAtMostOS10_13()); |
| EXPECT_FALSE(IsAtLeastOS10_13()); |
| |
| EXPECT_FALSE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_FALSE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else if (minor == 10) { |
| EXPECT_FALSE(IsOS10_9()); |
| EXPECT_FALSE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_TRUE(IsOS10_10()); |
| EXPECT_TRUE(IsAtMostOS10_10()); |
| EXPECT_TRUE(IsAtLeastOS10_10()); |
| |
| EXPECT_FALSE(IsOS10_11()); |
| EXPECT_TRUE(IsAtMostOS10_11()); |
| EXPECT_FALSE(IsAtLeastOS10_11()); |
| |
| EXPECT_FALSE(IsOS10_12()); |
| EXPECT_TRUE(IsAtMostOS10_12()); |
| EXPECT_FALSE(IsAtLeastOS10_12()); |
| |
| EXPECT_FALSE(IsOS10_13()); |
| EXPECT_TRUE(IsAtMostOS10_13()); |
| EXPECT_FALSE(IsAtLeastOS10_13()); |
| |
| EXPECT_FALSE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_FALSE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else if (minor == 11) { |
| EXPECT_FALSE(IsOS10_9()); |
| EXPECT_FALSE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_FALSE(IsOS10_10()); |
| EXPECT_FALSE(IsAtMostOS10_10()); |
| EXPECT_TRUE(IsAtLeastOS10_10()); |
| |
| EXPECT_TRUE(IsOS10_11()); |
| EXPECT_TRUE(IsAtMostOS10_11()); |
| EXPECT_TRUE(IsAtLeastOS10_11()); |
| |
| EXPECT_FALSE(IsOS10_12()); |
| EXPECT_TRUE(IsAtMostOS10_12()); |
| EXPECT_FALSE(IsAtLeastOS10_12()); |
| |
| EXPECT_FALSE(IsOS10_13()); |
| EXPECT_TRUE(IsAtMostOS10_13()); |
| EXPECT_FALSE(IsAtLeastOS10_13()); |
| |
| EXPECT_FALSE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_FALSE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else if (minor == 12) { |
| EXPECT_FALSE(IsOS10_9()); |
| EXPECT_FALSE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_FALSE(IsOS10_10()); |
| EXPECT_FALSE(IsAtMostOS10_10()); |
| EXPECT_TRUE(IsAtLeastOS10_10()); |
| |
| EXPECT_FALSE(IsOS10_11()); |
| EXPECT_FALSE(IsAtMostOS10_11()); |
| EXPECT_TRUE(IsAtLeastOS10_11()); |
| |
| EXPECT_TRUE(IsOS10_12()); |
| EXPECT_TRUE(IsAtMostOS10_12()); |
| EXPECT_TRUE(IsAtLeastOS10_12()); |
| |
| EXPECT_FALSE(IsOS10_13()); |
| EXPECT_TRUE(IsAtMostOS10_13()); |
| EXPECT_FALSE(IsAtLeastOS10_13()); |
| |
| EXPECT_FALSE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_FALSE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else if (minor == 13) { |
| EXPECT_FALSE(IsOS10_9()); |
| EXPECT_FALSE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_FALSE(IsOS10_10()); |
| EXPECT_FALSE(IsAtMostOS10_10()); |
| EXPECT_TRUE(IsAtLeastOS10_10()); |
| |
| EXPECT_FALSE(IsOS10_11()); |
| EXPECT_FALSE(IsAtMostOS10_11()); |
| EXPECT_TRUE(IsAtLeastOS10_11()); |
| |
| EXPECT_FALSE(IsOS10_12()); |
| EXPECT_FALSE(IsAtMostOS10_12()); |
| EXPECT_TRUE(IsAtLeastOS10_12()); |
| |
| EXPECT_TRUE(IsOS10_13()); |
| EXPECT_TRUE(IsAtMostOS10_13()); |
| EXPECT_TRUE(IsAtLeastOS10_13()); |
| |
| EXPECT_FALSE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_FALSE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else if (minor == 14) { |
| EXPECT_FALSE(IsOS10_9()); |
| EXPECT_FALSE(IsAtMostOS10_9()); |
| EXPECT_TRUE(IsAtLeastOS10_9()); |
| |
| EXPECT_FALSE(IsOS10_10()); |
| EXPECT_FALSE(IsAtMostOS10_10()); |
| EXPECT_TRUE(IsAtLeastOS10_10()); |
| |
| EXPECT_FALSE(IsOS10_11()); |
| EXPECT_FALSE(IsAtMostOS10_11()); |
| EXPECT_TRUE(IsAtLeastOS10_11()); |
| |
| EXPECT_FALSE(IsOS10_12()); |
| EXPECT_FALSE(IsAtMostOS10_12()); |
| EXPECT_TRUE(IsAtLeastOS10_12()); |
| |
| EXPECT_FALSE(IsOS10_13()); |
| EXPECT_FALSE(IsAtMostOS10_13()); |
| EXPECT_TRUE(IsAtLeastOS10_13()); |
| |
| EXPECT_TRUE(IsOS10_14()); |
| EXPECT_TRUE(IsAtMostOS10_14()); |
| EXPECT_TRUE(IsAtLeastOS10_14()); |
| |
| EXPECT_FALSE(IsOSLaterThan10_14_DontCallThis()); |
| } else { |
| // Not nine, ten, eleven, twelve, thirteen, or fourteen. Ah, ah, ah. |
| EXPECT_TRUE(false); |
| } |
| } else { |
| // Not ten. What you gonna do? |
| EXPECT_FALSE(true); |
| } |
| } |
| |
| TEST_F(MacUtilTest, ParseModelIdentifier) { |
| std::string model; |
| int32_t major = 1, minor = 2; |
| |
| EXPECT_FALSE(ParseModelIdentifier("", &model, &major, &minor)); |
| EXPECT_EQ(0U, model.length()); |
| EXPECT_EQ(1, major); |
| EXPECT_EQ(2, minor); |
| EXPECT_FALSE(ParseModelIdentifier("FooBar", &model, &major, &minor)); |
| |
| EXPECT_TRUE(ParseModelIdentifier("MacPro4,1", &model, &major, &minor)); |
| EXPECT_EQ(model, "MacPro"); |
| EXPECT_EQ(4, major); |
| EXPECT_EQ(1, minor); |
| |
| EXPECT_TRUE(ParseModelIdentifier("MacBookPro6,2", &model, &major, &minor)); |
| EXPECT_EQ(model, "MacBookPro"); |
| EXPECT_EQ(6, major); |
| EXPECT_EQ(2, minor); |
| } |
| |
| TEST_F(MacUtilTest, TestRemoveQuarantineAttribute) { |
| ScopedTempDir temp_dir_; |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| FilePath dummy_folder_path = temp_dir_.GetPath().Append("DummyFolder"); |
| ASSERT_TRUE(base::CreateDirectory(dummy_folder_path)); |
| const char* quarantine_str = "0000;4b392bb2;Chromium;|org.chromium.Chromium"; |
| const char* file_path_str = dummy_folder_path.value().c_str(); |
| EXPECT_EQ(0, setxattr(file_path_str, "com.apple.quarantine", |
| quarantine_str, strlen(quarantine_str), 0, 0)); |
| EXPECT_EQ(static_cast<long>(strlen(quarantine_str)), |
| getxattr(file_path_str, "com.apple.quarantine", |
| NULL, 0, 0, 0)); |
| EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); |
| EXPECT_EQ(-1, getxattr(file_path_str, "com.apple.quarantine", NULL, 0, 0, 0)); |
| EXPECT_EQ(ENOATTR, errno); |
| } |
| |
| TEST_F(MacUtilTest, TestRemoveQuarantineAttributeTwice) { |
| ScopedTempDir temp_dir_; |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| FilePath dummy_folder_path = temp_dir_.GetPath().Append("DummyFolder"); |
| const char* file_path_str = dummy_folder_path.value().c_str(); |
| ASSERT_TRUE(base::CreateDirectory(dummy_folder_path)); |
| EXPECT_EQ(-1, getxattr(file_path_str, "com.apple.quarantine", NULL, 0, 0, 0)); |
| // No quarantine attribute to begin with, but RemoveQuarantineAttribute still |
| // succeeds because in the end the folder still doesn't have the quarantine |
| // attribute set. |
| EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); |
| EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); |
| EXPECT_EQ(ENOATTR, errno); |
| } |
| |
| TEST_F(MacUtilTest, TestRemoveQuarantineAttributeNonExistentPath) { |
| ScopedTempDir temp_dir_; |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| FilePath non_existent_path = temp_dir_.GetPath().Append("DummyPath"); |
| ASSERT_FALSE(PathExists(non_existent_path)); |
| EXPECT_FALSE(RemoveQuarantineAttribute(non_existent_path)); |
| } |
| |
| } // namespace |
| |
| } // namespace mac |
| } // namespace base |