GitHub

Get-WebSocket

Synopsis

WebSockets in PowerShell.


Description

Get-WebSocket gets a websocket.

This will create a job that connects to a WebSocket and outputs the results.

If the -Watch parameter is provided, will output a continous stream of objects.



Examples

Create a WebSocket job that connects to a WebSocket and outputs the results.

$socketServer = Get-WebSocket -RootUrl "http://localhost:8387/" -HTML "<h1>WebSocket Server</h1>"
$socketClient = Get-WebSocket -SocketUrl "ws://localhost:8387/"
foreach ($n in 1..10) { $socketServer.Send(@{n=Get-Random}) }
$socketClient | Receive-Job -Keep

Get is the default verb, so we can just say WebSocket. -Watch will output a continous stream of objects from the websocket. For example, let’s Watch BlueSky, but just the text

websocket wss://jetstream2.us-west.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -Watch -Maximum 1kb |
    % { 
        $_.commit.record.text
    }

Watch BlueSky, but just the text and spacing

$blueSkySocketUrl = "wss://jetstream2.us-$(
    'east','west'|Get-Random
).bsky.network/subscribe?$(@(
    "wantedCollections=app.bsky.feed.post"
) -join '&')"
websocket $blueSkySocketUrl -Watch | 
    % { Write-Host "$(' ' * (Get-Random -Max 10))$($_.commit.record.text)$($(' ' * (Get-Random -Max 10)))"} -Max 1kb

Watch continuously in a background job.

websocket wss://jetstream2.us-east.bsky.network/subscribe?wantedCollections=app.bsky.feed.post

Watch the first message in -Debug mode.
This allows you to literally debug the WebSocket messages as they are encountered.

websocket wss://jetstream2.us-west.bsky.network/subscribe -QueryParameter @{
    wantedCollections = 'app.bsky.feed.post'
} -Max 1 -Debug

Watch BlueSky, but just the emoji

websocket jetstream2.us-east.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -Tail -Max 1kb |
    Foreach-Object {
        $in = $_
        if ($in.commit.record.text -match '[\p{IsHighSurrogates}\p{IsLowSurrogates}]+') {
            Write-Host $matches.0 -NoNewline
        }
    }

EXAMPLE 7

$emojiPattern = '[\p{IsHighSurrogates}\p{IsLowSurrogates}\p{IsVariationSelectors}\p{IsCombiningHalfMarks}]+)'
websocket wss://jetstream2.us-west.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -Tail |
    Foreach-Object {
        $in = $_
        $spacing = (' ' * (Get-Random -Minimum 0 -Maximum 7))
        if ($in.commit.record.text -match "(?>(?:$emojiPattern|\#\w+)") {
            $match = $matches.0                    
            Write-Host $spacing,$match,$spacing -NoNewline
        }
    }

EXAMPLE 8

websocket wss://jetstream2.us-east.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -Watch |
    Where-Object {
        $_.commit.record.embed.'$type' -eq 'app.bsky.embed.external'
    } |
    Foreach-Object {
        $_.commit.record.embed.external.uri
    }

BlueSky, but just the hashtags

websocket wss://jetstream2.us-west.bsky.network/subscribe -QueryParameter @{
    wantedCollections = 'app.bsky.feed.post'
} -WatchFor @{
    {$webSocketoutput.commit.record.text -match "\#\w+"}={
        $matches.0
    }                
} -Maximum 1kb

BlueSky, but just the hashtags (as links)

websocket wss://jetstream2.us-west.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -WatchFor @{
    {$webSocketoutput.commit.record.text -match "\#\w+"}={
        if ($psStyle.FormatHyperlink) {
            $psStyle.FormatHyperlink($matches.0, "https://bsky.app/search?q=$([Web.HttpUtility]::UrlEncode($matches.0))")
        } else {
            $matches.0
        }
    }
}

EXAMPLE 11

websocket wss://jetstream2.us-west.bsky.network/subscribe?wantedCollections=app.bsky.feed.post -WatchFor @{
    {$args.commit.record.text -match "\#\w+"}={
        $matches.0
    }
    {$args.commit.record.text -match '[\p{IsHighSurrogates}\p{IsLowSurrogates}]+'}={
        $matches.0
    }
}

We can decorate a type returned from a WebSocket, allowing us to add additional properties. For example, let’s add a Tags property to the app.bsky.feed.post type.

$typeName = 'app.bsky.feed.post'
Update-TypeData -TypeName $typeName -MemberName 'Tags' -MemberType ScriptProperty -Value {
    @($this.commit.record.facets.features.tag)
} -Force

# Now, let's get 10kb posts ( this should not take too long )
$somePosts =
    websocket "wss://jetstream2.us-west.bsky.network/subscribe?wantedCollections=$typeName" -PSTypeName $typeName -Maximum 10kb -Watch
$somePosts |
    ? Tags |
    Select -ExpandProperty Tags |
    Group |
    Sort Count -Descending |
    Select -First 10

