From c1eed492646cae50fc3219d1e2fb25b0142a7247 Mon Sep 17 00:00:00 2001 From: Oliver Date: Mon, 22 May 2023 23:27:55 +1000 Subject: [PATCH] feat: multi-probe UX improvements --- main.go | 43 +++++ probe.go | 10 +- ui/.eslintrc.js | 2 +- ui/src/App.jsx | 344 +++++++++++++++++++++++----------------- ui/src/Header.jsx | 30 ++-- ui/src/JSmpegPlayer.jsx | 37 +++-- ui/src/Salvos.jsx | 159 +++++++++---------- ui/src/imgs.js | 4 +- ui/src/index.js | 4 +- ui/src/index.scss | 7 + ui/src/useMatrix.js | 36 +++-- ui/webpack.config.js | 12 +- 12 files changed, 397 insertions(+), 291 deletions(-) diff --git a/main.go b/main.go index e6baed1..8456c83 100644 --- a/main.go +++ b/main.go @@ -12,6 +12,7 @@ import ( "strconv" "sync" "syscall" + "time" "github.com/gin-gonic/gin" "github.com/google/uuid" @@ -30,6 +31,8 @@ var ( WriteBufferSize: 1024, CheckOrigin: func(r *http.Request) bool { return true }, } + + ProbeStats []*ProbeChannelStatus ) type WebsocketConnection struct { @@ -53,6 +56,12 @@ type DestinationUpdate struct { Source SourceUpdate `json:"source"` } +type ProbeChannelStatus struct { + Id int `json:"id"` + ActiveSource bool `json:"active_source"` + Clients int `json:"clients"` +} + type SourceUpdate struct { Id int `json:"id"` Label string `json:"label"` @@ -92,8 +101,16 @@ func main() { if Config.Probe.Enabled { ProbeHandlers = make([]*ProbeSocketHandler, len(Config.Probe.RouterDestinations)) + ProbeStats = make([]*ProbeChannelStatus, len(Config.Probe.RouterDestinations)) + for i := range ProbeStats { + ProbeStats[i] = &ProbeChannelStatus{ + Id: i, + } + } + for i := range Config.Probe.RouterDestinations { ProbeHandlers[i] = &ProbeSocketHandler{ + Id: i, clients: make(map[*ProbeClient]bool), register: make(chan *ProbeClient), unregister: make(chan *ProbeClient), @@ -129,6 +146,20 @@ func main() { log.Println("Exiting") } +func SendProbeStats() { + payload, _ := json.Marshal(ProbeStats) + update := MatrixWSMessage{ + Type: "probe_stats", + Data: payload, + } + + MatrixWSConnectionsMutex.Lock() + for conn := range MatrixWSConnections { + conn.Socket.WriteJSON(update) + } + MatrixWSConnectionsMutex.Unlock() +} + func serveHTTP() { gin.SetMode(gin.ReleaseMode) @@ -255,6 +286,12 @@ func HandleMatrixWS(c *gin.Context) { MatrixWSConnectionsMutex.Unlock() defer connection.Socket.Close() + + go func() { + time.Sleep(500 * time.Millisecond) + SendProbeStats() + }() + for { _, messageBytes, err := connection.Socket.ReadMessage() if err != nil { @@ -300,6 +337,9 @@ func HandleProbeStream(c *gin.Context) { log.Printf("stream for probe %d connected from %s", index, c.RemoteIP()) + ProbeStats[index].ActiveSource = true + SendProbeStats() + for { data, err := ioutil.ReadAll(io.LimitReader(c.Request.Body, 1024)) if err != nil || len(data) == 0 { @@ -309,5 +349,8 @@ func HandleProbeStream(c *gin.Context) { ProbeHandlers[index].BroadcastData(&data) } + ProbeStats[index].ActiveSource = false + SendProbeStats() + log.Printf("stream for probe %d disconnected from %s", index, c.RemoteIP()) } diff --git a/probe.go b/probe.go index e8ef415..46ddf3a 100644 --- a/probe.go +++ b/probe.go @@ -87,6 +87,7 @@ func (c *ProbeClient) Run() { } type ProbeSocketHandler struct { + Id int clients map[*ProbeClient]bool // *client -> is connected (true/false) register chan *ProbeClient unregister chan *ProbeClient @@ -102,14 +103,18 @@ func (h *ProbeSocketHandler) Run() { select { case client := <-h.register: h.clients[client] = true - log.Printf("New client registered. Total: %d\n", len(h.clients)) + log.Printf("[probe %d] client connected. active clients: %d\n", h.Id, len(h.clients)) + ProbeStats[h.Id].Clients = len(h.clients) + SendProbeStats() case client := <-h.unregister: _, ok := h.clients[client] if ok { delete(h.clients, client) } - log.Printf("Client unregistered. Total: %d\n", len(h.clients)) + log.Printf("[probe %d] client disconnected. active clients: %d\n", h.Id, len(h.clients)) + ProbeStats[h.Id].Clients = len(h.clients) + SendProbeStats() case data := <-h.broadcast: h.BroadcastData(data) @@ -132,7 +137,6 @@ func (h *ProbeSocketHandler) ServeWS(c *gin.Context) { return } - log.Println("New client connected") client := NewProbeClient(ws, h.unregister) h.register <- client diff --git a/ui/.eslintrc.js b/ui/.eslintrc.js index 28b7f9e..df94dfe 100644 --- a/ui/.eslintrc.js +++ b/ui/.eslintrc.js @@ -26,4 +26,4 @@ module.exports = { { allowAsProps: true }, ], }, -}; \ No newline at end of file +}; diff --git a/ui/src/App.jsx b/ui/src/App.jsx index f5d9884..d6f6b51 100644 --- a/ui/src/App.jsx +++ b/ui/src/App.jsx @@ -1,105 +1,129 @@ import React, { useState, useEffect } from 'react'; -import useAxios from 'axios-hooks' +import useAxios from 'axios-hooks'; import Favicon from 'react-favicon'; -import { +import { blue, gray, - purple + purple, } from '@carbon/colors'; -import { +import { Button, Column, Content, Grid, Modal, - Row + Row, } from '@carbon/react'; -import { +import { PortInput, PortOutput, Maximize, QueryQueue, - ChooseItem + ChooseItem, } from '@carbon/icons-react'; import Header from './Header.jsx'; import Salvos from './Salvos.jsx'; -import useMatrix from './useMatrix' -import JSmpegPlayer from './JSmpegPlayer.jsx' +import useMatrix from './useMatrix'; +import JSmpegPlayer from './JSmpegPlayer.jsx'; -import imgs from './imgs' +import imgs from './imgs'; const App = function App() { const [{ data: config, loading: configLoading, error: configError }] = useAxios( - '/v1/config' - ) + '/v1/config', + ); - const {matrix, loading: matrixLoading, error: matrixError, route} = useMatrix() - const [selectedProbe, setSelectedProbe] = useState(0) - const [selectedDestination, setSelectedDestination] = useState(1) - const [ProbeSOTRouting, setProbeSODRouting] = useState(false) - const [probeFullscreen, setProbeFullscreen] = useState(false) - const [showSalvos, setShowSalvos] = useState(false) + const { + matrix, probeStats, loading: matrixLoading, error: matrixError, route, + } = useMatrix(); + const [selectedProbe, setSelectedProbe] = useState(0); + const [selectedDestination, setSelectedDestination] = useState(1); + const [ProbeSOTRouting, setProbeSODRouting] = useState(false); + const [probeFullscreen, setProbeFullscreen] = useState(false); + const [showSalvos, setShowSalvos] = useState(false); useEffect(() => { if (config?.probe.enabled && ProbeSOTRouting) { - route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id) + route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id); } - if (config?.probe.enabled && selectedDestination == config.probe.router_destinations[selectedProbe] && ProbeSOTRouting) { - setProbeSODRouting(false) + if (config?.probe.enabled && selectedDestination === config.probe.router_destinations[selectedProbe] && ProbeSOTRouting) { + setProbeSODRouting(false); } }, [selectedDestination]); useEffect(() => { if (config?.probe.enabled && ProbeSOTRouting) { - if (matrix.destinations[selectedDestination - 1].source.id != matrix.destinations[config.probe.router_destinations[selectedProbe] - 1].source.id) { - route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id) + if (matrix.destinations[selectedDestination - 1].source.id !== matrix.destinations[config.probe.router_destinations[selectedProbe] - 1].source.id) { + route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id); } } }, [matrix]); useEffect(() => { if (config?.probe.enabled && ProbeSOTRouting) { - if (matrix.destinations[selectedDestination - 1].source.id != matrix.destinations[config.probe.router_destinations[selectedProbe] - 1].source.id) { - route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id) + if (matrix.destinations[selectedDestination - 1].source.id !== matrix.destinations[config.probe.router_destinations[selectedProbe] - 1].source.id) { + route(config.probe.router_destinations[selectedProbe], matrix.destinations[selectedDestination - 1].source.id); } } }, [ProbeSOTRouting]); + useEffect(() => { + if (config?.probe.enabled) { + if (selectedDestination !== config.probe.router_destinations[selectedProbe]) { + setSelectedDestination(config.probe.router_destinations[selectedProbe]); + } + } + }, [selectedProbe]); + return ( <> <>
- {(matrixLoading || configLoading) && "Loading"} - {(matrixError || configError) && JSON.stringify({matrixError: matrixError, configError: configError})} - {config && matrix && + overflow: 'clip', + background: gray[80], + }} + > + {(matrixLoading || configLoading) && 'Loading'} + {(matrixError || configError) && JSON.stringify({ matrixError, configError })} + {config && matrix + && ( <> - {config.probe.enabled && + {config.probe.enabled + && ( Probe {selectedProbe + 1}: {matrix.destinations?.[config.probe.router_destinations[selectedProbe] - 1]?.source?.label}} + modalHeading={( + <> + + Probe + {selectedProbe + 1} + : + {' '} + + {' '} + {matrix.destinations?.[config.probe.router_destinations[selectedProbe] - 1]?.source?.label} + +)} passiveModal - onRequestClose={()=> setProbeFullscreen(false)} + onRequestClose={() => setProbeFullscreen(false)} className="fullscreenProbe" > - + - } + )} Salvos} passiveModal - onRequestClose={()=> setShowSalvos(false)} + onRequestClose={() => setShowSalvos(false)} > - + @@ -108,35 +132,32 @@ const App = function App() {

Destinations

- - { matrix.destinations && matrix.destinations.map((button, buttonIndex) => { - return ( - - - - ) - })} + { matrix.destinations && matrix.destinations.map((button) => ( + + + + ))} @@ -145,107 +166,135 @@ const App = function App() {

Controls

- - { config.probe.enabled && + { config.probe.enabled + && ( <> { config.probe.router_destinations.map((dst, index) => ( - ))} -
+
- -
- Probe Follow: {ProbeSOTRouting ? matrix.destinations?.[selectedDestination - 1]?.label : "None"} -
- Probe Source: {matrix.destinations?.[config.probe.router_destinations[selectedProbe] - 1]?.source?.label} + + Status: + + {' '} + {probeStats[selectedProbe]?.active_source ? `Streaming, ${probeStats[selectedProbe]?.clients} viewer${probeStats[selectedProbe]?.clients === 1 ? '' : 's'}` : 'No Transport Stream'} +
+
+ +
+ + Probe + {' '} + Follow: + + {' '} + {ProbeSOTRouting ? matrix.destinations?.[selectedDestination - 1]?.label : 'None'} +
+ + Probe + {' '} + Source: + + {' '} + {matrix.destinations?.[config.probe.router_destinations[selectedProbe] - 1]?.source?.label}
-
+
- -

+
+
- } + )} {/* END PROBE */}

Salvos

- - - ) - })} + { matrix.sources && matrix.sources.map((button) => ( + + + + ))}
- - } + + )}
); }; -export default App; \ No newline at end of file +export default App; diff --git a/ui/src/Header.jsx b/ui/src/Header.jsx index 62d8efe..68aeb1a 100644 --- a/ui/src/Header.jsx +++ b/ui/src/Header.jsx @@ -3,22 +3,20 @@ import { Header, HeaderContainer, HeaderName, - HeaderNavigation, - HeaderMenuButton, - HeaderMenuItem, - SkipToContent, } from 'carbon-components-react'; -const TutorialHeader = () => ( - ( -
- - Router Controller - -
- )} - /> -); +function TutorialHeader() { + return ( + ( +
+ + Router Controller + +
+ )} + /> + ); +} -export default TutorialHeader; \ No newline at end of file +export default TutorialHeader; diff --git a/ui/src/JSmpegPlayer.jsx b/ui/src/JSmpegPlayer.jsx index 4b9a68f..3e4ce97 100644 --- a/ui/src/JSmpegPlayer.jsx +++ b/ui/src/JSmpegPlayer.jsx @@ -2,40 +2,45 @@ import React, { useState, useEffect, useRef } from 'react'; import JSMpeg from '@cycjimmy/jsmpeg-player'; -import imgs from './imgs' +import imgs from './imgs'; -const JSmpegPlayer = ({ url, active }) => { - const videoRef = useRef(null) - const [player, setPlayer] = useState(null) +function JSmpegPlayer({ url, active }) { + const videoRef = useRef(null); + const [player, setPlayer] = useState(null); useEffect(() => { if (active) { - player?.destroy() + player?.destroy(); const p = new JSMpeg.VideoElement( videoRef.current, url, { autoplay: false, - poster: imgs.probe_slate + poster: imgs.probe_slate, }, - {} - ) + {}, + ); setPlayer(p); } else { - player?.destroy() - setPlayer(null) + player?.destroy(); + setPlayer(null); } }, [active, url]); return ( + // eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-static-element-interactions
{ - player.player.isPlaying ? player.player.pause() : player.player.play() + if (player?.player.isPlaying) { + player?.player.pause(); + } else { + player?.player.play(); + } }} - className="probePlayer" - ref={videoRef}> -
+ className={active ? 'probePlayer' : 'probeSlate'} + ref={videoRef} + /> ); -}; +} -export default JSmpegPlayer; \ No newline at end of file +export default JSmpegPlayer; diff --git a/ui/src/Salvos.jsx b/ui/src/Salvos.jsx index 5d2da32..7701447 100644 --- a/ui/src/Salvos.jsx +++ b/ui/src/Salvos.jsx @@ -1,117 +1,112 @@ -import React, { useState, useEffect } from 'react'; +import React, { useState } from 'react'; import useAxios from 'axios-hooks'; import axios from 'axios'; -import useMatrix from './useMatrix' - -import { +import { Button, - ButtonSet, Column, ComboBox, Grid, Table, TableHead, TableRow, - TableContainer, TableHeader, TableBody, TableCell, TableToolbar, TableToolbarContent, IconButton, - Modal } from '@carbon/react'; -import { +import { TrashCan, - Add, Redo, - QueryQueue } from '@carbon/icons-react'; +import useMatrix from './useMatrix'; const Salvos = function Salvos() { - const {matrix, loading: matrixLoading, error: matrixError, route} = useMatrix() - const [{ data: config, loading: configLoading, error: configError }, refresh] = useAxios( - '/v1/config' - ) + const { + matrix, route, + } = useMatrix(); - const [currentSalvo, setCurrentSalvo] = useState({}) - const [newSalvoName, setNewSalvoName] = useState() + const [{ data: config, loading: configLoading, error: configError }, refresh] = useAxios( + '/v1/config', + ); - useEffect(() => { - console.log("Current Salvo", currentSalvo) - }, [currentSalvo]) + const [currentSalvo, setCurrentSalvo] = useState({}); + const [newSalvoName, setNewSalvoName] = useState(); - const headers = ['Destination', 'Saved Source', 'Current Source', "Actions"]; + const headers = ['Destination', 'Saved Source', 'Current Source', 'Actions']; const recallCurrentSalvo = () => { - currentSalvo.destinations.map((destination, index) => { + currentSalvo.destinations.forEach((destination, index) => { setTimeout( () => route(destination.id, destination.source.id), - 100 * index - ) - }) - } + 100 * index, + ); + }); + }; const saveCurrentSalvo = () => { - const salvo = {...currentSalvo} - salvo.destinations.map(destination => destination = matrix.destinations[destination.id - 1]) - console.log(salvo) + const salvo = { ...currentSalvo }; + salvo.destinations.map((destination) => destination = matrix.destinations[destination.id - 1]); + console.log(salvo); axios.post('/v1/salvos', salvo) .then(() => { - refresh() + refresh(); }) .catch((err) => { - console.log("unable to save salvo", err) + console.log('unable to save salvo', err); }); - } + }; return ( - {configLoading && "Loading..."} - {configError && "Error"} - {!configLoading && !configError && + {configLoading && 'Loading...'} + {configError && 'Error'} + {!configLoading && !configError + && ( - Selected Salvo: - Selected Salvo: + (item ? item.label : '')} + itemToString={(item) => (item ? item.label : '')} placeholder="Type to Filter..." onChange={(event) => { - if (event.selectedItem == null) {return} + if (event.selectedItem == null) { return; } if (event.selectedItem.destinations == undefined) { - if(newSalvoName != "") { - setCurrentSalvo({ label: newSalvoName, destinations: []}) + if (newSalvoName != '') { + setCurrentSalvo({ label: newSalvoName, destinations: [] }); } } else { - setCurrentSalvo(event.selectedItem) + setCurrentSalvo(event.selectedItem); } }} onInputChange={(event) => setNewSalvoName(event)} /> - - - {currentSalvo.label && + {currentSalvo.label + && ( <> - +
{headers.map((header) => ( @@ -123,50 +118,50 @@ const Salvos = function Salvos() { {currentSalvo.destinations?.map((row) => row != null && ( - - {row.label} - {row.source.label} - {matrix.destinations[row.id-1].source.label} - - { route(row.id, row.source.id)}} - > - - - setCurrentSalvo({...currentSalvo, destinations: [...currentSalvo.destinations.filter(savedDestination => savedDestination.id != row.id)]})} - > - - - - + + {row.label} + {row.source.label} + {matrix.destinations[row.id - 1].source.label} + + { route(row.id, row.source.id); }} + > + + + setCurrentSalvo({ ...currentSalvo, destinations: [...currentSalvo.destinations.filter((savedDestination) => savedDestination.id !== row.id)] })} + > + + + + ))}
- !currentSalvo.destinations.find(({ id }) => destination.id === id))} - itemToString={(item) => (item ? `${item.label} <= ${item.source.label}` : '')} + itemToString={(item) => (item ? `${item.label} <= ${item.source.label}` : '')} placeholder="Type to Filter..." helperText="Add Destination to Salvo" - direction={currentSalvo.destinations.length > 10 ? "top" : "bottom"} + direction={currentSalvo.destinations.length > 10 ? 'top' : 'bottom'} onChange={ (event) => { - if (event.selectedItem) setCurrentSalvo({...currentSalvo, destinations: [...currentSalvo.destinations, event.selectedItem].sort((a, b) => (a.id - b.id))}) + if (event.selectedItem) setCurrentSalvo({ ...currentSalvo, destinations: [...currentSalvo.destinations, event.selectedItem].sort((a, b) => (a.id - b.id)) }); } } /> - } + )}
- } + )}
- ) -} + ); +}; -export default Salvos +export default Salvos; diff --git a/ui/src/imgs.js b/ui/src/imgs.js index 1d332a5..f09fee4 100644 --- a/ui/src/imgs.js +++ b/ui/src/imgs.js @@ -1,4 +1,4 @@ export default { favicon: '', - probe_slate: '' -} \ No newline at end of file + probe_slate: '', +}; diff --git a/ui/src/index.js b/ui/src/index.js index 9ffef6b..7a64a0f 100644 --- a/ui/src/index.js +++ b/ui/src/index.js @@ -3,6 +3,6 @@ import React from 'react'; import ReactDOM from 'react-dom'; import App from './App.jsx'; -import './index.scss' +import './index.scss'; -ReactDOM.render(, document.querySelector('#root')); \ No newline at end of file +ReactDOM.render(, document.querySelector('#root')); diff --git a/ui/src/index.scss b/ui/src/index.scss index e8479fa..54e6edf 100644 --- a/ui/src/index.scss +++ b/ui/src/index.scss @@ -55,3 +55,10 @@ $feature-flags: ( aspect-ratio: auto 1920 / 1080; width: 100%; } + +.probeSlate { + aspect-ratio: auto 1920 / 1080; + width: 100%; + background-image: url(); + background-size: cover; +} diff --git a/ui/src/useMatrix.js b/ui/src/useMatrix.js index 3eba217..c413961 100644 --- a/ui/src/useMatrix.js +++ b/ui/src/useMatrix.js @@ -1,15 +1,16 @@ import { useState, useEffect } from 'react'; -import useAxios from 'axios-hooks' +import useAxios from 'axios-hooks'; import useWebSocket from 'react-use-websocket'; function useMatrix() { const [matrix, setMatrix] = useState([]); + const [probeStats, setProbeStats] = useState([]); const [haveRecievedData, setHaveRecievedData] = useState(false); const wsuri = `${window.location.protocol === 'https:' ? 'wss:' : 'ws:'}//${window.location.hostname}${window.location.port ? `:${window.location.port}` : ''}/v1/ws/matrix`; const [{ data, loading, error }, refreshAxios] = useAxios( - '/v1/matrix' - ) + '/v1/matrix', + ); if (data && matrix.length === 0 && !haveRecievedData) setMatrix(data); @@ -34,12 +35,11 @@ function useMatrix() { useEffect(() => { if (lastMessage !== null) { setHaveRecievedData(true); - const tempMatrix = {...matrix}; + const tempMatrix = { ...matrix }; const update = JSON.parse(lastMessage.data); - console.log("received Matrix Update", update) + let found = false; switch (update.type) { - case "destination_update": - let found = false; + case 'destination_update': tempMatrix.destinations.forEach((dst, i) => { if (update.data.id === dst.id) { tempMatrix.destinations[i] = update.data; @@ -49,6 +49,12 @@ function useMatrix() { if (!found) { tempMatrix.destinations.push(update.data); } + break; + case 'probe_stats': + setProbeStats(update.data); + break; + default: + console.log('unexpected message', update); } setMatrix(tempMatrix); @@ -57,15 +63,17 @@ function useMatrix() { const route = (destination, source) => { sendJsonMessage({ - type: "route_request", + type: 'route_request', data: { - destination: destination, - source: source - } - }) - } + destination, + source, + }, + }); + }; - return {matrix, loading, error, route}; + return { + matrix, probeStats, loading, error, route, + }; } export default useMatrix; diff --git a/ui/webpack.config.js b/ui/webpack.config.js index 65df91a..f26de28 100644 --- a/ui/webpack.config.js +++ b/ui/webpack.config.js @@ -2,7 +2,7 @@ const path = require('path'); const ReactRefreshWebpackPlugin = require('@pmmmwh/react-refresh-webpack-plugin'); const HtmlWebpackPlugin = require('html-webpack-plugin'); -const WebpackPwaManifest = require('webpack-pwa-manifest') +const WebpackPwaManifest = require('webpack-pwa-manifest'); const MiniCssExtractPlugin = require('mini-css-extract-plugin'); const port = process.env.PORT || 3000; @@ -40,9 +40,9 @@ module.exports = (_, argv) => { icons: [ { src: path.resolve('src/router.png'), - sizes: [96, 128, 192, 256, 384, 512] + sizes: [96, 128, 192, 256, 384, 512], }, - ] + ], }), isDevelopment && new ReactRefreshWebpackPlugin(), ].filter(Boolean), @@ -73,11 +73,11 @@ module.exports = (_, argv) => { test: /\.s[ac]ss$/i, use: [ // Creates `style` nodes from JS strings - "style-loader", + 'style-loader', // Translates CSS into CommonJS - "css-loader", + 'css-loader', // Compiles Sass to CSS - "sass-loader", + 'sass-loader', ], }, {