mirror of
https://github.com/openhab/openhab-addons.git
synced 2025-01-25 14:55:55 +01:00
[ipcamera] Fix multiple mjpeg issues and allow stream to stay alive (#11921)
* Fix for a camera that has a space in boundary * Fixes to ipcamera.mjpeg Signed-off-by: Matthew Skinner <matt@pcmus.com>
This commit is contained in:
parent
46ba275edb
commit
cb3496f967
@ -212,7 +212,7 @@ public class DahuaHandler extends ChannelDuplexHandler {
|
|||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
String content = msg.toString();
|
String content = msg.toString();
|
||||||
if (content.startsWith("--myboundary")) {
|
if (content.startsWith("--myboundary") || content.startsWith("-- myboundary")) {
|
||||||
processEvent(content);
|
processEvent(content);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -232,17 +232,17 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (contentType.contains("multipart")) {
|
if (contentType.contains("multipart")) {
|
||||||
|
boundary = Helper.searchString(contentType, "boundary=");
|
||||||
if (mjpegUri.equals(requestUrl)) {
|
if (mjpegUri.equals(requestUrl)) {
|
||||||
if (msg instanceof HttpMessage) {
|
if (msg instanceof HttpMessage) {
|
||||||
// very start of stream only
|
// very start of stream only
|
||||||
mjpegContentType = contentType;
|
mjpegContentType = contentType;
|
||||||
CameraServlet localServlet = servlet;
|
CameraServlet localServlet = servlet;
|
||||||
if (localServlet != null) {
|
if (localServlet != null) {
|
||||||
localServlet.openStreams.updateContentType(contentType);
|
logger.debug("Setting Content-Type to:{}", contentType);
|
||||||
|
localServlet.openStreams.updateContentType(contentType, boundary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
boundary = Helper.searchString(contentType, "boundary=");
|
|
||||||
}
|
}
|
||||||
} else if (contentType.contains("image/jp")) {
|
} else if (contentType.contains("image/jp")) {
|
||||||
if (bytesToRecieve == 0) {
|
if (bytesToRecieve == 0) {
|
||||||
@ -669,8 +669,13 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void openCamerasStream() {
|
public void openCamerasStream() {
|
||||||
|
if (mjpegUri.isEmpty() || "ffmpeg".equals(mjpegUri)) {
|
||||||
|
setupFfmpegFormat(FFmpegFormat.MJPEG);
|
||||||
|
return;
|
||||||
|
}
|
||||||
closeChannel(getTinyUrl(mjpegUri));
|
closeChannel(getTinyUrl(mjpegUri));
|
||||||
mainEventLoopGroup.schedule(this::openMjpegStream, 0, TimeUnit.MILLISECONDS);
|
// Dahua cameras crash if you refresh (close and open) the stream without this delay.
|
||||||
|
mainEventLoopGroup.schedule(this::openMjpegStream, 300, TimeUnit.MILLISECONDS);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void openMjpegStream() {
|
private void openMjpegStream() {
|
||||||
@ -1311,6 +1316,12 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
|
|
||||||
pollCameraJob = threadPool.scheduleWithFixedDelay(this::pollCameraRunnable, 1000, 8000, TimeUnit.MILLISECONDS);
|
pollCameraJob = threadPool.scheduleWithFixedDelay(this::pollCameraRunnable, 1000, 8000, TimeUnit.MILLISECONDS);
|
||||||
|
|
||||||
|
// auto restart mjpeg stream now camera is back online.
|
||||||
|
CameraServlet localServlet = servlet;
|
||||||
|
if (localServlet != null && !localServlet.openStreams.isEmpty()) {
|
||||||
|
openCamerasStream();
|
||||||
|
}
|
||||||
|
|
||||||
if (!rtspUri.isEmpty()) {
|
if (!rtspUri.isEmpty()) {
|
||||||
updateState(CHANNEL_RTSP_URL, new StringType(rtspUri));
|
updateState(CHANNEL_RTSP_URL, new StringType(rtspUri));
|
||||||
}
|
}
|
||||||
@ -1342,6 +1353,7 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void pollingCameraConnection() {
|
void pollingCameraConnection() {
|
||||||
|
keepMjpegRunning();
|
||||||
if (thing.getThingTypeUID().getId().equals(GENERIC_THING)) {
|
if (thing.getThingTypeUID().getId().equals(GENERIC_THING)) {
|
||||||
if (rtspUri.isEmpty()) {
|
if (rtspUri.isEmpty()) {
|
||||||
logger.warn("Binding has not been supplied with a FFmpeg Input URL, so some features will not work.");
|
logger.warn("Binding has not been supplied with a FFmpeg Input URL, so some features will not work.");
|
||||||
@ -1643,7 +1655,17 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
// Only use ONVIF events if it is not an API camera.
|
// Only use ONVIF events if it is not an API camera.
|
||||||
onvifCamera.connect(thing.getThingTypeUID().getId().equals(ONVIF_THING));
|
onvifCamera.connect(thing.getThingTypeUID().getId().equals(ONVIF_THING));
|
||||||
}
|
}
|
||||||
cameraConnectionJob = threadPool.scheduleWithFixedDelay(this::pollingCameraConnection, 4, 30, TimeUnit.SECONDS);
|
cameraConnectionJob = threadPool.scheduleWithFixedDelay(this::pollingCameraConnection, 4, 8, TimeUnit.SECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void keepMjpegRunning() {
|
||||||
|
CameraServlet localServlet = servlet;
|
||||||
|
if (localServlet != null && !localServlet.openStreams.isEmpty()) {
|
||||||
|
if (!mjpegUri.isEmpty() && !"ffmpeg".equals(mjpegUri)) {
|
||||||
|
localServlet.openStreams.queueFrame(("--" + localServlet.openStreams.boundary + "\r\n\r\n").getBytes());
|
||||||
|
}
|
||||||
|
localServlet.openStreams.queueFrame(getSnapshot());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// What the camera needs to re-connect if the initialize() is not called.
|
// What the camera needs to re-connect if the initialize() is not called.
|
||||||
|
@ -175,27 +175,28 @@ public class CameraServlet extends IpCameraServlet {
|
|||||||
}
|
}
|
||||||
} while (true);
|
} while (true);
|
||||||
case "/ipcamera.mjpeg":
|
case "/ipcamera.mjpeg":
|
||||||
if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
|
if (openStreams.isEmpty()) {
|
||||||
if (openStreams.isEmpty()) {
|
|
||||||
handler.setupFfmpegFormat(FFmpegFormat.MJPEG);
|
|
||||||
}
|
|
||||||
output = new StreamOutput(resp);
|
|
||||||
openStreams.addStream(output);
|
|
||||||
} else if (openStreams.isEmpty()) {
|
|
||||||
logger.debug("First stream requested, opening up stream from camera");
|
logger.debug("First stream requested, opening up stream from camera");
|
||||||
handler.openCamerasStream();
|
handler.openCamerasStream();
|
||||||
output = new StreamOutput(resp, handler.mjpegContentType);
|
if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
|
||||||
openStreams.addStream(output);
|
output = new StreamOutput(resp);
|
||||||
} else {
|
} else {
|
||||||
ChannelTracking tracker = handler.channelTrackingMap.get(handler.mjpegUri);
|
output = new StreamOutput(resp, handler.mjpegContentType);
|
||||||
if (tracker == null || !tracker.getChannel().isOpen()) {
|
}
|
||||||
logger.debug("Not the first stream requested but the stream from camera was closed");
|
} else {
|
||||||
handler.openCamerasStream();
|
if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
|
||||||
openStreams.closeAllStreams();
|
output = new StreamOutput(resp);
|
||||||
|
} else {
|
||||||
|
ChannelTracking tracker = handler.channelTrackingMap.get(handler.mjpegUri);
|
||||||
|
if (tracker == null || !tracker.getChannel().isOpen()) {
|
||||||
|
logger.debug("Not the first stream requested but the stream from camera was closed");
|
||||||
|
handler.openCamerasStream();
|
||||||
|
openStreams.closeAllStreams();
|
||||||
|
}
|
||||||
|
output = new StreamOutput(resp, handler.mjpegContentType);
|
||||||
}
|
}
|
||||||
output = new StreamOutput(resp, handler.mjpegContentType);
|
|
||||||
openStreams.addStream(output);
|
|
||||||
}
|
}
|
||||||
|
openStreams.addStream(output);
|
||||||
do {
|
do {
|
||||||
try {
|
try {
|
||||||
output.sendFrame();
|
output.sendFrame();
|
||||||
|
@ -29,6 +29,7 @@ import org.eclipse.jdt.annotation.NonNullByDefault;
|
|||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class OpenStreams {
|
public class OpenStreams {
|
||||||
private List<StreamOutput> openStreams = Collections.synchronizedList(new ArrayList<StreamOutput>());
|
private List<StreamOutput> openStreams = Collections.synchronizedList(new ArrayList<StreamOutput>());
|
||||||
|
public String boundary = "thisMjpegStream";
|
||||||
|
|
||||||
public synchronized void addStream(StreamOutput stream) {
|
public synchronized void addStream(StreamOutput stream) {
|
||||||
openStreams.add(stream);
|
openStreams.add(stream);
|
||||||
@ -46,7 +47,8 @@ public class OpenStreams {
|
|||||||
return openStreams.isEmpty();
|
return openStreams.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void updateContentType(String contentType) {
|
public synchronized void updateContentType(String contentType, String boundary) {
|
||||||
|
this.boundary = boundary;
|
||||||
for (StreamOutput stream : openStreams) {
|
for (StreamOutput stream : openStreams) {
|
||||||
stream.updateContentType(contentType);
|
stream.updateContentType(contentType);
|
||||||
}
|
}
|
||||||
|
@ -20,6 +20,8 @@ import javax.servlet.ServletOutputStream;
|
|||||||
import javax.servlet.http.HttpServletResponse;
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The {@link StreamOutput} Streams mjpeg out to a client
|
* The {@link StreamOutput} Streams mjpeg out to a client
|
||||||
@ -29,11 +31,12 @@ import org.eclipse.jdt.annotation.NonNullByDefault;
|
|||||||
|
|
||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class StreamOutput {
|
public class StreamOutput {
|
||||||
|
public final Logger logger = LoggerFactory.getLogger(getClass());
|
||||||
private final HttpServletResponse response;
|
private final HttpServletResponse response;
|
||||||
private final String boundary;
|
private final String boundary;
|
||||||
private String contentType;
|
private String contentType;
|
||||||
private final ServletOutputStream output;
|
private final ServletOutputStream output;
|
||||||
private BlockingQueue<byte[]> fifo = new ArrayBlockingQueue<byte[]>(6);
|
private BlockingQueue<byte[]> fifo = new ArrayBlockingQueue<byte[]>(30);
|
||||||
private boolean connected = false;
|
private boolean connected = false;
|
||||||
public boolean isSnapshotBased = false;
|
public boolean isSnapshotBased = false;
|
||||||
|
|
||||||
@ -76,6 +79,7 @@ public class StreamOutput {
|
|||||||
try {
|
try {
|
||||||
fifo.add(frame);
|
fifo.add(frame);
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
|
logger.debug("FIFO buffer has run out of space:{}", e.getMessage());
|
||||||
fifo.remove();
|
fifo.remove();
|
||||||
fifo.add(frame);
|
fifo.add(frame);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user