As I was looking through my 3 years old code, I noticed an ugly solution (we all do these things, don't we). I needed to get the OU location of each object, so I decided to take the DistinguishedName attribute and drop the name of the object from the beginning of string therefore I end up with the full LDAP formatted path of the object (could have taken the CanonicalName attibute in reverse order and replace '\' with 'cn=' or 'dn=' or 'ou=', but then I would have to lookup each of those elements to figure if they are OUs or containers...etc.)
Let's take an example, the dinstinguishedName of an object is "CN=DroidServer,OU=ChalmunsCantina,OU=MosEisley,DC=tatooine,DC=com", so the LDAP path of the object can be determined by dropping the first part of this string before the first comma which leaves us with: "OU=ChalmunsCantina,OU=MosEisley,DC=tatooine,DC=com".
First attempt - original code in my script
Easy, lets split the string based on commas, put the elements into an array and drop the first element, then join the elements into a string again (now without the cn=objectname piece): $distinguishedName = "CN=DroidServer,OU=ChalmunsCantina,OU=MosEisley,DC=tatooine,DC=com"
$arrDN = New-Object System.Collections.ArrayList
$tmparr = $distinguishedName.Split(",")
$tmparr | %{[void]$arrDN.add($_)}
$arrDN.RemoveAt(0)
$accLocation = [string]::join(",",$arrDN)
$accLocation
This will take 96.5 milliseconds on my machine. |
Second attempt
Let's get rid of the foreach-object (%) when adding elements to $tmpArr and use the .AddRange method of the ArrayList instead - this will just add all elements in one go instead of going through element by element: $distinguishedName = "CN=DroidServer,OU=ChalmunsCantina,OU=MosEisley,DC=tatooine,DC=com"
$arrDN = New-Object System.Collections.ArrayList
$tmparr = $distinguishedName.Split(",")
[void]$arrDN.addrange($tmparr)
$arrDN.RemoveAt(0)
$accLocation = [string]::join(",",$arrDN)
$accLocation
25 milliseconds, not bad, 4 times quicker. |
Third attempt
To see if it can be even quicker, we'll need to "thinking outside the box" and see if there's any simpler solution than working with arrays and instead do this in one step and drop the first bit of the string which we don't need.It's not obvious in PowerShell because the -replace operator does not support the regular expressions which refer only to the first occurrence in a string. What we can do is make it drop all characters which are not commas and they are followed by a comma, that would make sure the "cn=computername," string is dropped and we end up with the full LDAP path of the object:
$distinguishedName = "CN=DroidServer,OU=ChalmunsCantina,OU=MosEisley,DC=tatooine,DC=com"
$accLocation = $distinguishedName -creplace "^[^,]*,",""
$accLocation
Explanation for the regex pattern:
- ^ start of the string
- [^,]* match one or more non-comma characters
- , match a comma character
0.4669 milliseconds! |
t