Get CPU % of a particular process using powershell - powershell

I want to get the CPU usage % (not processor time) of a particular process using a powershell command.
Example: (Windows 8 Task Manager)
I want to get that 2.9% with a command.

Here is the correct answer which is support case then you have multiple processs with same name https://stackoverflow.com/a/34844682/483997
# To get the PID of the process (this will give you the first occurrance if multiple matches)
$proc_pid = (get-process "slack").Id[0]
# To match the CPU usage to for example Process Explorer you need to divide by the number of cores
$cpu_cores = (Get-WMIObject Win32_ComputerSystem).NumberOfLogicalProcessors
# This is to find the exact counter path, as you might have multiple processes with the same name
$proc_path = ((Get-Counter "\Process(*)\ID Process").CounterSamples | ? {$_.RawValue -eq $proc_pid}).Path
# We now get the CPU percentage
$prod_percentage_cpu = [Math]::Round(((Get-Counter ($proc_path -replace "\\id process$","\% Processor Time")).CounterSamples.CookedValue) / $cpu_cores)

Get-Process -Name system | select CPU
Get the cpu time at 2 instance as (cpu2-cpu1)/(t2-t1)*100. You will get CPU value in %.
Get-Process -Name system | select CPU # Get the cpu time at 2 instance as (cpu2-cpu1)/(t2-t1)*100. You will get CPU value in %.
$processName = 'OUTLOOK'
$sleep_time = 1 # value in seconds
while (1) {
$CPU_t1 = Get-Process -Name $processName | Select CPU
$CPU_t1_sec = $($CPU_t1.CPU)
#Write-Host "CPU_t1: $($CPU_t1.CPU)"
$date1 = (Get-Date)
sleep -Seconds $sleep_time
$CPU_t2=Get-Process -Name $processName | Select CPU
#Write-Host "CPU_t2: $($CPU_t2.CPU)"
$CPU_t2_sec = $($CPU_t2.CPU)
$date2 = (Get-Date)
$date_diff = $date2 - $date1
$diff_time = $date_diff.seconds
#Write-Host "TimeDiff: $diff_time"
#compute them to get the percentage
$CPU_Utilization = ($CPU_t2_sec - $CPU_t1_sec)/$diff_time
$CPU_Utilization_per = $CPU_Utilization * 100
#Sleep $sleep_time
Clear-Host Write-Host "CPU_Utilization_Per: $CPU_Utilization_per"
#Write-Host "====================="
}

Get-Process -Name PowerShell | Select CPU
Is this what you're looking for?
Or something more monitoring based?
param
(
[String]
[Parameter(Mandatory)]
$Title
)
do
{
$process = Get-Process -Name $Title
$process
Start-Sleep -Seconds 1
} while ($process)

Related

CPU usage in percentage in Powershell

