The PowerShell Select-Object cmdlet enables you to seamlessly cherry-pick objects and properties from an input group of objects. But sometimes you just have to know more. The ExpandProperty is a switch for Select-Object that expands details about a particular property.
Discover the power of the PowerShell Expand Property (-ExpandProperty
) switch in this example-led tutorial.
Today’s sponsor is ScriptRunner, your #1 platform to accelerate your IT automation with PowerShell. They offer a FREE PDF cheat sheet, designed to be your go-to guide for the most important and frequently used Active Directory cmdlets. Download for free
Prerequisites
This tutorial will be a hands-on demonstration. If you’d like to follow along, all you need is a Windows Machine with PowerShell 5.1 and above. This article uses a Windows 10 machine with PowerShell 5.1.
Extracting Property Object Values with PowerShell Expand Property
Log in to your Windows machine and fire up a PowerShell terminal. In this section, you will get acquainted with the Select-Object
cmdlet and its ExpandProperty
switch.
1. Run the Get-Service cmdlet to get a list of all services on your system. Pipe the output to the Select-Object cmdlet to select and display only the name property of each service object, as shown below.
Get-Service | Select-Object -Property Name
You should now see a single column of objects representing a list of names of services on your system, as in the truncated shot below.
data:image/s3,"s3://crabby-images/9c74a/9c74ab9137dcc016307d309fad2c217f60671c5b" alt="Selecting the names of the existing system services"
2. Run the Get-Member cmdlet at the tail end of the previous pipeline to confirm the type of objects that select-object returns as below. Knowing the type of object helps write functions or cmdlets to extend your pipelines.
Get-Service | Select-Object -Property Name | Get-Member
Refer to the screenshot below, and you will notice that though the -Property switch outputs what appears to be a list of strings, those are the Selected list of System.ServiceProcess.ServiceController objects.
data:image/s3,"s3://crabby-images/c1de5/c1de546eba7ab4d02f6c3e8802412c2233a2bda5" alt="Viewing the type of a ServiceController object with Get-Member"
3. Replace the Get-Members cmdlet with mkdir to automatically create a folder for each service. The -first 5 switch limits the number of objects to return. In turn, the mkdir function will only create five folders based on the objects in the pipeline.
Get-Service | Select-Object -Property Name -First 5 | mkdir
You should see an output similar to the screenshot below, indicating the successful attempt.
data:image/s3,"s3://crabby-images/2e2a5/2e2a5304ca9abcce313a8b098780e13a39f4b6e7" alt="Creating a folder for the services on your system"
4. Remove the folders by replacing mkdir with rmdir as in the snippet below. Turn on the -Verbose switch to show verbose output in your terminal.
Get-Service | Select-Object -Property Name -First 5 | rmdir -Verbose
The rmdir cmdlet expects strings and cannot handle the hashtable representation of the ServiceController objects. Your attempt should fail with several errors, as in the screenshot below.
data:image/s3,"s3://crabby-images/dc223/dc223baadf043ce70bc320935b6959d3d75cc20c" alt="Demonstrating that some cmdlets expect string values instead of hashtables"
5. Change the -Property switch to the -ExpandProperty switch as in the following snippet. Doing so allows you to get the value of a property as a string when all you need is a string.
Get-Service | Select-Object -ExpandProperty Name -First 5 | rmdir -Verbose
Combined with rmdir, you should see your attempt to automatically remove the folders succeed without errors, as in the screenshot below.
data:image/s3,"s3://crabby-images/fd44e/fd44e6be9150039ac168df1307b3256264cb3e36" alt="Deleting folders named after the services expanded"
Creating a Set of Selected Objects with Specific Properties
To shorten the code snippets in the sections, create a variable for the first 20 services on your machine as follows.
Run the Get-Service
command, select the first 20
objects via the Select-Object
cmdlet, and store the objects in a variable named $testServices
. Select only the Name
, Status
, and DependentServices
properties as follows.
$testServices = Get-Service | Select-Object -Property Name,Status,DependentServices -First 20
The PowerShell Expand Property switch can only expand one property at a time, in contrast to the Property switch, which can select more than one property simultaneously.
Execute the variable name, $testServices
, to view the structure of its contents.
$testServices
You should see a table displaying the properties of the services you shortlisted, similar to the screenshot below.
data:image/s3,"s3://crabby-images/8a161/8a16172c9f1acdc9354ed9a43a5b611d32e4042c" alt="Displaying the contents of $testServices"
Expanding a Collection Property with PowerShell Expand Property
A PowerShell object can have properties that are collections of objects, also known as collection properties. In this section, you will learn how to use PowerShell Expand Property to expand a collection property to view its members’ properties.
Execute the Select-Object
cmdlet on the $testServices
variable and expand the DependentServices
property of each service.
$testServices | Select-Object -ExpandProperty DependentServices
Since DependentServices
is a collection property, the output is a list of the services in all populated collections, as shown below. Note the duplicates in the list due to concatenating results from the expanded DependentServices
.
data:image/s3,"s3://crabby-images/4fb59/4fb59df78dca27c26722c7b1bb64f0f2073ea430" alt="Expanding the DependentServices property"
Combining Property with PowerShell Expand Property for Categorized Output
Now you have a list of dependent services, but how do you know which service they depend on? In this section, you will learn to use the -Property
switch to categorize an expanded property.
Run the command below to categorize the DependentServices
properties by Name
.
$testServices | Select-Object -ExpandProperty DependentServices -Property Name
Since the dependent services already have a Name
property, the categorization step fails, as shown below.
The error below tells you that the initial list of services (in $testServices) already has a Name property conflicting with a Name property in the expanded DependentServices property.
data:image/s3,"s3://crabby-images/98eaf/98eaf17f4da600d19700f1aac87347d0a18656a3" alt="Attempting to group an expanded property by the parent object name"
One workaround is to create Calculated Properties to rename the identifier property as follows.
Save the previous command output modified to specify a calculated property called Depends On as the identifier property in a variable called $depServ. Depends on will contain the Name value for every service in $testService.
The property name, Depends on
, was arbitrarily chosen in this tutorial and can be any expression of your choice.
$depServ = $testServices | Select-Object -ExpandProperty DependentServices -Property @{name="Depends On"; expr={$_.Name}}
Pass the object in $depServ
through Select-Object
to select dependent service Name
, Status
, and Depends On
properties.
$depServ | Select-Object -Property Name,Status,"Depends On"
You can now tell which services depend on the services in $testServices
, as shown below.
data:image/s3,"s3://crabby-images/45add/45addd98f307ea5689cc1102caaa8cfe05322678" alt="Grouping an expanded property using a calculated property"
Combining Format-List with PowerShell Expand Property for Verbosity
PowerShell may limit which object properties to display, especially in tabular output, and the -ExpandProperty
switch output may experience the same. You could show the expanded properties in a list instead as a workaround.
Run the Get-Process
cmdlet below to retrieve all the modules
the explorer
process uses. Turn on the -ExpandProperty
switch to display the properties of the modules.
Get-Process explorer | select-object -ExpandProperty Modules
The output on your screen should resemble the following screenshot, automatically truncated to display only three properties.
data:image/s3,"s3://crabby-images/2ec89/2ec8984852ee4ade97aeda23fa73b84b5c4dfb18" alt="Viewing the truncated output of an expanded property"
Pipe the objects to the Format-List
cmdlet to view all the values for the module objects in a more manageable list.
Get-Process explorer | select-object -ExpandProperty Modules | Format-List
As shown below, the output lists all properties and their values. You can further save the data to a file or pass it to another cmdlet for further processing.
data:image/s3,"s3://crabby-images/6c45e/6c45ed13bd3e7b01cc1ce23148fbfeb55e9e097d" alt="Converting the output of an expanded property to a list with Format-List"
Conclusion
By successfully getting this far, you have seen how to benefit from the PowerShell Expand Property switch (-ExpandProperty) of the Select-Object cmdlet. PowerShell packs a host of useful cmdlets for many common tasks. You can augment your new knowledge by learning how to display files with PowerShell Cat quickly.