Offline SQL Querying of Active Directory
By Stuart Morgan on 6 April, 2016
Overview
Successful internal penetration tests, simulated attacks and red team exercises are usually predicated by a detailed understanding of the target network. As the majority of large corporations use Windows, this equates to an understanding of the structure of Active Directory. This could include high value users based on their group membership, specific hosts based on usage or operating system or useful information stored within AD itself. There are a number of very good tools and techniques that can extract information from Active Directory which include the various powershell scripts that make up PowerSploit, nishang and some of the post-exploitation modules offered by Metasploit. However, these all require a direct and continued connection to the target domain which could increase the likelihood of discovery and can suffer from a speed penalty, particularly if the connection to the target's network is not direct or latency free. I previously wrote a tool called ADEGrab, which facilitated the extraction of search results from Sysinternals' AD Explorer; the benefit of this tool being that it allowed offline replication of an active directory domain. Although AD Explorer was a huge improvement in terms of offline reconnaissance on a simulated attack, it still lacks some of the flexibility of traditional SQL databases. This post relates to a solution to this problem which can, if normal user credentials and a TCP connection to port 389 on a domain controller is available, copy key parts of Active Directory to an offline SQLite database, effectively allowing normal SQL queries to be executed against it. It allows enumeration of users, computers (including hostname, OS etc.) and groups, parses some LDAP attributes (e.g. sAMAccountType and userAccountControl) to enable you to intuitively search for accounts that, for example, have been locked out or are disabled and calculates nested group membership offline. This tool is an alternative to (and improves on) my metasploit post exploitation module that, at the time of writing, exists as an open pull request; it uses ADSI queries against the domain controller to calculate nested group membership.
Location
The tool is available at https://github.com/stufus/ADOffline; alternatively the repository can be cloned from Github.
git clone https://github.com/stufus/ADOffline.git
Usage Summary
It assumes that you have low privilege (e.g. standard user), access to a domain and that you are able to connect to TCP/389 (LDAP) on a domain controller. On an internal penetration test, you can access a domain controller directly; but, on a simulated attack or red team engagement, use a port forward or SOCKS proxy or equivalent. The basic stages are:
- Use ldapsearch to download the Active Directory structure. At the current time of writing, this script only parses the 'user' object class; this will have the effect of parsing all users, groups and computers on the domain. Ldapsearch will generate an LDIF file which is a text file containing a representation of the AD structure.
- Import this into a SQLite database using adoffline.py.
- Query the SQLite database using a command line tool or a front end.
Database Structure
A user, group and computer is, in essence, the same thing as far as LDAP is concerned. There are some attributes that do not make sense if you are not a computer (e.g. operatingSystem) but at a high level, each of these records is considered as a user at heart. However, there are a number of SQL views that can be used to offer easy identification of computers, groups and users in a more intuitive manner.
LDAP Fields Captured
The table below shows the LDAP attributes that ADOffline currently identifies and parses. The actual fields in the database are discussed later; this is because some of the attributes below are parsed and interpreted in order to make their meaning clearer.
Field | Description |
---|---|
objectClass | The type of object. |
dn | The distinguished name, used as a unique identifier. |
title | The job title of an individual. |
cn | The name that represents an object. This is usually the name of the user, computer or group. |
givenName | Contains the given name (first name) of the user. |
sn | Surname |
description | This is a free-text field which is usually used to store comments relating to this user, computer or group. Sometimes it can have useful information such as default passwords, the purpose of the user or an explanation of how to interact with them. By default, Ben Campbell's metasploit POST module (post/windows/gather/enum_ad_user_comments) works by searching the description field for 'pass' although this is configurable. |
instanceType | This is described by Microsoft as "A bitfield that dictates how the object is instantiated on a particular server. The value of this attribute can differ on different replicas even if the replicas are in sync.". Generally speaking, it seems to be 4. |
displayName | Usually the full name of the user. |
member | Groups can have zero or more 'member' attributes; this indicates that the DN specified is a member of that group. |
memberOf | Groups, users and computers can have zero or more 'memberOf' attributes; this indicates that the current DN is a member of the DN specified. |
name | Seems to be the same as displayName most of the time. |
dNSHostName | For computers, it is the DNS hostname of the computer. |
userAccountControl | Flags that control the behaviour of the user account. See https://msdn.microsoft.com/en-us/library/windows/desktop/ms680832%28v=vs.85%29.aspx for a description, but ADOffline parses them to make them easier to search for. |
badPwdCount | The number of times the user tried to log on to the account using an incorrect password. A value of 0 indicates that the value is unknown. See https://technet.microsoft.com/en-us/library/cc775412%28WS.10%29.aspx for a description; it appears that this is maintained on a per-DC basis and is reset (on the specific DC) when there is a successful login. |
primaryGroupID | The PrimaryGroupID attribute on a user or group object holds the RID of the primary group. Therefore, the user can be considered to be a member of this group even if no 'member' or 'memberOf' attributes are present. |
adminCount | Indicates that a given object has had its ACLs changed to a more secure value by the system because it was a member of one of the administrative groups (directly or transitively). Basically, anyone with adminCount=1 is or was a privileged user of some sort. |
objectSid | The SID. |
sAMAccountName | The username (the logon name used to support clients and servers running earlier versions of the operating system). |
sAMAccountType | This attribute contains information about every account type object. This is parsed by ADOffline. |
objectCategory | "An object class name used to group objects of this or derived classes." |
operatingSystem | The named operating system; only relevant to computers for obvious reasons. |
operatingSystemVersion | The version of the operating system. |
operating SystemServicePack | The identifier of the latest service pack installed |
managedBy | The distinguished name of the user that is assigned to manage this object. Useful as a starting point when looking for managed groups with additional permissions. |
info | One of the general fields available in AD. Sometimes used to store interesting information relating to a user/group/computer. |
comment | One of the general fields available in AD. Sometimes used to store interesting information relating to a user/group/computer. |
department | The department to which the user belongs. |
company | The name of the company. |
homeDirectory | The default home directory location which is mapped to the user's home directory. Useful to identify file servers quickly on the network, but be mindful of DFS (i.e. \\domain\home\user vs \\fileserver\home\user). |
userPrincipalName | Usually the user's e-mail address. |
manager | The user's manager; useful for generating organisational charts. Note that this is different from the 'managedBy' attribute; the manager seems to be for display/organisation chart purposes only. |
Another field that contains the user's e-mail address. | |
groupType | Contains a set of flags that define the type and scope of a group object. |
The SQLite Database
The database is created automatically and, internally, comprises two tables and a number of views.
Tables
The two tables are created using the code below:
CREATE TABLE raw_users ('objectClass','dn','title','comment','cn','sn','description','instanceType','displayName','name','dNSHostName','userAccountControl','badPwdCount','primaryGroupID','adminCount','objectSid','sid','rid','sAMAccountName','sAMAccountType','objectCategory','operatingSystem','operatingSystemServicePack','operatingSystemVersion','managedBy','givenName','info','department','company','homeDirectory','userPrincipalName','manager','mail','groupType');
CREATE TABLE raw_memberof ('dn_group' TEXT NOT NULL,'dn_member' TEXT NOT NULL, PRIMARY KEY('dn_group','dn_member'));
The first table (raw_users) holds the basic information retrieved from LDAP as discussed in the table above. The second table stores any DNs referenced by a member or memberOf attribute. For example, if UserA and UserB are members of GroupX, raw_memberof will contain:
dn_group | dn_member |
---|---|
GroupX | UserA |
GroupX | UserB |
(In reality, it will contain DNs rather than usernames, but this illustrates the point). The idea is that the raw_memberof table can be joined with the raw_users table to be able to determine who is a member of what.
Views
In order to make this easier to interact with, a number of views; a view is essentially a table which is generated at runtime from a SQL query.
Name | Purpose |
---|---|
view_raw_users | This view (which can be treated as a table for the purposes of querying) shows the contents of the raw_users table, but also adds a number of additional columns to split up the userAccountControl and sAMAccountType values. For example, you could search for ADS_UF_LOCKOUT=1 instead of (userAccountControl&00000010). |
view_groups | This will effectively display the contents of the view above, restricting the results to groups only, and adding in the groupType parameter parsing. In effect, this can be used to list all stored information about all groups. |
view_users | Displays the contents of the view_raw_users table, but only shows users (rather than groups and computers). |
view_computers | As above, but only shows computers. |
view_groupmembers | This uses the raw_memberof table to (internally) join the users table with itself. The effect is being able to search by all attributes on a group or its members. The group fields are denoted by the prefix group_ and the member fields are denoted by the prefix member_. For example, 'SELECT member_cn FROM view_groupmembers where group_cn = "Domain Admins"' would display all members of the Domain Admins group, taking into account nested groups. |
view_activegroupusers | This restricts the output of view_groupmembers to users who are not locked and not disabled. The same query as above, but only returning names of users who are active would be 'select member_cn from view_activegroupusers where group_cn = "Domain Admins"' |
This probably looks quite confusing and inefficient, and both are true. However, this design also offers a lot of flexibility; it enables almost any sensible search to be performed offline which is of particular use during simulated attacks. I would encourage you to take the time to get used to this. I have provided a description of each of the fields available in the database below.
Name | Purpose |
objectClass | The type of object. |
dn | The distinguished name (and primary key). |
title | The job title of an individual. |
cn | The name that represents an object. This is usually the name of the user, computer or group. |
givenName | Contains the given name (first name) of the user. |
sn | Surname |
description | This is a free-text field which is usually used to store comments relating to this user, computer or group. Sometimes it can have useful information such as default passwords, the purpose of the user or an explanation of how to interact with them. By default, Ben Campbell's metasploit POST module (post/windows/gather/enum_ad_user_comments) works by searching the description field for 'pass' although this is configurable. |
instanceType | This is described by Microsoft as "A bitfield that dictates how the object is instantiated on a particular server. The value of this attribute can differ on different replicas even if the replicas are in sync.". Generally speaking, it seems to be 4. |
displayName | Usually the full name of the user |
member | Groups can have zero or more 'member' attributes; this indicates that the DN specified is a member of that group. |
memberOf | Groups, users and computers can have zero or more 'memberOf' attributes; this indicates that the current DN is a member of the DN specified. |
name | Seems to be the same as displayName most of the time |
dNSHostName | For computers, it is the DNS hostname of the computer. |
userAccountControl | Flags that control the behaviour of the user account. See https://msdn.microsoft.com/en-us/library/windows/desktop/ms680832%28v=vs.85%29.aspx for a description, but ADOffline parses them to make them easier to search for. |
badPwdCount | The number of times the user tried to log on to the account using an incorrect password. A value of 0 indicates that the value is unknown. See https://technet.microsoft.com/en-us/library/cc775412%28WS.10%29.aspx for a description; it appears that this is maintained on a per-DC basis and is reset (on the specific DC) when there is a successful login. |
primaryGroupID | The PrimaryGroupID attribute on a user or group object holds the RID of the primary group. Therefore, the user can be considered to be a member of this group even if no 'member' or 'memberOf' attributes are present. |
adminCount | Indicates that a given object has had its ACLs changed to a more secure value by the system because it was a member of one of the administrative groups (directly or transitively). Basically, anyone with adminCount=1 is or was a privileged user of some sort. |
objectSid | The SID in binary form, converted to base64. |
sid | The SID expressed in expanded numeric form (e.g. S-1-5-21-xxxxxxxxxx-500) |
rid | The RID (i.e. the last part of the SID) |
sAMAccountName | The username (the logon name used to support clients and servers running earlier versions of the operating system.) |
sAMAccountType | This attribute contains information about every account type object. This is parsed by ADOffline. |
objectCategory | "An object class name used to group objects of this or derived classes." |
operatingSystem | The named operating system; only relevant to computers for obvious reasons. |
operatingSystemVersion | The version of the operating system. |
operatingSystemServicePack | The identifier of the latest service pack installed |
managedBy | The distinguished name of the user that is assigned to manage this object. Useful as a starting point when looking for managed groups with additional permissions. |
info | One of the general fields available in AD. Sometimes used to store interesting information relating to a user/group/computer. |
comment | One of the general fields available in AD. Sometimes used to store interesting information relating to a user/group/computer. |
department | The department to which the user belongs. |
company | The name of the company. |
homeDirectory | The default home directory location which is mapped to the user's home directory. Useful to identify file servers quickly on the network, but be mindful of DFS (i.e. \\domain\home\user vs \\fileserver\home\user). |
userPrincipalName | Usually the user's e-mail address. |
manager | The user's manager; useful for generating organisational charts. Note that this is different from the 'managedBy' attribute; the manager seems to be for display/organisation chart purposes only. |
Another field that contains the user's e-mail address. | |
groupType | Contains a set of flags that define the type and scope of a group object. |
ADS_UF_SCRIPT | If 1, the logon script is executed. |
ADS_UF_ACCOUNTDISABLE | If 1, the user account is disabled. |
ADS_UF_HOMEDIR_REQUIRED | If 1, the home directory is required. |
ADS_UF_LOCKOUT | If 1, the account is currently locked out. |
ADS_UF_PASSWD_NOTREQD | If 1, no password is required. |
ADS_UF_PASSWD_CANT_CHANGE | If 1, the user cannot change the password. |
ADS_UF_ENCRYPTED_TEXT_ PASSWORD_ALLOWED | If 1, the user can send an encrypted password. |
ADS_UF_TEMP_DUPLICATE_ACCOUNT | If 1, this is an account for users whose primary account is in another domain. This account provides user access to this domain, but not to any domain that trusts this domain. Also known as a local user account. |
ADS_UF_NORMAL_ACCOUNT | If 1, this is a default account type that represents a typical user. |
ADS_UF_INTERDOMAIN_ TRUST_ACCOUNT | If 1, this is a permit to trust account for a system domain that trusts other domains. |
ADS_UF_WORKSTATION_TRUST_ACCOUNT | If 1, this is a computer account for a computer that is a member of this domain. |
ADS_UF_SERVER_TRUST_ACCOUNT | If 1, this is a computer account for a system backup domain controller that is a member of this domain. |
ADS_UF_DONT_EXPIRE_PASSWD | If 1, the password for this account will never expire. |
ADS_UF_MNS_LOGON_ACCOUNT | If 1, this is an MNS logon account. |
ADS_UF_SMARTCARD_REQUIRED | If 1, the user must log on using a smart card. |
ADS_UF_TRUSTED_FOR_DELEGATION | If 1, the service account (user or computer account), under which a service runs, is trusted for Kerberos delegation. Any such service can impersonate a client requesting the service. |
ADS_UF_NOT_DELEGATED | If 1, the security context of the user will not be delegated to a service even if the service account is set as trusted for Kerberos delegation. |
ADS_UF_USE_DES_KEY_ONLY | If 1, restrict this principal to use only Data Encryption Standard (DES) encryption types for keys. |
ADS_UF_DONT_REQUIRE_PREAUTH | If 1, this account does not require Kerberos pre-authentication for logon. |
ADS_UF_PASSWORD_EXPIRED | If 1, the user password has expired. This flag is created by the system using data from the Pwd-Last-Set attribute and the domain policy. |
ADS_UF_TRUSTED_TO_ AUTHENTICATE_FOR_DELEGATION | If 1, the account is enabled for delegation. This is a security-sensitive setting; accounts with this option enabled should be strictly controlled. This setting enables a service running under the account to assume a client identity and authenticate as that user to other remote servers on the network. |
SAM_DOMAIN_OBJECT | See https://msdn.microsoft.com/en-us/library/windows/desktop/ms679637%28v=vs.85%29.aspx. If 1, this flag is set. |
SAM_GROUP_OBJECT | If 1, this flag is set (sAMAccountType attribute). |
SAM_NON_SECURITY_GROUP_OBJECT | If 1, this flag is set (sAMAccountType attribute). |
SAM_ALIAS_OBJECT | If 1, this flag is set (sAMAccountType attribute). |
SAM_NON_SECURITY_ALIAS_OBJECT | If 1, this flag is set (sAMAccountType attribute). |
SAM_USER_OBJECT | If 1, this flag is set (sAMAccountType attribute). |
SAM_NORMAL_USER_ACCOUNT | If 1, this flag is set (sAMAccountType attribute). |
SAM_MACHINE_ACCOUNT | If 1, this flag is set (sAMAccountType attribute). |
SAM_TRUST_ACCOUNT | If 1, this flag is set (sAMAccountType attribute). |
SAM_APP_BASIC_GROUP | If 1, this flag is set (sAMAccountType attribute). |
SAM_APP_QUERY_GROUP | If 1, this flag is set (sAMAccountType attribute). |
SAM_ACCOUNT_TYPE_MAX | If 1, this flag is set (sAMAccountType attribute). |
GT_GROUP_CREATED_BY_SYSTEM | If 1, this is a group that is created by the system. |
GT_GROUP_SCOPE_GLOBAL | If 1, this is a group with global scope. |
GT_GROUP_SCOPE_LOCAL | If 1, this is a group with domain local scope. |
GT_GROUP_SCOPE_UNIVERSAL | If 1, this is a group with universal scope. |
GT_GROUP_SAM_APP_BASIC | If 1, this specifies an APP_BASIC group for Windows Server Authorisation Manager. |
GT_GROUP_SAM_APP_QUERY | If 1, this specifies an APP_QUERY group for Windows Server Authorisation Manager. |
GT_GROUP_SECURITY | If 1, this specifies a security group. |
GT_GROUP_DISTRIBUTION | If 1, this specifies a distribution group (this is the inverse of the security group GT_GROUP_SECURITY). I have included it so that distribution groups can be identified more easily. |
Query Examples
Show active members of the Domain Admins group:
sqlite> select member_cn from view_activegroupusers where group_cn = "Domain Admins";
Show the effective groups that 'Stufus' is a member of:
sqlite> select group_cn from view_activegroupusers where member_cn = "Stufus";
Write a list of computer hostnames and operating system information (sorted by OS) to a CSV file:
sqlite> .mode csv
sqlite> .once hosts.csv
sqlite> select cn,dNSHostName,operatingSystem,operatingSystemVersion,operatingSystemServicePack from view_computers order by operatingSystem,operatingSystemVersion,operatingSystemServicePack;
Display a list of users who have 'pass' somewhere in their description or info fields:
sqlite> select cn,description,info FROM view_users WHERE (description LIKE '%pass%' OR info LIKE '%pass%');
Display a list of users who have something in their description, comment, title or info fields:
sqlite> select cn,description,title,comment,info from view_users where (description IS NOT NULL or title IS NOT NULL or comment IS NOT NULL or info IS NOT NULL);
Display a list of computers which have something in their description or info fields:
sqlite> select cn,description,info FROM view_computers WHERE (description IS NOT NULL or info IS NOT NULL);
Display the number of computers running each of the operating systems used in the target's estate:
sqlite> select count(dn),operatingSystem FROM view_computers where ADS_UF_ACCOUNTDISABLE=0 and ADS_UF_LOCKOUT=0 GROUP BY operatingSystem;
Display all of the Windows XP or Windows 2000 hosts, along with their description:
sqlite> select dnsHostName,description,info,operatingSystem from view_computers where operatingSystem LIKE '%Windows%2000%' OR operatingSystem LIKE '%Windows%XP%';
Generate a CSV file suitable for importing into Visio to generate an organisational chart (equivalent to make_csv_orgchart):
sqlite> .mode csv
sqlite> .once orgchart.csv
sqlite> select u.cn as employee_name,u.description as employee_description,u.title as employee_title,m.cn as manager_name from view_users u LEFT JOIN view_users m ON u.manager = m.dn where u.manager IS NOT NULL
Acquiring the data
The easiest way to acquire the LDAP schema is using the 'ldapsearch' command line tool which is part of OpenLDAP. For example:
$ ldapsearch -h -x -D -w -b -E pr=1000/noprompt -o ldif-wrap=no > client.ldif
When the LDAP file has been acquired, its filename is provided as the only parameter to ADOffline.
$ python adoffline.py client.ldif
.mMMMMMm. MMm M WW W WW RRRRR
mMMMMMMMMMMM. MM MM W W W R R
/MMMM- -MM. MM MM W W W R R
/MMM. _ \/ ^ M M M M W W W W RRRR
|M. aRRr /W| M M M M W W W W R R
\/ .. ^^^ wWWW| M M M W W R R
/WW\. .wWWWW/ M M M W W R R
|WWWWWWWWWWW/
.WWWWWW. ADOffline - Convert AD LDAP to SQL
stuart.morgan@mwrinfosecurity.com | @ukstufus
[31/Jan/16 23:31:52] Creating database: /tmp/tmpBQXkFH.20160131233152.ad-ldap.db
[31/Jan/16 23:31:52] Reading LDIF...done
[31/Jan/16 23:31:53] Parsing LDIF...
Reading line 7001808/7001808 (100%)
[31/Jan/16 23:59:24] Calculating chain of ancestry (nested groups)...
Processed user 27385/27385 (100%)
[01/Feb/16 00:27:35] Completed
Users: 27385
Groups: 38334
Computers: 29142
Associations: 2102885
$ ls -lah /tmp/tmpBQXkFH.20160131233152.ad-ldap.db
-rw-r--r-- 1 stuart wheel 1.0G Feb 1 00:27 /tmp/tmpBQXkFH.20160131233152.ad-ldap.db
The main reason why the database is so large is because of the indexes and the replication of data. This database is not designed to be optimised or efficient at the expense of intuitive use, mainly because disk space is cheap.
Interpreting the data
From the command line, the SQLite3 client can be used to query the data. $ sqlite3 /tmp/tmpBQXkFH.20160131233152.ad-ldap.db However, for those who like their GUIs, DB Browser for SQLite works very well. It is an open source, cross platform SQLite query browser.