I want the CPU to be displayed as a percentage at the end of the code I wrote. Is there a way?
this is my code
Get-Process svchost | % $_.CPU
Hardly made it this far, only last episode left
Thanks #Santiago for pointing out the issue that it was the reading for the same svchost instance everytime, this is fixed now.
The code below borrows heavily from this thread: Powershell Get a specific process counter with id process
to get the correct reading.
This gets you the Path, the Id and %.
$procResult = Get-Process -Name svchost
$resultObjects = #();
foreach ($result in $procResult) {
$p = $((Get-Counter '\Process(*)\ID Process' -ErrorAction SilentlyContinue).CounterSamples | % { [regex]$a = "^.*\($([regex]::Escape($_.InstanceName))(.*)\).*$";[PSCustomObject]#{InstanceName=$_.InstanceName;PID=$_.CookedValue;InstanceId=$a.Matches($($_.Path)).groups[1].value}})
$target = $p | where { $_.PID -eq $result.Id }
$counterResult = Get-Counter -Counter "\Process($($target.InstanceName+$target.InstanceId))\% Processor Time" -ErrorAction SilentlyContinue
if ($null -eq $counterResult) {
continue;
}
$resultObjects += [PSCustomObject]#{
Path = $result.Path
Id = $result.Id
CPU = $counterResult.CounterSamples[0].CookedValue
}
}
$resultObjects | ft
You won't get the CPU usage in % with your code above. I really do not know why you insist on not changing the structure of your code as the structure of code is irrelevant as long as it is readable and get the job done.

Averaging temperature within a nested loop

I'm trying to write a script that fetches the CPU temperature using OpenHardwareMonitor and will alert if it averages above a certain threshold over a given duration.
Getting the current temperature is working perfectly, but I'm struggling to average them out because on some systems there may be one or more CPU so I can't just log the temperature every minute and divide by how many minutes have passed, below is the best I've got (that doesn't work), please could some kind soul help me see where I'm going wrong?
$tempThreshold = 80 # Will alert if average temperature is above this
$minutes = 3 # Duration in minutes to test for
start-process "$env:SystemRoot\TEMP\OpenHardwareMonitor\openhardwaremonitor.exe"
# Loop each minute
for ($i=1; $i -le $minutes; $i++) {
$arrSensors=#{}
if ((Get-WmiObject -Namespace "Root\OpenHardwareMonitor" -Query "SELECT * FROM Sensor WHERE Sensortype='Temperature'" | ? {$_.Name -match 'CPU Package'}).value) {
Get-WmiObject -Namespace "Root\OpenHardwareMonitor" -Query "SELECT * FROM Sensor WHERE Sensortype='Temperature'" | ? {$_.Name -Match "CPU Package"} | % {$arrSensors[$_.Name]=$_.Value}
} else {
Get-WmiObject -Namespace "Root\OpenHardwareMonitor" -Query "SELECT * FROM Sensor WHERE Sensortype='Temperature'" | ? {$_.Identifier -match 'cpu/'} | % {$arrSensors[$_.Name]=$_.Value}
}
$temps = foreach ($sensor in $arrSensors.getEnumerator()) {
Start-Sleep -Seconds 60
$temp = $temp+$sensor.value
}
}
# Alert if average above threshold
if ($temps/$minutes -gt $tempThreshold) {
Write-Host "Alert function here"
}
i was also looking for monitoring cpu temperature on windows. my backend was grafana with an influxdb. using ohm and wmi was very cpu intense and i found https://github.com/nickbabcock/OhmGraphite#influxdb-configuration. this app is based on ohm but sends the data directly to the influxdb (no expensive wmi-calls needed). maybe this is also of interest for you. using grafana you could average the value and set a threshold there.
but - maybe that is not an option for you so here the script to get the average using the ohm application.
the downside of this script is, that the script runs for 2 minutes - so you won't get the alert every minute (unless you run it twice with a minute appart (which could be easily setup using scheduled tasks).
$tempThreshold = 80 # Will alert if average temperature is above this
$minutes = 3 # Duration in minutes to test for
start-process "$env:SystemRoot\TEMP\OpenHardwareMonitor\openhardwaremonitor.exe"
$counter = 0
$sum = 0
while($counter -lt $minutes) {
$cpuArray = Get-WmiObject -Namespace "Root\OpenHardwareMonitor" -Query "SELECT * FROM Sensor WHERE Sensortype='Temperature'"
if($cpuArray | where name -like 'cpu package*') {
$cpuTempArray = $cpuArray | where name -like 'cpu package*'
} else {
$cpuTempArray = $cpuArray | where name -like 'cpu core*'
}
foreach($cpuTemp in $cpuTempArray){
$sum += $cpuTemp.value
}
$counter++
if($counter -ne $minutes) {
Start-Sleep -Seconds 60
}
}
$average = $sum/($counter*$cpuTempArray.Count)
# Alert if average above threshold
if($average -gt $tempThreshold) {
Write-Host "Alert function here"
}
as mentioned my other answer the other script runs once and IMHO cannot be seen as "monitoring" - as in 'alert whenever the threshold is hit' - like ever minute.
the following script runs in an endless loop. it collects the set amount of data ($minutes) as average of every minutes. you might wanna change this to store the max value of all cpus in the $valueArray and then take the average of this.
$tempThreshold = 80 # Will alert if average temperature is above this
$minutes = 3 # Duration in minutes to test for
Start-Process -FilePath "$env:SystemRoot\TEMP\OpenHardwareMonitor\openhardwaremonitor.exe"
$valueArray = [System.Collections.ArrayList]#()
while($true) {
# keep only #no of values in array specified in $minutes
if($valueArray.Count -gt $minutes) {
# remove oldest entry from array
$valueArray.RemoveAt(0)
}
if($valueArray.Count -eq $minutes) {
$average = ($valueArray | Measure-Object -Sum).Sum/$minutes
# Alert if average above threshold
if($average -gt $tempThreshold) {
Write-Host "Alert function here"
}
}
$sensorArray = Get-WmiObject -Namespace "Root\OpenHardwareMonitor" -Query "SELECT * FROM Sensor WHERE Sensortype='Temperature'"
if($sensorArray | where name -like 'asdf cpu package*') {
$cpuTempArray = $sensorArray | where name -like 'cpu package*'
} else {
$cpuTempArray = $sensorArray | where name -like 'cpu core*'
}
$counter = 0
$sum = 0
foreach($cpuTemp in $cpuTempArray) {
$counter++
$sum += $cpuTemp.value
}
$currentAverageTemp = $sum/$counter
$valueArray.Add($currentAverageTemp)
Start-Sleep -Seconds 60
}

