Luego de de una extensa recopilación de datos resolvimos esta cuestión que tienen algunos los lectores. Te regalamos la respuesta y nuestro deseo es que resulte de gran apoyo.
Solución:
Adapté el código de aquí y lo integré con un servidor http y controles socket.io: https://github.com/phoboslab/jsmpeg
Servidor:
V4L2 -> FFMPEG (MPEG1 TS) -> Servidor HTTP NODE -> Difusión NODE Websocket
Cliente:
Websocket -> Javascript (Decodificar MPEG1 TS y pintar en lienzo html) -> Lienzo HTML
Esta pila logra [email protected] con 240 ms de latencia. Aún lejos de mi objetivo, pero lo suficientemente bueno como MVP. Los controles en ambas direcciones tienen una latencia de 7ms, lo cual es excelente.
Esta pila es retenida por la etapa de transcodificación y decodificación, y el RPI se calienta mucho. El transporte de datos sin procesar a través de websocket se ve bien, voy a perfilar la latencia de cada paso en el futuro.
Ejecución:
[email protected]
:~ $ node node.js &
[email protected]:~ $ ffmpeg -f v4l2 -framerate 20 -video_size 640x480 -i /dev/video0 -f mpegts -codec:v mpeg1video -s 640x480 -b:v 600k -bf 0 http://localhost:8080/mystream
NODE.JS del lado del servidor
//operating system library. Used to get local IP address
var os = require("os");
//file system library. Used to load file stored inside back end server (https://nodejs.org/api/fs.html)
var fs = require("fs");
//http system library. Handles basic html requests
var http = require("http").createServer(http_handler);
//url library. Used to process html url requests
var url = require("url");
//Websocket
var io = require("socket.io")(http);
//Websocket used to stream video
var websocket = require("ws");
//-----------------------------------------------------------------------------------
// CONFIGURATION
//-----------------------------------------------------------------------------------
//Port the server will listen to
var server_port = 8080;
var websocket_stream_port = 8082;
//Path of the http and css files for the http server
var file_index_name = "index.html";
var file_css_name = "style.css";
var file_jsplayer_name = "jsmpeg.min.js";
//Http and css files loaded into memory for fast access
var file_index;
var file_css;
var file_jsplayer;
//Name of the local video stream
var stream_name = "mystream";
//-----------------------------------------------------------------------------------
// DETECT SERVER OWN IP
//-----------------------------------------------------------------------------------
//If just one interface, store the server IP Here
var server_ip;
//Get local IP address of the server
//https://stackoverflow.com/questions/3653065/get-local-ip-address-in-node-js
var ifaces = os.networkInterfaces();
Object.keys(ifaces).forEach
(
function (ifname)
var alias = 0;
ifaces[ifname].forEach
(
function (iface)
iface.internal !== false)
// skip over internal (i.e. 127.0.0.1) and non-ipv4 addresses
return;
if (alias >= 1)
// this single interface has multiple ipv4 addresses
console.log('INFO: Server interface ' +alias +' - ' + ifname + ':' + alias, iface.address);
else
server_ip = iface.address;
// this interface has only one ipv4 adress
console.log('INFO: Server interface - ' +ifname, iface.address);
++alias;
);
);
//-----------------------------------------------------------------------------------
// HTTP SERVER
//-----------------------------------------------------------------------------------
// Fetch and serves local files to client
//Create http server and listen to the given port
http.listen
(
server_port,
function( )
console.log('INFO: ' +server_ip +' listening to html requests on port ' +server_port);
//Pre-load http, css and js files into memory to improve http request latency
file_index = load_file( file_index_name );
file_css = load_file( file_css_name );
file_jsplayer = load_file( file_jsplayer_name );
);
//-----------------------------------------------------------------------------------
// HTTP REQUESTS HANDLER
//-----------------------------------------------------------------------------------
// Answer to client http requests. Serve http, css and js files
function http_handler(req, res)
//If client asks for root
if (req.url == '/')
//Request main page
res.writeHead( 200, "Content-Type": detect_content(file_index_name),"Content-Length":file_index.length );
res.write(file_index);
res.end();
console.log("INFO: Serving file: " +req.url);
//If client asks for css file
else if (req.url == ("/" +file_css_name))
//Request main page
res.writeHead( 200, "Content-Type": detect_content(file_css_name),"Content-Length" :file_css.length );
res.write(file_css);
res.end();
console.log("INFO: Serving file: " +req.url);
//If client asks for css file
else if (req.url == ("/" +file_jsplayer_name))
//Request main page
res.writeHead( 200, "Content-Type": detect_content(file_jsplayer_name),"Content-Length" :file_jsplayer.length );
res.write(file_jsplayer);
res.end();
console.log("INFO: Serving file: " +req.url);
//Listening to the port the stream from ffmpeg will flow into
else if (req.url = "/mystream")
res.connection.setTimeout(0);
console.log( "Stream Connected: " +req.socket.remoteAddress + ":" +req.socket.remotePort );
req.on
(
"data",
function(data)
streaming_websocket.broadcast(data);
/*
if (req.socket.recording)
req.socket.recording.write(data);
*/
//console.log("broadcast: ", data.length);
);
req.on
(
"end",
function()
console.log("local stream has ended");
if (req.socket.recording)
req.socket.recording.close();
);
//If client asks for an unhandled path
else
res.end();
console.log("ERR: Invalid file request" +req.url);
//-----------------------------------------------------------------------------------
// WEBSOCKET SERVER: CONTROL/FEEDBACK REQUESTS
//-----------------------------------------------------------------------------------
// Handle websocket connection to the client
io.on
(
"connection",
function (socket)
console.log("connecting...");
socket.emit("welcome", payload: "Server says hello" );
//Periodically send the current server time to the client in string form
setInterval
(
function()
socket.emit("server_time", server_time: get_server_time() );
,
//Send every 333ms
333
);
socket.on
(
"myclick",
function (data)
timestamp_ms = get_timestamp_ms();
socket.emit("profile_ping", timestamp: timestamp_ms );
console.log("button event: " +" client says: " +data.payload);
);
//"ArrowLeft"
socket.on
(
"keyboard",
function (data)
timestamp_ms = get_timestamp_ms();
socket.emit("profile_ping", timestamp: timestamp_ms );
console.log("keyboard event: " +" client says: " +data.payload);
);
//profile packets from the client are answer that allows to compute roundway trip time
socket.on
(
"profile_pong",
function (data)
timestamp_ms_pong = get_timestamp_ms();
timestamp_ms_ping = data.timestamp;
console.log("Pong received. Round trip time[ms]: " +(timestamp_ms_pong -timestamp_ms_ping));
);
);
//-----------------------------------------------------------------------------------
// WEBSOCKET SERVER: STREAMING VIDEO
//-----------------------------------------------------------------------------------
// Websocket Server
var streaming_websocket = new websocket.Server(port: websocket_stream_port, perMessageDeflate: false);
streaming_websocket.connectionCount = 0;
streaming_websocket.on
(
"connection",
function(socket, upgradeReq)
);
streaming_websocket.broadcast = function(data)
streaming_websocket.clients.forEach
(
function each(client)
if (client.readyState === websocket.OPEN)
client.send(data);
);
;
//-----------------------------------------------------------------------------------
// FUNCTIONS
//-----------------------------------------------------------------------------------
//-----------------------------------------------------------------------------------
// SERVER DATE&TIME
//-----------------------------------------------------------------------------------
// Get server time in string form
function get_server_time()
my_date = new Date();
return my_date.toUTCString();
//-----------------------------------------------------------------------------------
// TIMESTAMP
//-----------------------------------------------------------------------------------
// Profile performance in ms
function get_timestamp_ms()
my_date = new Date();
return 1000.0* my_date.getSeconds() +my_date.getMilliseconds()
//-----------------------------------------------------------------------------------
// FILE LOADER
//-----------------------------------------------------------------------------------
// Load files into memory for improved latency
function load_file( file_name )
var file_tmp;
var file_path = __dirname +"/" +file_name;
//HTML index file
try
file_tmp = fs.readFileSync( file_path );
catch (err)
console.log("ERR: " +err.code +" failed to load: " +file_path);
throw err;
console.log("INFO: " +file_path +" has been loaded into memory");
return file_tmp;
//-----------------------------------------------------------------------------------
// CONTENT TYPE DETECTOR
//-----------------------------------------------------------------------------------
// Return the right content type to give correct information to the client browser
function detect_content( file_name )
if (file_name.includes(".html"))
return "text/html";
else if (file_name.includes(".css"))
return "text/css";
else if (file_name.includes(".js"))
return "application/javascript";
else
throw "invalid extension";
HTML del lado del cliente
Maze Runner
Html+Css Server +low latency Websocket server
This button will emit a websocket event. The server will be informed in real time of the event.
This input can be filled through websockets directly by the server in real time
This canvas is painted by the javascript player and shows the live stream.'
Reproductor de JavaScript
Puede obtener el reproductor de JavaScript que utilicé desde aquí: https://github.com/phoboslab/jsmpeg/blob/master/jsmpeg.min.js
Si te ha resultado de provecho este artículo, sería de mucha ayuda si lo compartieras con otros desarrolladores y nos ayudes a extender nuestra información.