1*6777b538SAndroid Build Coastguard Worker// Copyright 2012 The Chromium Authors 2*6777b538SAndroid Build Coastguard Worker// Use of this source code is governed by a BSD-style license that can be 3*6777b538SAndroid Build Coastguard Worker// found in the LICENSE file. 4*6777b538SAndroid Build Coastguard Worker 5*6777b538SAndroid Build Coastguard Worker#include "base/mac/mac_util.h" 6*6777b538SAndroid Build Coastguard Worker 7*6777b538SAndroid Build Coastguard Worker#import <Cocoa/Cocoa.h> 8*6777b538SAndroid Build Coastguard Worker#include <errno.h> 9*6777b538SAndroid Build Coastguard Worker#include <stddef.h> 10*6777b538SAndroid Build Coastguard Worker#include <stdint.h> 11*6777b538SAndroid Build Coastguard Worker#include <sys/xattr.h> 12*6777b538SAndroid Build Coastguard Worker 13*6777b538SAndroid Build Coastguard Worker#include "base/apple/foundation_util.h" 14*6777b538SAndroid Build Coastguard Worker#include "base/apple/scoped_cftyperef.h" 15*6777b538SAndroid Build Coastguard Worker#include "base/files/file_path.h" 16*6777b538SAndroid Build Coastguard Worker#include "base/files/file_util.h" 17*6777b538SAndroid Build Coastguard Worker#include "base/files/scoped_temp_dir.h" 18*6777b538SAndroid Build Coastguard Worker#include "base/system/sys_info.h" 19*6777b538SAndroid Build Coastguard Worker#include "testing/gtest/include/gtest/gtest.h" 20*6777b538SAndroid Build Coastguard Worker#include "testing/platform_test.h" 21*6777b538SAndroid Build Coastguard Worker 22*6777b538SAndroid Build Coastguard Workernamespace base::mac { 23*6777b538SAndroid Build Coastguard Worker 24*6777b538SAndroid Build Coastguard Workernamespace { 25*6777b538SAndroid Build Coastguard Worker 26*6777b538SAndroid Build Coastguard Workerusing MacUtilTest = PlatformTest; 27*6777b538SAndroid Build Coastguard Worker 28*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, GetUserDirectoryTest) { 29*6777b538SAndroid Build Coastguard Worker // Try a few keys, make sure they come back with non-empty paths. 30*6777b538SAndroid Build Coastguard Worker FilePath caches_dir; 31*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(apple::GetUserDirectory(NSCachesDirectory, &caches_dir)); 32*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(caches_dir.empty()); 33*6777b538SAndroid Build Coastguard Worker 34*6777b538SAndroid Build Coastguard Worker FilePath application_support_dir; 35*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(apple::GetUserDirectory(NSApplicationSupportDirectory, 36*6777b538SAndroid Build Coastguard Worker &application_support_dir)); 37*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(application_support_dir.empty()); 38*6777b538SAndroid Build Coastguard Worker 39*6777b538SAndroid Build Coastguard Worker FilePath library_dir; 40*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(apple::GetUserDirectory(NSLibraryDirectory, &library_dir)); 41*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(library_dir.empty()); 42*6777b538SAndroid Build Coastguard Worker} 43*6777b538SAndroid Build Coastguard Worker 44*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestLibraryPath) { 45*6777b538SAndroid Build Coastguard Worker FilePath library_dir = apple::GetUserLibraryPath(); 46*6777b538SAndroid Build Coastguard Worker // Make sure the string isn't empty. 47*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(library_dir.value().empty()); 48*6777b538SAndroid Build Coastguard Worker} 49*6777b538SAndroid Build Coastguard Worker 50*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestGetAppBundlePath) { 51*6777b538SAndroid Build Coastguard Worker FilePath out; 52*6777b538SAndroid Build Coastguard Worker 53*6777b538SAndroid Build Coastguard Worker // Make sure it doesn't crash. 54*6777b538SAndroid Build Coastguard Worker out = apple::GetAppBundlePath(FilePath()); 55*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(out.empty()); 56*6777b538SAndroid Build Coastguard Worker 57*6777b538SAndroid Build Coastguard Worker // Some more invalid inputs. 58*6777b538SAndroid Build Coastguard Worker const char* const invalid_inputs[] = { 59*6777b538SAndroid Build Coastguard Worker "/", "/foo", "foo", "/foo/bar.", "foo/bar.", "/foo/bar./bazquux", 60*6777b538SAndroid Build Coastguard Worker "foo/bar./bazquux", "foo/.app", "//foo", 61*6777b538SAndroid Build Coastguard Worker }; 62*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(invalid_inputs); i++) { 63*6777b538SAndroid Build Coastguard Worker out = apple::GetAppBundlePath(FilePath(invalid_inputs[i])); 64*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(out.empty()) << "loop: " << i; 65*6777b538SAndroid Build Coastguard Worker } 66*6777b538SAndroid Build Coastguard Worker 67*6777b538SAndroid Build Coastguard Worker // Some valid inputs; this and |expected_outputs| should be in sync. 68*6777b538SAndroid Build Coastguard Worker struct { 69*6777b538SAndroid Build Coastguard Worker const char *in; 70*6777b538SAndroid Build Coastguard Worker const char *expected_out; 71*6777b538SAndroid Build Coastguard Worker } valid_inputs[] = { 72*6777b538SAndroid Build Coastguard Worker { "FooBar.app/", "FooBar.app" }, 73*6777b538SAndroid Build Coastguard Worker { "/FooBar.app", "/FooBar.app" }, 74*6777b538SAndroid Build Coastguard Worker { "/FooBar.app/", "/FooBar.app" }, 75*6777b538SAndroid Build Coastguard Worker { "//FooBar.app", "//FooBar.app" }, 76*6777b538SAndroid Build Coastguard Worker { "/Foo/Bar.app", "/Foo/Bar.app" }, 77*6777b538SAndroid Build Coastguard Worker { "/Foo/Bar.app/", "/Foo/Bar.app" }, 78*6777b538SAndroid Build Coastguard Worker { "/F/B.app", "/F/B.app" }, 79*6777b538SAndroid Build Coastguard Worker { "/F/B.app/", "/F/B.app" }, 80*6777b538SAndroid Build Coastguard Worker { "/Foo/Bar.app/baz", "/Foo/Bar.app" }, 81*6777b538SAndroid Build Coastguard Worker { "/Foo/Bar.app/baz/", "/Foo/Bar.app" }, 82*6777b538SAndroid Build Coastguard Worker { "/Foo/Bar.app/baz/quux.app/quuux", "/Foo/Bar.app" }, 83*6777b538SAndroid Build Coastguard Worker { "/Applications/Google Foo.app/bar/Foo Helper.app/quux/Foo Helper", 84*6777b538SAndroid Build Coastguard Worker "/Applications/Google Foo.app" }, 85*6777b538SAndroid Build Coastguard Worker }; 86*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(valid_inputs); i++) { 87*6777b538SAndroid Build Coastguard Worker out = apple::GetAppBundlePath(FilePath(valid_inputs[i].in)); 88*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(out.empty()) << "loop: " << i; 89*6777b538SAndroid Build Coastguard Worker EXPECT_STREQ(valid_inputs[i].expected_out, 90*6777b538SAndroid Build Coastguard Worker out.value().c_str()) << "loop: " << i; 91*6777b538SAndroid Build Coastguard Worker } 92*6777b538SAndroid Build Coastguard Worker} 93*6777b538SAndroid Build Coastguard Worker 94*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestGetInnermostAppBundlePath) { 95*6777b538SAndroid Build Coastguard Worker FilePath out; 96*6777b538SAndroid Build Coastguard Worker 97*6777b538SAndroid Build Coastguard Worker // Make sure it doesn't crash. 98*6777b538SAndroid Build Coastguard Worker out = apple::GetInnermostAppBundlePath(FilePath()); 99*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(out.empty()); 100*6777b538SAndroid Build Coastguard Worker 101*6777b538SAndroid Build Coastguard Worker // Some more invalid inputs. 102*6777b538SAndroid Build Coastguard Worker const char* const invalid_inputs[] = { 103*6777b538SAndroid Build Coastguard Worker "/", 104*6777b538SAndroid Build Coastguard Worker "/foo", 105*6777b538SAndroid Build Coastguard Worker "foo", 106*6777b538SAndroid Build Coastguard Worker "/foo/bar.", 107*6777b538SAndroid Build Coastguard Worker "foo/bar.", 108*6777b538SAndroid Build Coastguard Worker "/foo/bar./bazquux", 109*6777b538SAndroid Build Coastguard Worker "foo/bar./bazquux", 110*6777b538SAndroid Build Coastguard Worker "foo/.app", 111*6777b538SAndroid Build Coastguard Worker "//foo", 112*6777b538SAndroid Build Coastguard Worker }; 113*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(invalid_inputs); i++) { 114*6777b538SAndroid Build Coastguard Worker SCOPED_TRACE(testing::Message() 115*6777b538SAndroid Build Coastguard Worker << "case #" << i << ", input: " << invalid_inputs[i]); 116*6777b538SAndroid Build Coastguard Worker out = apple::GetInnermostAppBundlePath(FilePath(invalid_inputs[i])); 117*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(out.empty()); 118*6777b538SAndroid Build Coastguard Worker } 119*6777b538SAndroid Build Coastguard Worker 120*6777b538SAndroid Build Coastguard Worker // Some valid inputs; this and |expected_outputs| should be in sync. 121*6777b538SAndroid Build Coastguard Worker struct { 122*6777b538SAndroid Build Coastguard Worker const char* in; 123*6777b538SAndroid Build Coastguard Worker const char* expected_out; 124*6777b538SAndroid Build Coastguard Worker } valid_inputs[] = { 125*6777b538SAndroid Build Coastguard Worker {"FooBar.app/", "FooBar.app"}, 126*6777b538SAndroid Build Coastguard Worker {"/FooBar.app", "/FooBar.app"}, 127*6777b538SAndroid Build Coastguard Worker {"/FooBar.app/", "/FooBar.app"}, 128*6777b538SAndroid Build Coastguard Worker {"//FooBar.app", "//FooBar.app"}, 129*6777b538SAndroid Build Coastguard Worker {"/Foo/Bar.app", "/Foo/Bar.app"}, 130*6777b538SAndroid Build Coastguard Worker {"/Foo/Bar.app/", "/Foo/Bar.app"}, 131*6777b538SAndroid Build Coastguard Worker {"/F/B.app", "/F/B.app"}, 132*6777b538SAndroid Build Coastguard Worker {"/F/B.app/", "/F/B.app"}, 133*6777b538SAndroid Build Coastguard Worker {"/Foo/Bar.app/baz", "/Foo/Bar.app"}, 134*6777b538SAndroid Build Coastguard Worker {"/Foo/Bar.app/baz/", "/Foo/Bar.app"}, 135*6777b538SAndroid Build Coastguard Worker {"/Foo/Bar.app/baz/quux.app/quuux", "/Foo/Bar.app/baz/quux.app"}, 136*6777b538SAndroid Build Coastguard Worker {"/Applications/Google Foo.app/bar/Foo Helper.app/quux/Foo Helper", 137*6777b538SAndroid Build Coastguard Worker "/Applications/Google Foo.app/bar/Foo Helper.app"}, 138*6777b538SAndroid Build Coastguard Worker }; 139*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(valid_inputs); i++) { 140*6777b538SAndroid Build Coastguard Worker SCOPED_TRACE(testing::Message() 141*6777b538SAndroid Build Coastguard Worker << "case #" << i << ", input " << valid_inputs[i].in); 142*6777b538SAndroid Build Coastguard Worker out = apple::GetInnermostAppBundlePath(FilePath(valid_inputs[i].in)); 143*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(out.empty()); 144*6777b538SAndroid Build Coastguard Worker EXPECT_STREQ(valid_inputs[i].expected_out, out.value().c_str()); 145*6777b538SAndroid Build Coastguard Worker } 146*6777b538SAndroid Build Coastguard Worker} 147*6777b538SAndroid Build Coastguard Worker 148*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, MacOSVersion) { 149*6777b538SAndroid Build Coastguard Worker int32_t major, minor, bugfix; 150*6777b538SAndroid Build Coastguard Worker base::SysInfo::OperatingSystemVersionNumbers(&major, &minor, &bugfix); 151*6777b538SAndroid Build Coastguard Worker 152*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(major * 1'00'00 + minor * 1'00 + bugfix, MacOSVersion()); 153*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(major, MacOSMajorVersion()); 154*6777b538SAndroid Build Coastguard Worker} 155*6777b538SAndroid Build Coastguard Worker 156*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, ParseOSProductVersion) { 157*6777b538SAndroid Build Coastguard Worker // Various strings in shapes that would be expected to be returned from the 158*6777b538SAndroid Build Coastguard Worker // API that would need to be parsed. 159*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(10'06'02, ParseOSProductVersionForTesting("10.6.2")); 160*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(10'15'00, ParseOSProductVersionForTesting("10.15")); 161*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(13'05'01, ParseOSProductVersionForTesting("13.5.1")); 162*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(14'00'00, ParseOSProductVersionForTesting("14.0")); 163*6777b538SAndroid Build Coastguard Worker 164*6777b538SAndroid Build Coastguard Worker // Various strings in shapes that would not be expected, but that should parse 165*6777b538SAndroid Build Coastguard Worker // without CHECKing. 166*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(13'04'01, ParseOSProductVersionForTesting("13.4.1 (c)")); 167*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(14'00'00, ParseOSProductVersionForTesting("14.0.0")); 168*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(18'00'00, ParseOSProductVersionForTesting("18")); 169*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(18'03'04, ParseOSProductVersionForTesting("18.3.4.3.2.5")); 170*6777b538SAndroid Build Coastguard Worker 171*6777b538SAndroid Build Coastguard Worker // Various strings in shapes that are so unexpected that they should not 172*6777b538SAndroid Build Coastguard Worker // parse. 173*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("Mac OS X 10.0"), 174*6777b538SAndroid Build Coastguard Worker ""); 175*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting(""), ""); 176*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting(" "), ""); 177*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("."), ""); 178*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("10.a.5"), ""); 179*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("१०.१५.७"), ""); 180*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("7.6.1"), ""); 181*6777b538SAndroid Build Coastguard Worker EXPECT_DEATH_IF_SUPPORTED(ParseOSProductVersionForTesting("10.16"), ""); 182*6777b538SAndroid Build Coastguard Worker} 183*6777b538SAndroid Build Coastguard Worker 184*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestRemoveQuarantineAttribute) { 185*6777b538SAndroid Build Coastguard Worker ScopedTempDir temp_dir_; 186*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); 187*6777b538SAndroid Build Coastguard Worker FilePath dummy_folder_path = temp_dir_.GetPath().Append("DummyFolder"); 188*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(base::CreateDirectory(dummy_folder_path)); 189*6777b538SAndroid Build Coastguard Worker const char* quarantine_str = "0000;4b392bb2;Chromium;|org.chromium.Chromium"; 190*6777b538SAndroid Build Coastguard Worker const char* file_path_str = dummy_folder_path.value().c_str(); 191*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(0, setxattr(file_path_str, "com.apple.quarantine", 192*6777b538SAndroid Build Coastguard Worker quarantine_str, strlen(quarantine_str), 0, 0)); 193*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(static_cast<long>(strlen(quarantine_str)), 194*6777b538SAndroid Build Coastguard Worker getxattr(file_path_str, "com.apple.quarantine", /*value=*/nullptr, 195*6777b538SAndroid Build Coastguard Worker /*size=*/0, /*position=*/0, /*options=*/0)); 196*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); 197*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(-1, 198*6777b538SAndroid Build Coastguard Worker getxattr(file_path_str, "com.apple.quarantine", /*value=*/nullptr, 199*6777b538SAndroid Build Coastguard Worker /*size=*/0, /*position=*/0, /*options=*/0)); 200*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(ENOATTR, errno); 201*6777b538SAndroid Build Coastguard Worker} 202*6777b538SAndroid Build Coastguard Worker 203*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestRemoveQuarantineAttributeTwice) { 204*6777b538SAndroid Build Coastguard Worker ScopedTempDir temp_dir_; 205*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); 206*6777b538SAndroid Build Coastguard Worker FilePath dummy_folder_path = temp_dir_.GetPath().Append("DummyFolder"); 207*6777b538SAndroid Build Coastguard Worker const char* file_path_str = dummy_folder_path.value().c_str(); 208*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(base::CreateDirectory(dummy_folder_path)); 209*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(-1, 210*6777b538SAndroid Build Coastguard Worker getxattr(file_path_str, "com.apple.quarantine", /*value=*/nullptr, 211*6777b538SAndroid Build Coastguard Worker /*size=*/0, /*position=*/0, /*options=*/0)); 212*6777b538SAndroid Build Coastguard Worker // No quarantine attribute to begin with, but RemoveQuarantineAttribute still 213*6777b538SAndroid Build Coastguard Worker // succeeds because in the end the folder still doesn't have the quarantine 214*6777b538SAndroid Build Coastguard Worker // attribute set. 215*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); 216*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(RemoveQuarantineAttribute(dummy_folder_path)); 217*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(ENOATTR, errno); 218*6777b538SAndroid Build Coastguard Worker} 219*6777b538SAndroid Build Coastguard Worker 220*6777b538SAndroid Build Coastguard WorkerTEST_F(MacUtilTest, TestRemoveQuarantineAttributeNonExistentPath) { 221*6777b538SAndroid Build Coastguard Worker ScopedTempDir temp_dir_; 222*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); 223*6777b538SAndroid Build Coastguard Worker FilePath non_existent_path = temp_dir_.GetPath().Append("DummyPath"); 224*6777b538SAndroid Build Coastguard Worker ASSERT_FALSE(PathExists(non_existent_path)); 225*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(RemoveQuarantineAttribute(non_existent_path)); 226*6777b538SAndroid Build Coastguard Worker} 227*6777b538SAndroid Build Coastguard Worker 228*6777b538SAndroid Build Coastguard Worker} // namespace 229*6777b538SAndroid Build Coastguard Worker 230*6777b538SAndroid Build Coastguard Worker} // namespace base::mac 231