Parameters

SocketUrl

The WebSocket Uri.

Type Required Position PipelineInput Aliases
[Uri] false 1 true (ByPropertyName) Url
Uri
WebSocketUri
WebSocketUrl

RootUrl

One or more root urls. If these are provided, a WebSocket server will be created with these listener prefixes.

Type Required Position PipelineInput Aliases
[String[]] true named true (ByPropertyName) HostHeader
Host
CNAME
ListenerPrefix
ListenerPrefixes
ListenerUrl

Route

A route table for all requests.

Type Required Position PipelineInput Aliases
[IDictionary] false named true (ByPropertyName) Routes
RouteTable
WebHook
WebHooks

HTML

The Default HTML. This will be displayed when visiting the root url.

Type Required Position PipelineInput Aliases
[String] false named true (ByPropertyName) DefaultHTML
Home
Index
IndexHTML
DefaultPage

PaletteName

The name of the palette to use. This will include the 4bitcss stylesheet.

Type Required Position PipelineInput Aliases
[String] false named true (ByPropertyName) Palette
ColorScheme
ColorPalette

GoogleFont

The Google Font name.

Type Required Position PipelineInput Aliases
[String] false named true (ByPropertyName) FontName

CodeFont

The Google Font name to use for code blocks. (this should be a monospace font)

Type Required Position PipelineInput Aliases
[String] false named true (ByPropertyName) PreFont
CodeFontName
PreFontName

JavaScript

A list of javascript files or urls to include in the content.

Type Required Position PipelineInput
[String[]] false named true (ByPropertyName)

ImportMap

A javascript import map. This allows you to import javascript modules.

Type Required Position PipelineInput Aliases
[IDictionary] false named true (ByPropertyName) ImportsJavaScript
JavaScriptImports
JavaScriptImportMap

QueryParameter

A collection of query parameters. These will be appended onto the -SocketUrl. Multiple values for a single parameter will be passed as multiple parameters.

Type Required Position PipelineInput Aliases
[IDictionary] false named true (ByPropertyName) QueryParameters
Query

Handler

A ScriptBlock that can handle the output of the WebSocket or the Http Request. This may be run in a separate -Runspace or -RunspacePool. The output of the WebSocket or the Context will be passed as an object.

Type Required Position PipelineInput
[ScriptBlock] false named false

ForwardEvent

If set, will forward websocket messages as events. Only events that match -Filter will be forwarded.

Type Required Position PipelineInput Aliases
[Switch] false named true (ByPropertyName) Forward

Variable

Any variables to declare in the WebSocket job. These variables will also be added to the job as properties.

Type Required Position PipelineInput
[IDictionary] false named false

Any Http Headers to include in the WebSocket request or server response.

Type Required Position PipelineInput Aliases
[IDictionary] false named false Headers

Name

The name of the WebSocket job.

Type Required Position PipelineInput
[String] false named false

InitializationScript

The script to run when the WebSocket job starts.

Type Required Position PipelineInput
[ScriptBlock] false named false

BufferSize

The buffer size. Defaults to 16kb.

Type Required Position PipelineInput
[Int32] false named true (ByPropertyName)

Broadcast

If provided, will send an object. If this is a scriptblock, it will be run and the output will be sent.

Type Required Position PipelineInput Aliases
[PSObject] false named false Send

OnConnect

The ScriptBlock to run after connection to a websocket. This can be useful for making any initial requests.

Type Required Position PipelineInput
[ScriptBlock] false named false

OnError

The ScriptBlock to run when an error occurs.

Type Required Position PipelineInput
[ScriptBlock] false named false

OnOutput

The ScriptBlock to run when the WebSocket job outputs an object.

Type Required Position PipelineInput
[ScriptBlock] false named false

OnWarning

The Scriptblock to run when the WebSocket job produces a warning.

Type Required Position PipelineInput
[ScriptBlock] false named false

Authenticate

If provided, will authenticate the WebSocket. Many websockets require an initial authentication handshake after an initial message is received.
This parameter can be either a ScriptBlock or any other object. If it is a ScriptBlock, it will be run with the output of the WebSocket passed as the first argument. This will run after the socket is connected but before any messages are received.

Type Required Position PipelineInput Aliases
[PSObject] false named false Authorize
HelloMessage

Handshake

If provided, will shake hands after the first websocket message is received. This parameter can be either a ScriptBlock or any other object. If it is a ScriptBlock, it will be run with the output of the WebSocket passed as the first argument. This will run after the socket is connected and the first message is received.

Type Required Position PipelineInput Aliases
[PSObject] false named false Identify

Watch

If set, will watch the output of the WebSocket job, outputting results continuously instead of outputting a websocket job.

Type Required Position PipelineInput Aliases
[Switch] false named false Tail

RawText

If set, will output the raw text that comes out of the WebSocket.

Type Required Position PipelineInput Aliases
[Switch] false named false Raw

