I have a script running as Local System which does some stuff, including checking if it's a user logged on, and if yes, it runs a PowerShell snippet to show a toast notification, such as below.
If the PS runs as current user, it works ok. If it runs as LocalSystem, the current user does not see the toast because the output is sent Session 0 (for local system account).
Is it possible to display a toast notification to logged on user, if running as Local System and without requesting user's credentials?
Add-Type -AssemblyName System.Windows.Forms
$global:balloon = New-Object System.Windows.Forms.NotifyIcon
$path = (Get-Process -id $pid).Path
$balloon.Icon = [System.Drawing.Icon]::ExtractAssociatedIcon($path)
$balloon.BalloonTipIcon = [System.Windows.Forms.ToolTipIcon]::Info
$balloon.BalloonTipText = "$Text"
$balloon.BalloonTipTitle = "$Title"
$balloon.Visible = $true
$balloon.ShowBalloonTip($Miliseconds)
Intro
It is possible to display a toast notification to the current active logged on user using PowerShell if running as the SYSTEM user in Session 0 without requesting the user's credentials.
Two solutions to do this follow the "Background Notes"
Background Notes
Please note that this section is written for all viewers of this post, rather than just the original questioner.
SYSTEM
refers to the synonymsNT Authority\SYSTEM
andLocal System
.Many Windows services run as the
SYSTEM
user although others run as users with less privileges such asLOCAL SERVICE
andNETWORK SERVICE
.For each logged in user a Windows session is created numbered from 1 which contains the user's windows.
An additional background session called Session 0 is also created which the Windows services and User Mode Drivers run in. More info at the following link.
Sessions, Desktops, and Windows Stations
All services except the Per-User Services run in Session 0.
If you are using a service for such a script I recommend looking at using one of the two alternatives below:
Create a Session 1 sub-process as already done by some services.
Use the
Windows Task Scheduler
instead to run your main script or notification script in the same session as the current active user. This schedule task can be set to trigger on an event.Note the following security caveat. Scripts in Powershell 5 can be interrupted giving control to a user. For Powershell 6 and later this behaviour is disabled by use of the noninteractive option.
Solution Notes
Two solutions are presented below to solve the original problem. Both use an intervening program to move from Session 0 to Session 1.
Both will flash briefly a PowerShell window which is disconcerting for users and is hard to hide. Some hiding tips are provided in the link below.
How To Run A PowerShell Script Without Display a Window
The following solutions as typed require paths without spaces. Full paths must be used if given. You will have to edit these paths to suit.
Test methodology is included.
The first solution requires the
PSExec.exe
program. It is part ofPSTools
available at the following link. It is also used to test both solutions.PSTools
The second solution requires the
ServiceUI.exe
program. It is part of theMicrosoft Deployment Toolkit (MDT)
available at the following link.Microsoft Deployment Toolkit (MDT)
The
ServiceUI.exe
program is buried in theMDT
install directory as follows.Microsoft Deployment Toolkit\Templates\Distribution\Tools\x64\ServiceUI.exe
I copied it to
E:\Programs\MDT\ServiceUI.exe
to make it simpler to use in PowerShellScripts needed
BalloonTest.ps1
WhoAmISession.ps1
Test Setup
Start a
cmd.exe
window as theAdministrator
.The following command will provide a SYSTEM user Session 0 PowerShell execution environment as shown in the image.
Solution 1: PSExec.exe
To show the Toast notification in
PowerShell
if running as theSYSTEM
user inSession 0
use the following command. This will show the Toast notification if the current active logged in user is usingSession 1
. Some modification will be needed for other sessions.To test start a
"cmd.exe"
window as the"Administrator"
and enter the following command. This will show the Toast notification if the current active logged in user is usingSession 1
as shown in the image.Solution 2: ServiceUI.exe
To show the Toast notification in
PowerShell
if running as theSYSTEM
user inSession 0
use the following command. This will show the Toast notification to the current active logged in user.To test start a
"cmd.exe"
window as the"Administrator"
and enter the following command. This will show the Toast notification to the current active logged in user as shown in the image.