\network adapter(*)\bytes total/sec vs. task manager/performance -powershell

On Windows 10, in task manager/performance it gives the values for 'Receive' and 'Send' for network adapters. I got curious as to how MS does this so I thought of performance counters in powershell. So I cobbled together a quick bit of code to experiment with, put it in a background job, this:
$Computer = hostname;
$sndrecsum = {((get-counter -Counter '\network adapter(*)\Bytes Received/sec' -MaxSamples 30).CounterSamples | measure cookedvalue –Sum).Sum};
$sptstb = New-Object System.Net.WebClient;
$sptstb.DownloadFile('http://client.akamai.com/install/test-objects/10MB.bin', 'Out-Null');
Start-Job -Name sxrxs -scriptblock $sndrecsum -ArgumentList $Computer;
Get-Job -Name sxrxs | Wait-Job; $sndrecsumout = Get-Job -Name sxrxs | Receive-Job; ($sndrecsumout | measure -Maximum).Maximum
here's the issue. The output '($sndrecsumout | measure -Maximum).Maximum' should be getting the maximum value in the array $sndrecsumout. It does do that but the max value in the array is always wayyyy lower than it should be and never matches that shown in task manager/performance for 'Receive' and 'Send' (after conversion to the prevailing units shown in task manager/performance at the same time). So i'm not sure if task manager/performance is using performance counters or if i'm screwing up somewhere.
Anyway, two questions: First, how does task manager/performance get the values for 'Receive' and 'Send'? - and - Two, am I screwing up with the way i'm trying to do this?
Thanks for replies and advice in advance.
ok, think i've got it figured out here, at least a little, maybe. This tracks with the Windows 10 performance monitor for the interface;
$sptst = New-Object System.Net.WebClient; $Computer = hostname; $strlf += $("" | Out-String);
$totx = {
get-counter -Counter '\Network Interface(Intel[R] Ethernet Connection [2] I218-V)\Bytes Total/sec' -MaxSamples 40 | ForEach {[math]::Round((($_.countersamples.cookedvalue | measure -sum).sum / 1Mb), 4)}
};
$recx = {
get-counter -Counter '\network Interface(Intel[R] Ethernet Connection [2] I218-V)\Bytes Received/sec' -MaxSamples 40 | ForEach {[math]::Round((($_.countersamples.cookedvalue | measure -sum).sum / 1Mb), 4)}
};
$sntx = {
get-counter -Counter '\network Interface(Intel[R] Ethernet Connection [2] I218-V)\Bytes Sent/sec' -MaxSamples 40 | ForEach {[math]::Round((($_.countersamples.cookedvalue | measure -sum).sum / 1Mb), 4)}
};
Start-Job -Name xtot -scriptblock $totx -ArgumentList $Computer;
Start-Job -Name xrec -scriptblock $recx -ArgumentList $Computer;
Start-Job -Name xsnt -scriptblock $sntx -ArgumentList $Computer;
$ts = Measure-Command -Expression {$sptst.DownloadFile('http://testmy.net/dl-10MB', 'Out-Null')}; $dxtmex = [math]::Round($ts.TotalSeconds, 4);
Get-Job -Name xtot -HasMoreData $True | Wait-Job ; $outxtot = Get-Job -Name xtot | Receive-Job;
$xthrx = ($outxtot | measure -Maximum).Maximum; $testnuma = $xthrx -match '^[\d\.]+$'; if ($testnuma -eq 'True') {$MBxthrx = $xthrx; $Mbpsxthrx = $xthrx / 0.125};
Get-Job -Name xrec -HasMoreData $True | Wait-Job; $outxrec = Get-Job -Name xrec | Receive-Job;
$xrecx = ($outxrec | measure -Maximum).Maximum ; $testnumb = $xrecx -match '^[\d\.]+$'; if ($testnumb -eq 'True') {$MBxrecx = $xrecx; $Mbpsxrecx = $xrecx / 0.125};
Get-Job -Name xsnt -HasMoreData $True | Wait-Job; $outxsnt = Get-Job -Name xsnt | Receive-Job;
$xsntx = ($outxsnt | measure -Maximum).Maximum ; $testnumb = $xsntx -match '^[\d\.]+$'; if ($testnumb -eq 'True') {$MBxsntx = $xsntx; $Mbpsxsntx = $xsntx / 0.125};
$txtl = '|Throughput = ' + $Mbpsxthrx + ' Mbps (' + $MBxthrx + ' MB/s) [ Test Time = ' + $dxtmex + ' seconds ]';
$rxsx = 'Received (D/L) : ' + $Mbpsxrecx + ' Mbps (' + $MBxrecx + ' MB/s)' + ' / Sent (U/L) : ' + $Mbpsxsntx + ' Mbps (' + $MBxsntx + ' MB/s)';
Return $txtl + $strlf + $rxsx
Edit: removed previous answer - this one is much better and improved. Ignore the labels 'Throughput' - 'Received (D/L)' - 'Sent (U/L)' as just called them something to differentiate while playing around with it so call them anything you wish if desired. 'Throughput' in the windows performance monitor is expressed as a percentage and not like this anyway but the results expressed in 'Throughput' above code when converted for percentage in math (not shown in above code) seem to track with the perf monitor which is why I chose the label 'Throughput' here. Also during this figured out how to get the correct instance name for the interface for use in the counters, I don't show its use (or the code for it) this post, and just put the interface name in above code, to keep strictly on topic but if someone wants to see it and its OK to do i'll post it here also. I'll just probably come back later and post it anyway if its Ok in this post and if not ya'll can remove it if you wish if it would be off topic some. Any improvements/suggestions welcome.
Thank You.

