What is Admin Count?
Before we discuss Admin Count, a little background is needed. AD contains an object called AdminSDHolder. Its purpose is to protect objects. Specifically, objects which are members of administrative groups.
AD objects have an attribute called “Admin Count”. The default value is for most objects. Changing the value to “1”, flags the account as protected by AdminSDHolder.
By adding a user to an administrative AD group. You change the value to “1”. As a result, the user object is subject to stricter ACLs. Such as, disabled permission inheritance. Furthermore, many admins are not aware of this.
Prevention is the best medicine, or so they say. If you are unaware of these conditions and attributes. Please read this article.
Finding Affected Accounts
We have noticed that accounts cannot be managed by the Help Desk. We discover that Admin Count is “1”. As a result, we need to find all affected accounts.
Below is a script which will change for user objects affected. First of all, this script will not make changes. This is has been separated by design.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
| #Set Variable users to get all User objects within OU specified in searchbase
$users = Get-ADUser -ldapfilter “(objectclass=user)” -searchbase “SEARCH”
$CSVPath = "ENTER CSV OUTPATH"
$list = @()
ForEach($user in $users)
{
# Binding the users to DS
$ou = [ADSI](“LDAP://” + $user)
$sec = $ou.psbase.objectSecurity
if ($sec.get_AreAccessRulesProtected()) #If the account is protected. The statement returns true and runs the script block.
{
$list += get-aduser $user.DistinguishedName -Properties "admincount" | select Name,
@{N="AdminCount"; E={$_.AdminCount}}
}
}
$list | Export-Csv $CSVPath -NoTypeInformation
|
Correcting the Problem
In order to correct the problem, we run another script. This script is very close to the first. The reason for two scripts is change control. Our first script doesn’;t contain functionality to make changes. As a result, we lower the chance of mistake.
As you will see, the second script is similar to the first in a number of ways. You run this from a computer joined to the affected domain. Therefore, if you’;re an external party this is to be run on a client side system.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
| $users = Get-ADUser -ldapfilter “(objectclass=user)” -searchbase “SEARCH BASE”
#Get domain values
$domain = Get-ADDomain
$domainPdc = $domain.PDCEmulator
$domainDn = $domain.DistinguishedName
#HashTable to be used for the reset
$replaceAttributeHashTable = New-Object HashTable
$replaceAttributeHashTable.Add("AdminCount",0)
$isProtected = $false ## allows inheritance
$preserveInheritance = $true ## preserve inheritance rules
ForEach($user in $users)
{
# Binding the users to DS
$ou = [ADSI](“LDAP://” + $user)
$sec = $ou.psbase.objectSecurity
if ($sec.get_AreAccessRulesProtected())
{
#Changes AdminCount back to not set
Get-ADuser $user.DistinguishedName -Properties "admincount" | Set-ADUser -Remove $replaceAttributeHashTable -Server $domainPdc
#Change security and commit
$sec.SetAccessRuleProtection($isProtected, $preserveInheritance)
$ou.psbase.commitchanges()
}
}
|
Final Notes
If you do not correct the root cause, Admin Count and security permissions will revert within the hour. You must remove accounts from the administrative group.