Active directory - get list of users with password expiring soon

2.7k Views Asked by At

I need to get a list of users from Active directory whose passwords are expiring soon (say in 5 days).

I need to do this by adding a filter to the DirectorySearcher as it will be fastest. I have added the samaccountname pattern to the filter but I can't figure out how to add pwdLastSet to it. Ideally the filter would reduce the user list to only those who fulfill the password expiration criteria.

        using (DirectoryEntry searchRoot = GetXYZAccountOU())
        {
            DirectorySearcher ds = new DirectorySearcher(searchRoot);
            ds.SearchScope = SearchScope.Subtree;

            ds.Filter = "(&" +                                
                            "(samaccountname=XYZ*)"
             + ")";


            SearchResultCollection result = ds.FindAll();

            foreach (SearchResult searchResult in result)
            {
                var de = searchResult.GetDirectoryEntry();
                //long pwdLastSetVal = (long)de.Properties["pwdLastSet"][0];

                //Console.WriteLine(de.Properties["displayName"].Value + ": " + DateTime.FromFileTimeUtc(pwdLastSetVal));
                Console.WriteLine(de.Properties["displayName"].Value);
            }

            Console.Read();
        }

Here XYZ is the starting letters of my users' samaccountname.

If I run this code I can get the displayName and some other attributes but not the pwdLastSet or the computed attribute msDS-UserPasswordExpiryTimeComputed while I can see both of them in the Active directory browser.

1

There are 1 best solutions below

0
On

You have to know in advance how long passwords are valid for, and query the pwdLastSet attribute. But of course the date is stored in a weird format.

Let's assume they're valid for 30 days. Then you can construct the query like this:

var date = DateTime.Now.AddDays(-30).ToFileTime();
var query = $"(&(objectclass=user)(objectcategory=person)(!pwdlastset=0)(pwdlastset<={date})(!userAccountControl:1.2.840.113556.1.4.803:=65536))";

Accounts can be set to never expire passwords, so you have to account for that in your query. The userAccountControl condition does that.

If you don't want to use a magic number in your code, you can look up how long passwords are valid for on the domain by looking at the maxPwdAge attribute at the root of the domain (which is stored in a different, weird format):

var domain = new DirectoryEntry("LDAP://domain.com");
Int64 pwdAge = (Int64) domain.Properties["maxPwdAge"][0];
var maxPwdAge = pwdAge / -864000000000; //convert to days