I am one of those IT Pros who keeps close tabs on system resources. I like to know what is being used and by what. As you might imagine, a cmdlet like Get-Process, is pretty useful to me. One of the things I'm always checking is how much memory Google Chrome is taking. I don't mean to pick on Chrome as I derive great benefit from it. But because I keep it open for days at a time I think system resources get a little carried away. So every once in a while I like to see how many Chrome processes are running and how much they are using.
ManageEngine ADManager Plus - Download Free Trial
Exclusive offer on ADManager Plus for US and UK regions. Claim now!
PS C:\> get-process chrome
Handles NPM(K) PM(K) WS(K) VM(M) CPU(s) Id ProcessName
------- ------ ----- ----- ----- ------ -- -----------
177 26 35564 35104 203 1.61 1976 chrome
197 45 98676 107984 263 57.55 3016 chrome
179 25 26872 20500 187 0.36 5956 chrome
261 33 78456 103908 413 165.72 6260 chrome
187 26 39656 39980 198 21.00 7744 chrome
180 32 88896 85836 263 58.20 7828 chrome
1610 92 119076 178436 492 286.77 8276 chrome
192 38 87136 91124 257 13.98 9140 chrome
193 39 73280 82288 245 122.19 9284 chrome
194 27 57264 50276 204 15.20 9932 chrome
201 32 22936 26780 264 34.91 10076 chrome
Now, what I really want is a total. I can get that for a single property easy enough using Measure-Object.
PS C:\> get-process chrome | measure vm -sum Count : 11 Average : Sum : 3151364096 Maximum : Minimum : Property : VM PS C:\> (get-process chrome | measure vm -sum).sum/1mb 3000.12890625
But ideally I'd like to get totals for all of the properties I see with Get-Process. So I wrote an advanced PowerShell function called Get-ProcessTotal.
#requires -version 3.0
Function Get-ProcessTotal {
<#
.SYNOPSIS
Get total values for multiple instances of same process
.DESCRIPTION
This command is designed to get the total value of common process properties:
Handles,NPM,PM,WS,VM, and CPU. The default behavior is to get the sum total
of these properties for processes that might have multiple instances such as
svchost. Although you can also use a wildcard for the process name and get
totals for a group of processes. As an alternative to the sum, you can get the
average of these properties.
The command writes a custom object to the pipeline and uses a custom format view
so that the output looks like what you get with Get-Process. This command will
create a format ps1xml file on-the-fly called ProcessTotal.Format.ps1xml and
store it in the windowsPowerShell folder under Documents. After the file has been
created this command will simply load it into your PowerShell session.
The command writes a custom object that will include the total number of
processes as well as a property that contains the original processes.
.PARAMETER Name
The name of the process you want to measure. You can use a wildcard. See
the examples.
.EXAMPLE
PS C:\> get-processtotal chrome
Handles NPM(K) PM(K) WS(K) VM(M) CPU(s) ProcessName
------- ------ ----- ----- ----- ------ -----------
4801 612 1120888 1282976 4444 2153.59 chrome
.EXAMPLE
PS C:\> get-processtotal chrome -average | select *
Handles : 300
NPM : 39135
PM : 71778048
WS : 82160640
VM : 291475200
CPU : 134.6279296875
ProcessName : chrome
Computername : JH-WIN81-ENT
Total : 16
Processes : {System.Diagnostics.Process (chrome), System.Diagnostics.Process (chrome),
System.Diagnostics.Process (chrome), System.Diagnostics.Process (chrome)...}
Measure : Average
.EXAMPLE
PS C:\> get-processtotal vm* -comp chi-hvr2
Handles NPM(K) PM(K) WS(K) VM(M) CPU(s) ProcessName
------- ------ ----- ----- ----- ------ -----------
3193 205 110904 166300 873 0 vm*
CPU values are not available for remote computers.
.EXAMPLE
PS C:\> invoke-command {mkdir $env:userprofile\Documents\WindowsPowerShell | out-null} -comp chi-hvr2
PS C:\> invoke-command $(get-item Function:\Get-ProcessTotal).Scriptblock -comp chi-hvr2 -arg VM*
Handles NPM(K) PM(K) WS(K) VM(M) CPU(s) ProcessName PSComputerName
------- ------ ----- ----- ----- ------ ----------- --------------
3198 206 111012 166352 874 10178.39 VM* chi-hvr2
The first command creates the WindowsPowerShell folder on the remote computer.
The second command invokes the Get-ProcessTotal command on the remote computer
which creates the format ps1xml file also remotely.
.NOTES
Last Updated: 2/21/2014
Version : 0.9
Learn more:
PowerShell in Depth: An Administrator's Guide
PowerShell Deep Dives
Learn PowerShell 3 in a Month of Lunches
Learn PowerShell Toolmaking in a Month of Lunches
****************************************************************
* DO NOT USE IN A PRODUCTION ENVIRONMENT UNTIL YOU HAVE TESTED *
* THOROUGHLY IN A LAB ENVIRONMENT. USE AT YOUR OWN RISK. IF *
* YOU DO NOT UNDERSTAND WHAT THIS SCRIPT DOES OR HOW IT WORKS, *
* DO NOT USE IT OUTSIDE OF A SECURE, TEST SETTING. *
****************************************************************
.LINK
https://jdhitsolutions.com/blog/2014/02/theres-sum-thing-happening-here
.LINK
Get-Process
Measure-Object
.INPUTS
String
.OUTPUTS
Custom object
#>
[cmdletbinding(DefaultParameterSetName="Sum")]
Param(
[Parameter(Position=0, Mandatory=$True,
HelpMessage = "Enter the name of a process")]
[ValidateNotNullorEmpty()]
[Alias("processname")]
[string]$Name,
[Parameter(ParameterSetName="Sum")]
[switch]$Sum,
[Parameter(ParameterSetName="Average")]
[switch]$Average,
[ValidateNotNullorEmpty()]
[string]$Computername=$env:COMPUTERNAME
)
Begin {
Write-Verbose -Message "Starting $($MyInvocation.Mycommand)"
Write-Verbose -Message "Querying processes on $computername"
if ($Average) {
$measure="Average"
}
else {
$measure="Sum"
}
Write-Verbose -Message "Measuring the $measure"
#update format data if not found
if (-Not (Get-FormatData -TypeName "My.ProcessTotal")) {
$formatXML=Join-Path -path "$env:USERPROFILE\Documents\WindowsPowerShell" -ChildPath "ProcessTotal.format.ps1xml"
#test if ps1xml file exists and if not create the file
if (-Not (Test-Path -path $formatXMl)) {
#the format xml. The Here string must be left justified to start and finish
$xml=@'
<?xml version="1.0" encoding="utf-8" ?>
<!-- *******************************************************************
This was copied from the DotNetType.format.ps1xml file and modified
to fit my needs.
******************************************************************** -->
<Configuration>
<ViewDefinitions>
<View>
<Name>process</Name>
<ViewSelectedBy>
<TypeName>My.ProcessTotal</TypeName>
</ViewSelectedBy>
<TableControl>
<TableHeaders>
<TableColumnHeader>
<Label>Handles</Label>
<Width>7</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader>
<Label>NPM(K)</Label>
<Width>7</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader>
<Label>PM(K)</Label>
<Width>8</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader>
<Label>WS(K)</Label>
<Width>10</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader>
<Label>VM(M)</Label>
<Width>5</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader>
<Label>CPU(s)</Label>
<Width>8</Width>
<Alignment>right</Alignment>
</TableColumnHeader>
<TableColumnHeader />
</TableHeaders>
<TableRowEntries>
<TableRowEntry>
<TableColumnItems>
<TableColumnItem>
<ScriptBlock>[int]$_.Handles</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<ScriptBlock>[int]($_.NPM / 1024)</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<ScriptBlock>[int]($_.PM / 1024)</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<ScriptBlock>[int]($_.WS / 1024)</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<ScriptBlock>[int]($_.VM / 1048576)</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<ScriptBlock>
if ($_.CPU -ne $())
{
[math]::Round($_.CPU,2)
}
else {
"NA"
}
</ScriptBlock>
</TableColumnItem>
<TableColumnItem>
<PropertyName>ProcessName</PropertyName>
</TableColumnItem>
</TableColumnItems>
</TableRowEntry>
</TableRowEntries>
</TableControl>
</View>
</ViewDefinitions>
</Configuration>
'@
Try {
$xml | Out-File -FilePath $formatXML -Encoding ascii -ErrorAction Stop
}
Catch {
Write-Warning "Failed to create format xml file $formatXML"
Write-Warning $_.Exception.Message
}
} #if not test path for format xml file
#update format data
Write-Verbose -Message "Adding a custom format view from $formatxml."
Update-FormatData -AppendPath $formatXML -ErrorAction Stop
} #if not get-formatdata
#define a hashtable of parameters to splat to Measure-Object so we
#can get either -Sum or -Average
$measureParams=@{$Measure=$True;Property=$Null}
} #begin
Process {
Try {
Write-Verbose -Message "Getting processes for $Name"
$processes = Get-Process $Name -Computername $computername -ErrorAction Stop
}
Catch {
Write-Warning "Failed to find process $name on $computername"
#bail out
Return
}
If ($processes) {
Write-Verbose -message "Defining a hash table of property values"
$properties="Handles","NPM","PM","WS","VM","CPU"
#initialize the ordered hash table
$propertyHash=[ordered]@{}
#add each property to the hash table getting the sum or average
foreach ($property in $properties) {
Write-Verbose -message "Adding $property"
$measureParams.Property = $property
$value = $processes | Measure-Object @measureParams | Select -ExpandProperty $measure
$propertyHash.Add($property,$value)
} #foreach property
Write-Verbose -message "Adding remaining properties"
$propertyHash.Add("ProcessName",$Name)
$propertyHash.Add("Computername",$processes[0].MachineName)
$propertyHash.Add("Total",$processes.count)
$propertyHash.Add("Processes",$processes)
$propertyHash.Add("Measure",$Measure)
#create a custom object
Write-Verbose -Message "Creating object from hashtable"
$obj = New-Object -TypeName psobject -Property $propertyHash
#add a type name
Write-Verbose -Message "Adding type name"
$obj.psobject.TypeNames[0]="My.ProcessTotal"
#write the object to the pipeline
$obj
} #if $processes
} #process
End {
Write-Verbose -Message "Ending $($MyInvocation.Mycommand)"
} #end
} #end function
#add an optional alias
Set-Alias -Name gpt -Value Get-ProcessTotal
The main part of the function gets all instances of a process and then creates a custom object with a sum total for several properties.
$properties="Handles","NPM","PM","WS","VM","CPU"
#initialize the ordered hash table
$propertyHash=[ordered]@{}
#add each property to the hash table getting the sum or average
foreach ($property in $properties) {
Write-Verbose -message "Adding $property"
$measureParams.Property = $property
$value = $processes | Measure-Object @measureParams | Select -ExpandProperty $measure
$propertyHash.Add($property,$value)
} #foreach property
The output also includes the computer name, in case you are querying a remote computer, the total count and also the original process objects in case you need to do something else with them.
$propertyHash.Add("ProcessName",$Name)
$propertyHash.Add("Computername",$processes[0].MachineName)
$propertyHash.Add("Total",$processes.count)
$propertyHash.Add("Processes",$processes)
$propertyHash.Add("Measure",$Measure)
#create a custom object
Write-Verbose -Message "Creating object from hashtable"
$obj = New-Object -TypeName psobject -Property $propertyHash
That could be sufficient, but all the values would be in bytes and the default display would be a list. Instead I wanted default output like I get with Get-Process. So I took advantage of PowerShell's extensible type system and created my own format data xml file. Actually what I did was to find the section in the DotNetTypes.format.ps1xml file for process objects, copied it and tweaked it.
My original idea was to have a separate XML file. If I was building a module, that would be the best choice. But this was a single, stand-alone function so I added some logic to create the XML file on first use and add it to my PowerShell session. The XML content is stored as a Here string in the function.
if (-Not (Get-FormatData -TypeName "My.ProcessTotal")) {
$formatXML=Join-Path -path "$env:USERPROFILE\Documents\WindowsPowerShell" -ChildPath "ProcessTotal.format.ps1xml"
#test if ps1xml file exists and if not create the file
if (-Not (Test-Path -path $formatXMl)) {
#the format xml. The Here string must be left justified to start and finish
$xml=@'
<?xml version="1.0" encoding="utf-8" ?>
...
Try {
$xml | Out-File -FilePath $formatXML -Encoding ascii -ErrorAction Stop
}
Catch {
Write-Warning "Failed to create format xml file $formatXML"
Write-Warning $_.Exception.Message
}
} #if not test path for format xml file
#update format data
Write-Verbose -Message "Adding a custom format view from $formatxml."
Update-FormatData -AppendPath $formatXML -ErrorAction Stop
} #if not get-formatdata
In short, the function first tests to see if there is format data for an object of type My.ProcessTotal. If not, in then tests for the XML file which I'm storing in the WindowsPowerShell folder. If it doesn't exist, the file is created. In any event once the file exists it is then loaded into PowerShell using Update-FormatData.
How does PowerShell know what type to use? I told it. When I defined my custom object I also gave it a new type name.
$obj.psobject.TypeNames[0]="My.ProcessTotal"
Once the function is loaded into my session I can now get a result like this:
PS C:\> Get-ProcessTotal chrome Handles NPM(K) PM(K) WS(K) VM(M) CPU(s) ProcessName ------- ------ ----- ----- ----- ------ ----------- 3560 424 747336 855356 3027 948.83 chrome
The output is just like Get-Process except the values are sums. I also added an option to get an average instead just in case. If you pipe to Get-Member you'll see the new type definition, as well as the other non-default properties.
PS C:\> Get-ProcessTotal chrome | get-member TypeName: My.ProcessTotal Name MemberType Definition ---- ---------- ---------- Equals Method bool Equals(System.Object obj) GetHashCode Method int GetHashCode() GetType Method type GetType() ToString Method string ToString() Computername NoteProperty System.String Computername=JH-WIN81-ENT CPU NoteProperty System.Double CPU=960.484375 Handles NoteProperty System.Double Handles=3569 Measure NoteProperty System.String Measure=Sum NPM NoteProperty System.Double NPM=435088 PM NoteProperty System.Double PM=762679296 Processes NoteProperty System.Object[] Processes=System.Object[] ProcessName NoteProperty System.String ProcessName=chrome Total NoteProperty System.Int32 Total=11 VM NoteProperty System.Double VM=3170050048 WS NoteProperty System.Double WS=879349760
There are some other examples in the comment based help. The script also creates an alias so if you don't want it be sure to comment out the line at the end.
I hope some of you will kick it around and let me know what you think. Or at the very least I hope you picked up a new technique or idea.
