functions/Connect-TibberWebSocket.ps1

function Connect-TibberWebSocket {
    <#
    .Synopsis
        Create a new GraphQL over WebSocket connection.
    .Description
        Calling this function will return a connection object for the established WebSocket connection.
        The object returned is intended to be used with other functions for communication with the endpoint.
    .Example
        $connection = Connect-TibberWebSocket
        Write-Host "New connection created: $($connection.WebSocket.State)"
    .Link
        https://docs.microsoft.com/en-us/dotnet/api/system.net.websockets.clientwebsocket
    .Link
        https://github.com/enisdenjo/graphql-ws/blob/master/PROTOCOL.md
    .Link
        https://developer.tibber.com/docs/guides/calling-api
    #>

    param (
        # Specifies the home Id, e.g. '96a14971-525a-4420-aae9-e5aedaa129ff'.
        [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName)]
        [Alias('Id')]
        [string] $HomeId,

        # Specifies the number of retry attempts if WebSocket initialization fails.
        [ValidateRange(0, [int]::MaxValue)]
        [Alias('Retries', 'MaxRetries')]
        [int] $RetryCount = 5,

        # Specifies the time to wait for WebSocket operations, or -1 to wait indefinitely.
        [Parameter(ValueFromPipelineByPropertyName)]
        [ValidateRange(-1, [int]::MaxValue)]
        [Alias('Timeout')]
        [int] $TimeoutInSeconds = 10
    )

    dynamicParam {
        $dynamicParameters = Invoke-TibberQuery -DynamicParameter
        return $dynamicParameters
    }

    begin {
        # Setup parameters
        $dynamicParametersValues = @{ }
        foreach ($key in $dynamicParameters.Keys) {
            if ($PSBoundParameters[$key]) {
                $dynamicParametersValues[$key] = $PSBoundParameters[$key]
            }
        }
        $querySplat = @{
            WarningAction = 'Ignore'
        } + $dynamicParametersValues
        $querySplat.Force = $true

        # Get the WebSocket subscription URI
        # Note: Will cache access token and user agent (if provided)
        $querySplat.Query = "{ viewer { websocketSubscriptionUrl } }"
        [Uri] $wssUri = (Invoke-TibberQuery @querySplat).viewer.websocketSubscriptionUrl
        Write-Verbose -Message "Got the WebSocket subscription URI: $wssUri"

        # Setup request headers
        $fullUserAgent = Get-UserAgent -UserAgent $script:TibberUserAgentCache
    }

    process {
        $retryCounter = $RetryCount
        while ($retryCounter-- -ge 0) {
            # If this is a retry, release used resources
            if ($webSocket) {
                $webSocket.Dispose()
                $cancellationTokenSource.Dispose()
            }

            # Verify realtime device availability
            if (-Not $dynamicParametersValues.Force) {
                $querySplat.Query = "{ viewer { home(id:`"$HomeId`"){ features { realTimeConsumptionEnabled } } } }"
                $realTimeConsumptionEnabled = (Invoke-TibberQuery @querySplat).viewer.home.features.realTimeConsumptionEnabled
                if (-Not $realTimeConsumptionEnabled) {
                    throw "No realtime device available, please try again after making sure your device is properly connected and reporting data"
                }
                Write-Verbose -Message "Verified realtime device availability: $realTimeConsumptionEnabled"
            }

            # Setup WebSocket for communication
            $webSocket = New-Object Net.WebSockets.ClientWebSocket
            $webSocket.Options.AddSubProtocol('graphql-transport-ws')
            if ($PSVersionTable.PSVersion.Major -gt 5) {
                $webSocket.Options.SetRequestHeader('User-Agent', $fullUserAgent)
            }
            $cancellationTokenSource = New-Object Threading.CancellationTokenSource
            $cancellationToken = $cancellationTokenSource.Token
            $recvBuffer = New-Object ArraySegment[byte] -ArgumentList @(, $([byte[]] @(, 0) * 16384))

            # Connect WebSocket
            $result = $webSocket.ConnectAsync($wssUri, $cancellationToken)
            Wait-WebSocketOp -OperationName 'ConnectAsync' -Result $result -TimeoutInSeconds $TimeoutInSeconds
            Write-Verbose -Message "WebSocket connected to $wssUri [User agent = $fullUserAgent]"

            # Init WebSocket
            $command = @{
                type    = 'connection_init'
                payload = @{
                    token = $script:TibberAccessTokenCache
                }
            }
            if ($PSVersionTable.PSVersion.Major -le 5) {
                $command.payload.userAgent = $fullUserAgent
            }
            $command = $command | ConvertTo-Json -Depth 10
            Write-WebSocket -Data $command -WebSocket $webSocket -CancellationToken $cancellationToken -TimeoutInSeconds $TimeoutInSeconds
            Write-Verbose -Message "Init message sent to: $wssUri [connection_init]"

            # WebSocket init acknowledgement
            # Note: Not using 'Read-WebSocket', need the reslut object for retries
            $result = $webSocket.ReceiveAsync($recvBuffer, $cancellationToken)
            Wait-WebSocketOp -OperationName 'ReceiveAsync' -Result $result -TimeoutInSeconds $TimeoutInSeconds -IgnoreError:$($retryCounter -gt 0)
            Write-Debug -Message "WebSocket status:"
            Write-Debug -Message ($webSocket | Select-Object * | Out-String)
            if ($result.Result.CloseStatus) {
                if ($retryCounter -gt 0) {
                    $retryWaitTime = Get-WebSockerConnectWaitTime -Retry ($RetryCount - $retryCounter)
                    Write-Verbose -Message "Retrying in $retryWaitTime seconds, $retryCounter attempts left"
                    Start-Sleep -Seconds $retryWaitTime
                    continue
                }
            }
            $response = [Text.Encoding]::ASCII.GetString($recvBuffer.Array, 0, $result.Result.Count)
            Write-Verbose -Message "Init response: $response"

            # Output connection object
            return [PSCustomObject]@{
                HomeId                  = $HomeId
                URI                     = $wssUri
                WebSocket               = $webSocket
                CancellationTokenSource = $cancellationTokenSource
                ConnectionAttempts      = $RetryCount - $retryCounter
            }
        }
    }
}