Posh-Notify.psm1
<#
.synopsis * Posts a message to a Microsoft Office 365 service via webhook .description * Posts a message to a Microsoft Office 365 service via webhook * Not sure of the max character limit, so truncating is disabled * Supports Text and Title via argument. Other Office 365 card attributes can be added via properties in a hash table passed through to the AdditionalProperties argument * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples * No default webhook - these are provided by wrapper functions like Send-TeamsNotification and Send-OutlookNotification .parameter Text * The text to display * Array, so works for array or single string * i.e. will send a table as easily as a line of text * Will concatenate array as new lines, not spaces .parameter Webhook * The Webhook URL of the service to post to .parameter AdditionalProperties * Construct a fancier card via additional options * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples .notes * TODO: * Support \r\n line endings * Implement AdditionalProperties * Author: Ben Renninson * Email: ben@goldensyrupgames.com * From: https://github.com/GSGBen/Posh-Notify #> function Send-Office365Notification { Param ( [Parameter(Position=0,Mandatory=$true,ValueFromPipeline=$true)][string[]]$Text, [Parameter(Position=1,Mandatory=$false)][string]$Title, [Parameter(Position=2,Mandatory=$false)][string]$Webhook, [Parameter(Position=3,Mandatory=$false)][System.Collections.IDictionary]$AdditionalProperties ) #Join each element that may be passed through in the pipeline Process{ #If passed through the pipeline, join to a single array # after making sure there are no empty lines, # and breaking up multi-line strings - the formatting below needs an array of single strings $CombinedArray += $Text -replace '\n\n',"`n `n" -split "`n" } #only send after joining End{ #Specified webhook or fail if ($Webhook) { $FinalWebhook = $Webhook } else { Write-Error "No webhook specified and `$TeamsWebhook not set!" return } #Join combined array into multi-line text. Need to restart the code formatting wraps each time. Also need two spaces in front of the newline character (which should show up as \n to teams, via the json conversion). $JoinString = '``` ' + "`n" + ' ```' $FinalText += $CombinedArray -join $JoinString #construct body as hash table. #Add starting and ending code formatting wraps. Internal ones are above with the array combination $Body = @{ "title" = $Title "text" = '```' + $FinalText + '```' } #Add the addition properties if they were used if ($AdditionalProperties) { $Body = Merge-HashTable $Body $AdditionalProperties } #Convert hashtable to json $BodyAsJson = ConvertTo-Json $Body #Send the text to the webhook Invoke-RestMethod -ContentType "application/json" -Body $BodyAsJson -Method Post -URI $FinalWebhook } } <# .synopsis * Posts a message to a Microsoft Teams channel via webhook .description * Posts a message to a Microsoft Teams channel via webhook * Not sure of the max character limit, so truncating is disabled * Supports Text and Title via argument. Other Office 365 card attributes can be added via properties in a hash table passed through to the AdditionalProperties argument * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples * Selects the webhook to use in the following priority (first most preferred) order, with the aim of easiest use * Webhook argument passed to function * $TeamsWebhook * e.g. set $Global:TeamsWebhook in your powershell profile as a default * if none of the above, it errors. No prompting because of the automation focus .parameter Text * The text to display * Array, so works for array or single string * i.e. will send a table as easily as a line of text * Will concatenate array as new lines, not spaces .parameter Webhook * The Webhook URL of the channel to post to * To get this * right click on the channel and select 'connectors' * edit an existing webhook connector and get the url or * add a new 'webhook' connector them get the url .parameter AdditionalProperties * Construct a fancier card via additional options * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples .notes * TODO: * Implement AdditionalProperties * Author: Ben Renninson * Email: ben@goldensyrupgames.com * From: https://github.com/GSGBen/Posh-Notify #> function Send-TeamsNotification { Param ( [Parameter(Position=0,Mandatory=$true,ValueFromPipeline=$true)][string[]]$Text, [Parameter(Position=1,Mandatory=$false)][string]$Title, [Parameter(Position=2,Mandatory=$false)][string]$Webhook, [Parameter(Position=3,Mandatory=$false)][System.Collections.IDictionary]$AdditionalProperties ) #Specified webhook or global or fail if ($Webhook) { $FinalWebhook = $Webhook } elseif ($TeamsWebhook) { $FinalWebhook = $TeamsWebhook } else { Write-Error "No webhook specified and `$TeamsWebhook not set!" return } Send-Office365Notification -Text $Text -Title $Title -Webhook $FinalWebhook -AdditionalProperties $AdditionalProperties } <# .synopsis * Posts a message to a Microsoft Office 365 Outlook mailbox via webhook .description * Posts a message to a Microsoft Office 365 Outlook mailbox via webhook * Not sure of the max character limit, so truncating is disabled * Supports Text and Title via argument. Other Office 365 card attributes can be added via properties in a hash table passed through to the AdditionalProperties argument * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples * Selects the webhook to use in the following priority (first most preferred) order, with the aim of easiest use * Webhook argument passed to function * $OutlookWebhook * e.g. set $Global:OutlookWebhook in your powershell profile as a default * if none of the above, it errors. No prompting because of the automation focus .parameter Text * The text to display * Array, so works for array or single string * i.e. will send a table as easily as a line of text * Will concatenate array as new lines, not spaces .parameter Webhook * The Webhook URL of the mailbox to post to * To get this * log into https://outlook.office365.com/owa * Select the gear icon in the top right hand corner * manage connectors * add a new 'webhook' connector them get the url .parameter AdditionalProperties * Construct a fancier card via additional options * See https://messagecardplayground.azurewebsites.net/ for the valid attributes, with examples .notes * TODO: * Implement AdditionalProperties * Author: Ben Renninson * Email: ben@goldensyrupgames.com * From: https://github.com/GSGBen/Posh-Notify #> function Send-OutlookNotification { Param ( [Parameter(Position=0,Mandatory=$true,ValueFromPipeline=$true)][string[]]$Text, [Parameter(Position=1,Mandatory=$false)][string]$Title, [Parameter(Position=2,Mandatory=$false)][string]$Webhook, [Parameter(Position=3,Mandatory=$false)][System.Collections.IDictionary]$AdditionalProperties ) #Specified webhook or global or fail if ($Webhook) { $FinalWebhook = $Webhook } elseif ($TeamsWebhook) { $FinalWebhook = $OutlookWebhook } else { Write-Error "No webhook specified and `$OutlookWebhook not set!" return } Send-Office365Notification -Text $Text -Title $Title -Webhook $FinalWebhook -AdditionalProperties $AdditionalProperties } <# .synopsis - Posts a message to a Discord channel via webhook .description - Posts a message to a Discord channel via webhook - Discord webhooks specify both the user posting it and the channel - Max is 2000 chars - By default this function will send the first 2000 characters of the Text input (minus characters required for formatting, prefix and suffix). If you specify -LastChars it'll send the last 2000 - Should support Discord markdown formatting - Uses $Global:DiscordDefaultWebhook if no Webhook is specified .parameter Text - The text to display - Array, so works for array or single string - Will concatenate array as new lines, not spaces .parameter Prefix - Comes before Text. Not truncated at the moment, so don't make this + Suffix anywhere close to 2000 .parameter Suffix - Comes after Text. Not truncated at the moment, so don't make this + Prefix anywhere close to 2000 .parameter Webhook - The Webhook URL of the channel to post to .parameter LastChars - By default this function will send the first 2000 characters (minus characters required for formatting). If you specify -LastChars it'll send the last 2000 - E.g. if you want as much info as possible from some output but definitely want the final result .parameter NoSeparators - By default we separate messages with some basic formatting (e.g. ***). Specify this to not .notes Author: Ben Renninson Email: ben@goldensyrupgames.com From: https://github.com/GSGBen/Posh-Notify #> function Send-DiscordNotification { Param ( [Parameter(Position=0,Mandatory=$true,ValueFromPipeline=$true)][string[]]$Text, [Parameter(Position=1,Mandatory=$false)][string[]]$Prefix, [Parameter(Position=2,Mandatory=$false)][string[]]$Suffix, [Parameter(Position=3,Mandatory=$false)][string]$Webhook, [Parameter(Position=4,Mandatory=$false)][switch]$LastChars = $false, [Parameter(Position=5,Mandatory=$false)][switch]$NoSeparators = $false ) #Specified webhook or global or fail if ($Webhook) { $FinalWebhook = $Webhook } elseif ($Global:DiscordDefaultWebhook) { $FinalWebhook = $Global:DiscordDefaultWebhook } else { Write-Error "No webhook specified and `$Global:DiscordDefaultWebhook not set!" return } #Separate messages because if there are multiple from the same user quickly, it looks like one $StartSeparator = "**---**`n" $EndSeparator = "`n**---**" # Convert the text, which may be an array of strings, to a single string. Join with new lines $FinalText = $Text -join "`r`n" $FinalPrefix = $Prefix -join "`r`n" $FinalSuffix = $Suffix -join "`r`n" #region TRUNCATION-------------------------------------------------------- # Specified by discord $MaxLength = 2000 # Adjust Text input to fit. Truncate to 0 if required, but no lower $UsableLength = [math]::max($MaxLength - $StartSeparator.Length - $EndSeparator.Length - $FinalPrefix.Length - $FinalSuffix.Length,0) # Cut the text if it's greater than 2000, taking into account all other formatting if ($FinalText.Length -gt 2000) { if ($LastChars) #select from end { $FinalText = $FinalText.Substring(($FinalText.Length-$UsableLength),$UsableLength) } else # select from start { $FinalText = $FinalText.Substring(0,$UsableLength) } } else { # It's within the allowed character limit. Do nothing } #endregion TRUNCATION-------------------------------------------------------- # add prefix and suffix $FinalText = $FinalPrefix + $FinalText + $FinalSuffix #Skip separators if specified if ($NoSeparators) { #Don't add them } else { $FinalText = $StartSeparator + $FinalText + $EndSeparator } $Body = @{ "content" = $FinalText } #Convert hashtable to json $BodyAsJson = ConvertTo-Json $Body #Send the text to the webhook Invoke-RestMethod -ContentType "application/json" -Body $BodyAsJson -Method Post -URI $FinalWebhook } <# .synopsis * Plays a notification beep .description * *BEEEEEEEEP* .parameter Pitch * 2000 is kinda high and audible. The default. * Nothing below 190 .parameter Length * 500 is reasonable. The default. .notes * Author: Ben Renninson * Email: ben@goldensyrupgames.com * From: https://github.com/GSGBen/Posh-Notify #> function Beep { Param ( [Parameter(Position=0,Mandatory=$false,ValueFromPipeline=$true)][int]$Pitch = 2000, [Parameter(Position=1,Mandatory=$false)][int]$Length = 500 ) [console]::beep($Pitch,$Length) } <# .synopsis * Plays a notification Arpeggio .description * *BEEEEEEEEP* .notes * Author: Ben Renninson * Email: ben@goldensyrupgames.com * From: https://github.com/GSGBen/Posh-Notify #> function Arpeggio { Param ( [Parameter(Position=0,Mandatory=$false,ValueFromPipeline=$true)][int]$Pitch = 2000, [Parameter(Position=1,Mandatory=$false)][int]$Length = 500 ) Beep 880 300 Beep 1108.73 300 Beep 1318.51 300 Beep 1760 300 } #region----------ALIASES New-Alias -Name Notify-Office365 -Value Send-Office365Notification New-Alias -Name Notify-Teams -Value Send-TeamsNotification New-Alias -Name Notify-Outlook -Value Send-OutlookNotification New-Alias -Name Notify-Discord -Value Send-DiscordNotification Export-ModuleMember -Alias * #endregion |