// Copyright (c) 2006-2008 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. // This file contains unit tests for the RestrictedToken. #define _ATL_NO_EXCEPTIONS #include <atlbase.h> #include <atlsecurity.h> #include <vector> #include "sandbox/src/restricted_token.h" #include "sandbox/src/sid.h" #include "testing/gtest/include/gtest/gtest.h" namespace sandbox { // Tests the initializatioin with an invalid token handle. TEST(RestrictedTokenTest, InvalidHandle) { RestrictedToken token; ASSERT_EQ(ERROR_INVALID_HANDLE, token.Init(reinterpret_cast<HANDLE>(0x5555))); } // Tests the initialization with NULL as parameter. TEST(RestrictedTokenTest, DefaultInit) { // Get the current process token. HANDLE token_handle = NULL; ASSERT_TRUE(::OpenProcessToken(::GetCurrentProcess(), TOKEN_ALL_ACCESS, &token_handle)); ASSERT_NE(NULL, reinterpret_cast<ULONG_PTR>(token_handle)); ATL::CAccessToken access_token; access_token.Attach(token_handle); // Create the token using the current token. RestrictedToken token_default; ASSERT_EQ(ERROR_SUCCESS, token_default.Init(NULL)); // Get the handle to the restricted token. HANDLE restricted_token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token_default.GetRestrictedTokenHandle(&restricted_token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(restricted_token_handle); ATL::CSid sid_user_restricted; ATL::CSid sid_user_default; ATL::CSid sid_owner_restricted; ATL::CSid sid_owner_default; ASSERT_TRUE(restricted_token.GetUser(&sid_user_restricted)); ASSERT_TRUE(access_token.GetUser(&sid_user_default)); ASSERT_TRUE(restricted_token.GetOwner(&sid_owner_restricted)); ASSERT_TRUE(access_token.GetOwner(&sid_owner_default)); // Check if both token have the same owner and user. ASSERT_EQ(sid_user_restricted, sid_user_default); ASSERT_EQ(sid_owner_restricted, sid_owner_default); } // Tests the initialization with a custom token as parameter. TEST(RestrictedTokenTest, CustomInit) { // Get the current process token. HANDLE token_handle = NULL; ASSERT_TRUE(::OpenProcessToken(::GetCurrentProcess(), TOKEN_ALL_ACCESS, &token_handle)); ASSERT_NE(NULL, reinterpret_cast<ULONG_PTR>(token_handle)); ATL::CAccessToken access_token; access_token.Attach(token_handle); // Change the primary group. access_token.SetPrimaryGroup(ATL::Sids::World()); // Create the token using the current token. RestrictedToken token; ASSERT_EQ(ERROR_SUCCESS, token.Init(access_token.GetHandle())); // Get the handle to the restricted token. HANDLE restricted_token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&restricted_token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(restricted_token_handle); ATL::CSid sid_restricted; ATL::CSid sid_default; ASSERT_TRUE(restricted_token.GetPrimaryGroup(&sid_restricted)); ASSERT_TRUE(access_token.GetPrimaryGroup(&sid_default)); // Check if both token have the same owner. ASSERT_EQ(sid_restricted, sid_default); } // Verifies that the token created by the object are valid. TEST(RestrictedTokenTest, ResultToken) { RestrictedToken token; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSid(ATL::Sids::World().GetPSID())); HANDLE restricted_token; ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&restricted_token)); ASSERT_TRUE(::IsTokenRestricted(restricted_token)); DWORD length = 0; TOKEN_TYPE type; ASSERT_TRUE(::GetTokenInformation(restricted_token, ::TokenType, &type, sizeof(type), &length)); ASSERT_EQ(type, TokenPrimary); HANDLE impersonation_token; ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandleForImpersonation(&impersonation_token)); ASSERT_TRUE(::IsTokenRestricted(impersonation_token)); ASSERT_TRUE(::GetTokenInformation(impersonation_token, ::TokenType, &type, sizeof(type), &length)); ASSERT_EQ(type, TokenImpersonation); ::CloseHandle(impersonation_token); ::CloseHandle(restricted_token); } // Verifies that the token created has "Restricted" in its default dacl. TEST(RestrictedTokenTest, DefaultDacl) { RestrictedToken token; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSid(ATL::Sids::World().GetPSID())); HANDLE handle; ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(handle); ATL::CDacl dacl; ASSERT_TRUE(restricted_token.GetDefaultDacl(&dacl)); bool restricted_found = false; unsigned int ace_count = dacl.GetAceCount(); for (unsigned int i = 0; i < ace_count ; ++i) { ATL::CSid sid; ACCESS_MASK mask = 0; dacl.GetAclEntry(i, &sid, &mask); if (sid == ATL::Sids::RestrictedCode() && mask == GENERIC_ALL) { restricted_found = true; break; } } ASSERT_TRUE(restricted_found); } // Tests the method "AddSidForDenyOnly". TEST(RestrictedTokenTest, DenySid) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddSidForDenyOnly(Sid(WinWorldSid))); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenGroups groups; ASSERT_TRUE(restricted_token.GetGroups(&groups)); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; groups.GetSidsAndAttributes(&sids, &attributes); for (unsigned int i = 0; i < sids.GetCount(); i++) { if (ATL::Sids::World() == sids[i]) { ASSERT_EQ(SE_GROUP_USE_FOR_DENY_ONLY, attributes[i] & SE_GROUP_USE_FOR_DENY_ONLY); } } } // Tests the method "AddAllSidsForDenyOnly". TEST(RestrictedTokenTest, DenySids) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddAllSidsForDenyOnly(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenGroups groups; ASSERT_TRUE(restricted_token.GetGroups(&groups)); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; groups.GetSidsAndAttributes(&sids, &attributes); // Verify that all sids are really gone. for (unsigned int i = 0; i < sids.GetCount(); i++) { if ((attributes[i] & SE_GROUP_LOGON_ID) == 0 && (attributes[i] & SE_GROUP_INTEGRITY) == 0) { ASSERT_EQ(SE_GROUP_USE_FOR_DENY_ONLY, attributes[i] & SE_GROUP_USE_FOR_DENY_ONLY); } } } // Tests the method "AddAllSidsForDenyOnly" using an exception list. TEST(RestrictedTokenTest, DenySidsException) { RestrictedToken token; HANDLE token_handle = NULL; std::vector<Sid> sids_exception; sids_exception.push_back(Sid(WinWorldSid)); ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddAllSidsForDenyOnly(&sids_exception)); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenGroups groups; ASSERT_TRUE(restricted_token.GetGroups(&groups)); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; groups.GetSidsAndAttributes(&sids, &attributes); // Verify that all sids are really gone. for (unsigned int i = 0; i < sids.GetCount(); i++) { if ((attributes[i] & SE_GROUP_LOGON_ID) == 0 && (attributes[i] & SE_GROUP_INTEGRITY) == 0) { if (ATL::Sids::World() == sids[i]) { ASSERT_EQ(NULL, attributes[i] & SE_GROUP_USE_FOR_DENY_ONLY); } else { ASSERT_EQ(SE_GROUP_USE_FOR_DENY_ONLY, attributes[i] & SE_GROUP_USE_FOR_DENY_ONLY); } } } } // Tests test method AddOwnerSidForDenyOnly. TEST(RestrictedTokenTest, DenyOwnerSid) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddUserSidForDenyOnly()); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenGroups groups; ASSERT_TRUE(restricted_token.GetGroups(&groups)); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; groups.GetSidsAndAttributes(&sids, &attributes); ATL::CSid user_sid; ASSERT_TRUE(restricted_token.GetUser(&user_sid)); for (unsigned int i = 0; i < sids.GetCount(); ++i) { if (user_sid == sids[i]) { ASSERT_EQ(SE_GROUP_USE_FOR_DENY_ONLY, attributes[i] & SE_GROUP_USE_FOR_DENY_ONLY); } } } // Tests the method DeleteAllPrivileges. TEST(RestrictedTokenTest, DeleteAllPrivileges) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.DeleteAllPrivileges(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenPrivileges privileges; ASSERT_TRUE(restricted_token.GetPrivileges(&privileges)); ASSERT_EQ(0, privileges.GetCount()); } // Tests the method DeleteAllPrivileges with an exception list. TEST(RestrictedTokenTest, DeleteAllPrivilegesException) { RestrictedToken token; HANDLE token_handle = NULL; std::vector<std::wstring> exceptions; exceptions.push_back(SE_CHANGE_NOTIFY_NAME); ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.DeleteAllPrivileges(&exceptions)); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenPrivileges privileges; ASSERT_TRUE(restricted_token.GetPrivileges(&privileges)); ATL::CTokenPrivileges::CNames privilege_names; ATL::CTokenPrivileges::CAttributes privilege_name_attributes; privileges.GetNamesAndAttributes(&privilege_names, &privilege_name_attributes); ASSERT_EQ(1, privileges.GetCount()); for (unsigned int i = 0; i < privileges.GetCount(); ++i) { ASSERT_EQ(privilege_names[i], SE_CHANGE_NOTIFY_NAME); } } // Tests the method DeletePrivilege. TEST(RestrictedTokenTest, DeletePrivilege) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.DeletePrivilege(SE_CHANGE_NOTIFY_NAME)); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenPrivileges privileges; ASSERT_TRUE(restricted_token.GetPrivileges(&privileges)); ATL::CTokenPrivileges::CNames privilege_names; ATL::CTokenPrivileges::CAttributes privilege_name_attributes; privileges.GetNamesAndAttributes(&privilege_names, &privilege_name_attributes); for (unsigned int i = 0; i < privileges.GetCount(); ++i) { ASSERT_NE(privilege_names[i], SE_CHANGE_NOTIFY_NAME); } } // Checks if a sid is in the restricting list of the restricted token. // Asserts if it's not the case. If count is a positive number, the number of // elements in the restricting sids list has to be equal. void CheckRestrictingSid(const ATL::CAccessToken &restricted_token, ATL::CSid sid, int count) { DWORD length = 1000; BYTE *memory = new BYTE[1000]; TOKEN_GROUPS *groups = reinterpret_cast<TOKEN_GROUPS*>(memory); ASSERT_TRUE(::GetTokenInformation(restricted_token.GetHandle(), TokenRestrictedSids, groups, length, &length)); ATL::CTokenGroups atl_groups(*groups); delete[] memory; if (count >= 0) ASSERT_EQ(count, atl_groups.GetCount()); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; atl_groups.GetSidsAndAttributes(&sids, &attributes); bool present = false; for (unsigned int i = 0; i < sids.GetCount(); ++i) { if (sids[i] == sid) { present = true; break; } } ASSERT_TRUE(present); } // Tests the method AddRestrictingSid. TEST(RestrictedTokenTest, AddRestrictingSid) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSid(ATL::Sids::World().GetPSID())); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); CheckRestrictingSid(restricted_token, ATL::Sids::World(), 1); } // Tests the method AddRestrictingSidCurrentUser. TEST(RestrictedTokenTest, AddRestrictingSidCurrentUser) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSidCurrentUser()); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CSid user; restricted_token.GetUser(&user); CheckRestrictingSid(restricted_token, user, 1); } // Tests the method AddRestrictingSidLogonSession. TEST(RestrictedTokenTest, AddRestrictingSidLogonSession) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSidLogonSession()); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CSid session; restricted_token.GetLogonSid(&session); CheckRestrictingSid(restricted_token, session, 1); } // Tests adding a lot of restricting sids. TEST(RestrictedTokenTest, AddMultipleRestrictingSids) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSidCurrentUser()); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSidLogonSession()); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSid(ATL::Sids::World().GetPSID())); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CSid session; restricted_token.GetLogonSid(&session); DWORD length = 1000; BYTE *memory = new BYTE[1000]; TOKEN_GROUPS *groups = reinterpret_cast<TOKEN_GROUPS*>(memory); ASSERT_TRUE(::GetTokenInformation(restricted_token.GetHandle(), TokenRestrictedSids, groups, length, &length)); ATL::CTokenGroups atl_groups(*groups); delete[] memory; ASSERT_EQ(3, atl_groups.GetCount()); } // Tests the method "AddRestrictingSidAllSids". TEST(RestrictedTokenTest, AddAllSidToRestrictingSids) { RestrictedToken token; HANDLE token_handle = NULL; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_SUCCESS, token.AddRestrictingSidAllSids()); ASSERT_EQ(ERROR_SUCCESS, token.GetRestrictedTokenHandle(&token_handle)); ATL::CAccessToken restricted_token; restricted_token.Attach(token_handle); ATL::CTokenGroups groups; ASSERT_TRUE(restricted_token.GetGroups(&groups)); ATL::CSid::CSidArray sids; ATL::CAtlArray<DWORD> attributes; groups.GetSidsAndAttributes(&sids, &attributes); // Verify that all group sids are in the restricting sid list. for (unsigned int i = 0; i < sids.GetCount(); i++) { if ((attributes[i] & SE_GROUP_INTEGRITY) == 0) { CheckRestrictingSid(restricted_token, sids[i], -1); } } // Verify that the user is in the restricting sid list. ATL::CSid user; restricted_token.GetUser(&user); CheckRestrictingSid(restricted_token, user, -1); } // Test to be executed only in release because they are triggering DCHECKs. #ifndef _DEBUG // Checks the error code when the object is initialized twice. TEST(RestrictedTokenTest, DoubleInit) { RestrictedToken token; ASSERT_EQ(ERROR_SUCCESS, token.Init(NULL)); ASSERT_EQ(ERROR_ALREADY_INITIALIZED, token.Init(NULL)); } #endif } // namespace sandbox