Binary

If set, will output the raw bytes that come out of the WebSocket.

Type Required Position PipelineInput Aliases
[Switch] false named false RawByte
RawBytes
Bytes
Byte

Force

If set, will force a new job to be created, rather than reusing an existing job.

Type Required Position PipelineInput
[Switch] false named false

SubProtocol

The subprotocol used by the websocket. If not provided, this will default to json.

Type Required Position PipelineInput
[String] false named true (ByPropertyName)

NoSubProtocol

If set, will not set a subprotocol. This will only work with certain websocket servers, but will not work with an HTTP Listener WebSocket.

Type Required Position PipelineInput
[Switch] false named true (ByPropertyName)

Filter

One or more filters to apply to the output of the WebSocket. These can be strings, regexes, scriptblocks, or commands. If they are strings or regexes, they will be applied to the raw text. If they are scriptblocks, they will be applied to the deserialized JSON. These filters will be run within the WebSocket job.

Type Required Position PipelineInput
[PSObject[]] false named true (ByPropertyName)

WatchFor

If set, will watch the output of a WebSocket job for one or more conditions. The conditions are the keys of the dictionary, and can be a regex, a string, or a scriptblock. The values of the dictionary are what will happen when a match is found.

Type Required Position PipelineInput Aliases
[IDictionary] false named true (ByPropertyName) WhereFor
Wherefore

TimeOut

The timeout for the WebSocket connection. If this is provided, after the timeout elapsed, the WebSocket will be closed.

Type Required Position PipelineInput Aliases
[TimeSpan] false named true (ByPropertyName) Lifespan

PSTypeName

If provided, will decorate the objects outputted from a websocket job. This will only decorate objects converted from JSON.

Type Required Position PipelineInput Aliases
[String[]] false named true (ByPropertyName) PSTypeNames
Decorate
Decoration

Maximum

The maximum number of messages to receive before closing the WebSocket.

Type Required Position PipelineInput
[Int64] false named true (ByPropertyName)

ThrottleLimit

The throttle limit used when creating background jobs.

Type Required Position PipelineInput
[Int32] false named true (ByPropertyName)

ConnectionTimeout

The maximum time to wait for a connection to be established. By default, this is 7 seconds.

Type Required Position PipelineInput
[TimeSpan] false named true (ByPropertyName)

Runspace

The Runspace where the handler should run. Runspaces allow you to limit the scope of the handler.

Type Required Position PipelineInput
[Runspace] false named true (ByPropertyName)

RunspacePool

The RunspacePool where the handler should run. RunspacePools allow you to limit the scope of the handler to a pool of runspaces.

Type Required Position PipelineInput Aliases
[RunspacePool] false named true (ByPropertyName) Pool

IncludeTotalCount

Type Required Position PipelineInput
[Switch] false named false

Skip

Type Required Position PipelineInput
[UInt64] false named false

First

Type Required Position PipelineInput
[UInt64] false named false

Syntax

Get-WebSocket [[-SocketUrl] <Uri>] [-QueryParameter <IDictionary>] [-Handler <ScriptBlock>] [-ForwardEvent] [-Variable <IDictionary>] [-Header <IDictionary>] [-Name <String>] [-InitializationScript <ScriptBlock>] [-BufferSize <Int32>] [-Broadcast <PSObject>] [-OnConnect <ScriptBlock>] [-OnError <ScriptBlock>] [-OnOutput <ScriptBlock>] [-OnWarning <ScriptBlock>] [-Authenticate <PSObject>] [-Handshake <PSObject>] [-Watch] [-RawText] [-Binary] [-Force] [-SubProtocol <String>] [-NoSubProtocol] [-Filter <PSObject[]>] [-WatchFor <IDictionary>] [-TimeOut <TimeSpan>] [-PSTypeName <String[]>] [-Maximum <Int64>] [-ThrottleLimit <Int32>] [-ConnectionTimeout <TimeSpan>] [-Runspace <Runspace>] [-RunspacePool <RunspacePool>] [-IncludeTotalCount] [-Skip <UInt64>] [-First <UInt64>] [<CommonParameters>]
Get-WebSocket -RootUrl <String[]> [-Route <IDictionary>] [-HTML <String>] [-PaletteName <String>] [-GoogleFont <String>] [-CodeFont <String>] [-JavaScript <String[]>] [-ImportMap <IDictionary>] [-Handler <ScriptBlock>] [-Variable <IDictionary>] [-Header <IDictionary>] [-Name <String>] [-InitializationScript <ScriptBlock>] [-BufferSize <Int32>] [-Broadcast <PSObject>] [-Force] [-TimeOut <TimeSpan>] [-Maximum <Int64>] [-ThrottleLimit <Int32>] [-Runspace <Runspace>] [-RunspacePool <RunspacePool>] [-IncludeTotalCount] [-Skip <UInt64>] [-First <UInt64>] [<CommonParameters>]