Powershell Shutdown VM Script

I have a script that can shutdown vms based on names entered into an array.
That part works fine, but the next part after the shutdown initiates is supposed to wait for a period and notify how many vms it is waiting to shut down before it moves on to another phase of shutting down other vms.
I am not getting any vms counted out of the array. Here is the code for a particular phase:
$waittime = 5 #Seconds
#Create Phase 1 Array
[Array] $PHASE1 = "TestSvr2008"
# For each of the VMs on the Hyperv hosts that are powered on
Foreach ($VM in ($PHASE1 | %{ Get-VM | Where { $_.State -eq "Running" }})){
# Shutdown the guest cleanly
$VM | Stop-VM -Confirm:$false }
# Set the amount of time to wait before assuming the remaining powered on guests are stuck
$waittime = 120 #Seconds
$Time = (Get-Date).TimeofDay
do {
# Wait for the VMs to be Shutdown cleanly
sleep 1.0
$timeleft = $waittime - ($Newtime.seconds)
$numvms = ($PHASE1 | %{ Get-VM | Where { $_.$VM -eq $PHASE1 }}).Count
Write "Waiting for shutdown of $numvms VMs or until $timeleft seconds"
$Newtime = (Get-Date).TimeofDay - $Time
} until ((#($PHASE1 | %{ Get-VM | Where { $_.$VM -eq $PHASE1 }}).Count) -eq 0 -or ($Newtime).Seconds -ge $waittime)
Thanks
This line in do and until blocks has issues:
$numvms = ($PHASE1 | %{ Get-VM | Where { $_.$VM -eq $PHASE1 }}).Count
$_.$VM is typo, it should be $_.VM
$_.VM -eq $PHASE1 tries to check if VM name is equal to array. Comparisons do not work that way.
ForeEach-Object is unnecessary
Get-Vm accepts array of VM names by pipeline or directly as Name parameter. So you can do this:
$numvms = ($PHASE1 | Get-VM).Count
or
$numvms = (Get-VM -Name $PHASE1).Count

Pipe answer from tasklist to check mem usage of process and kill if exceeding threshold

I found a powershell version that does the job but it uses WMI and thats the process i have issue with, its leaking and hits the 512MB roof and stops working "out of memory".
$processToMonitor = 'wmiprvse.exe'
$threshold = 513MB
Register-WmiEvent -Query "SELECT * FROM __InstanceModificationEvent WITHIN 5 WHERE TargetInstance ISA 'Win32_Process' AND TargetInstance.Name='$processToMonitor' AND TargetInstance.WorkingSetSize > $threshold" -Action { Get-Process -PID $event.SourceEventArgs.NewEvent.TargetInstance.ProcessId | Stop-Process -Force -ErrorAction SilentlyContinue | Out-Null }
So the above will not work when WMI is out of memory, so could i use some other way of doing the same thing?
If i use:
tasklist | find "WmiPrvSE.exe"
It spits out the current WmiPrvSE processes with pid, and memory usage. And ideas of how i can use that?
There is a patch for this WMI leak issue but i cant patch all servers as soon as i want.
try this:
$task = (tasklist | select-string 'Wmiprvse.exe') -split '\s+'
$proc = $task[0] -replace '\.exe'
[int]$mem = $task[-2] -replace '\.'
if ($mem -gt (513MB /1KB) )
{
(get-process $proc).kill()
}
but this is equivalent and preferable ( two session of wmiprvse.exe can exist!):
Get-Process wmiprvse | ? { $_.ws -gt 513MB } | Stop-Process -Force

Categories