mirror of
https://github.com/openhab/openhab-addons.git
synced 2025-01-25 14:55:55 +01:00
[ipcamera] Add support to change MJPEG source to a different RTSP sub stream to lower CPU usage (#17591)
* Add support to change MJPEG sources to different RTSP sub streams Signed-off-by: Matthew Skinner <matt@pcmus.com>
This commit is contained in:
parent
3cd5879209
commit
ed45e7526d
@ -128,8 +128,10 @@ Thing ipcamera:hikvision:West "West Camera"
|
|||||||
|
|
||||||
### Reolink
|
### Reolink
|
||||||
|
|
||||||
- NVR's made by Reolink have ONVIF disabled by default and may require a screen connected to the hardware to enable ONVIF or newer firmwares may be able to do this via their app or web UI.
|
- NVR's made by Reolink have ONVIF disabled by default and may require a screen connected to the hardware to enable ONVIF, or newer firmwares may be able to do this via their app or web UI.
|
||||||
- This binding will use the Reolink API for polling the alarms if the `nvrChannel` is 1 or higher and does not need ONVIF to be enabled. To use ONVIF event methods for the alarms, you can set `nvrChannel` to 0.
|
- This binding will use the Reolink API for polling the alarms if the `nvrChannel` is 1 or higher and does not need ONVIF to be enabled. To use ONVIF event methods for the alarms, you can set `nvrChannel` to 0.
|
||||||
|
- Cameras have ONVIF, RTSP and HTTP disabled by default, to enable these required features, do the following: DEVICE SETTINGS>NETWORK>ADVANCED>PORT SETTINGS> then turn these on leaving the default port number alone.
|
||||||
|
- Consider setting the substream of the camera to be 4 FPS and to 640*360 as this will lower CPU load if using the ipcamera.mjpeg stream.
|
||||||
|
|
||||||
## Discovery
|
## Discovery
|
||||||
|
|
||||||
@ -194,7 +196,7 @@ If you do not specify any of these, the binding will use the default which shoul
|
|||||||
| `alarmInputUrl` | A URL you can use for the FFmpeg created Audio and Motion Alarms as they don't require high res feeds as they are not seen. |
|
| `alarmInputUrl` | A URL you can use for the FFmpeg created Audio and Motion Alarms as they don't require high res feeds as they are not seen. |
|
||||||
| `customMotionAlarmUrl`| Foscam only, for custom enable motion alarm use. More info found in Foscam's setup steps. |
|
| `customMotionAlarmUrl`| Foscam only, for custom enable motion alarm use. More info found in Foscam's setup steps. |
|
||||||
| `customAudioAlarmUrl`| Foscam only, for custom enable audio alarm use. More info found in Foscam's setup steps. |
|
| `customAudioAlarmUrl`| Foscam only, for custom enable audio alarm use. More info found in Foscam's setup steps. |
|
||||||
| `mjpegUrl`| A HTTP URL for MJPEG format streams. If you enter 'ffmpeg' the stream can be generated from the RTSP URL. |
|
| `mjpegUrl`| A HTTP or RTSP URL to use for MJPEG streams. If you enter 'ffmpeg' the stream can be generated using the RTSP URL. |
|
||||||
| `ffmpegInput`| Best if this stream is in H.264 format and can be RTSP or HTTP URLs. Leave this blank to use the auto detected RTSP address for ONVIF cameras. |
|
| `ffmpegInput`| Best if this stream is in H.264 format and can be RTSP or HTTP URLs. Leave this blank to use the auto detected RTSP address for ONVIF cameras. |
|
||||||
| `ffmpegInputOptions` | Allows you to specify any options before the -i on the commands for FFmpeg. If you have an ESP32 camera that only has a mjpeg stream then make this equal `-f mjpeg`. |
|
| `ffmpegInputOptions` | Allows you to specify any options before the -i on the commands for FFmpeg. If you have an ESP32 camera that only has a mjpeg stream then make this equal `-f mjpeg`. |
|
||||||
| `ffmpegLocation`| The full path including the filename for where you have installed FFmpeg. The default should work for most Linux installs but if using windows use this format: `c:\ffmpeg\bin\ffmpeg.exe` |
|
| `ffmpegLocation`| The full path including the filename for where you have installed FFmpeg. The default should work for most Linux installs but if using windows use this format: `c:\ffmpeg\bin\ffmpeg.exe` |
|
||||||
@ -434,14 +436,15 @@ You can cast it which can be handy to show a moving picture that keeps repeating
|
|||||||
## MJPEG Streams
|
## MJPEG Streams
|
||||||
|
|
||||||
Cameras that have built in MJPEG abilities can stream to openHAB with the MJPEG format with next to no CPU load, less than 1 second lag, and FFmpeg does not need to be installed.
|
Cameras that have built in MJPEG abilities can stream to openHAB with the MJPEG format with next to no CPU load, less than 1 second lag, and FFmpeg does not need to be installed.
|
||||||
Cameras without this ability can still use this binding to convert their RTSP H.264 format to MJPEG (keep reading for more on this below) and this will take a lot of CPU power to handle the conversion.
|
Cameras without this ability can still use this binding to convert their RTSP H.264 format to MJPEG (keep reading for more on this below) and this may use a lot of CPU power to handle the conversion.
|
||||||
|
The lower the resolution of the stream, the lower the CPU load, so consider using a substream of the camera in the `mjpegUrl` configuration field.
|
||||||
The alternative is to use HLS format which does not need the conversion and does not use any CPU to speak of.
|
The alternative is to use HLS format which does not need the conversion and does not use any CPU to speak of.
|
||||||
For video without a delay, you need MJPEG and without a camera that can create it, you will need to use a lot of CPU power.
|
For video without a delay, you need MJPEG and without a camera that can create it, you will need to use a lot of CPU power.
|
||||||
This can be done in a dedicated video server which will be the only way with lots of cameras, unless you purchase cameras that have the ability built in.
|
This can be done in a dedicated video server which will be the only way with lots of cameras, unless you purchase cameras that have the ability built in.
|
||||||
|
|
||||||
An alternative way to keep the CPU load low is to use the `snapshots.mjpeg` feature of the binding to create a stream from the cameras snapshots instead of the RTSP stream.
|
An alternative way to keep the CPU load low is to use the `snapshots.mjpeg` or `autofps.mjpeg` feature of the binding to create a stream from the cameras snapshots instead of the RTSP stream.
|
||||||
|
|
||||||
The main cameras that can do MJPEG with very low CPU load are Amcrest, Dahua, Hikvision, Foscam HD and Instar HD.
|
The main cameras that can do MJPEG with very low CPU load are Amcrest, Dahua, ESP32 Camera, Hikvision, Foscam HD and Instar HD.
|
||||||
To set this up, see [Special Notes for Different Brands](#special-notes-for-different-brands).
|
To set this up, see [Special Notes for Different Brands](#special-notes-for-different-brands).
|
||||||
The binding can then distribute this stream to many devices around your home whilst the camera only sees a single open stream.
|
The binding can then distribute this stream to many devices around your home whilst the camera only sees a single open stream.
|
||||||
|
|
||||||
@ -597,7 +600,7 @@ Webview url="http://192.168.6.4:8080/static/html/file.html" height=5
|
|||||||
There are two ways to cast a camera.
|
There are two ways to cast a camera.
|
||||||
|
|
||||||
1. openHAB Cloud Connector and using metadata/tags.
|
1. openHAB Cloud Connector and using metadata/tags.
|
||||||
1. Chromecast Bindings `playuri` channel.
|
2. Chromecast Bindings `playuri` channel.
|
||||||
|
|
||||||
The first method once setup allows you to ask "OK Google show X camera", or "OK Google show X camera on Y display".
|
The first method once setup allows you to ask "OK Google show X camera", or "OK Google show X camera on Y display".
|
||||||
By optionally naming the display that you wish to use, it can be cast directly to your Chromecast (connected to your TV) by speaking to a Google Nest Mini.
|
By optionally naming the display that you wish to use, it can be cast directly to your Chromecast (connected to your TV) by speaking to a Google Nest Mini.
|
||||||
|
@ -63,7 +63,7 @@ public class Ffmpeg {
|
|||||||
ipCameraHandler = handle;
|
ipCameraHandler = handle;
|
||||||
String altInput = input;
|
String altInput = input;
|
||||||
// Input can be snapshots not just rtsp or http
|
// Input can be snapshots not just rtsp or http
|
||||||
if (!password.isEmpty() && !input.contains("@") && input.contains("rtsp")) {
|
if (!password.isEmpty() && !input.contains("@") && input.toLowerCase().startsWith("rtsp")) {
|
||||||
String credentials = username + ":" + this.password + "@";
|
String credentials = username + ":" + this.password + "@";
|
||||||
// will not work for https: but currently binding does not use https
|
// will not work for https: but currently binding does not use https
|
||||||
altInput = input.substring(0, 7) + credentials + input.substring(7);
|
altInput = input.substring(0, 7) + credentials + input.substring(7);
|
||||||
|
@ -712,7 +712,7 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void openCamerasStream() {
|
public void openCamerasStream() {
|
||||||
if (mjpegUri.isEmpty() || "ffmpeg".equals(mjpegUri)) {
|
if (usingRtspForMjpeg()) {
|
||||||
setupFfmpegFormat(FFmpegFormat.MJPEG);
|
setupFfmpegFormat(FFmpegFormat.MJPEG);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -811,7 +811,7 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
logger.warn("The camera tried to use a FFmpeg feature when the location for FFmpeg is not known.");
|
logger.warn("The camera tried to use a FFmpeg feature when the location for FFmpeg is not known.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (rtspUri.toLowerCase().contains("rtsp")) {
|
if (mjpegUri.toLowerCase().startsWith("rtsp://") || rtspUri.toLowerCase().startsWith("rtsp://")) {
|
||||||
if (inputOptions.isEmpty()) {
|
if (inputOptions.isEmpty()) {
|
||||||
inputOptions = "-rtsp_transport tcp";
|
inputOptions = "-rtsp_transport tcp";
|
||||||
}
|
}
|
||||||
@ -938,10 +938,17 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
} else {
|
} else {
|
||||||
inputOptions += " -hide_banner";
|
inputOptions += " -hide_banner";
|
||||||
}
|
}
|
||||||
ffmpegMjpeg = new Ffmpeg(this, format, cameraConfig.getFfmpegLocation(), inputOptions, rtspUri,
|
if (mjpegUri.toLowerCase().startsWith("rtsp://")) {
|
||||||
cameraConfig.getMjpegOptions(), "http://127.0.0.1:" + SERVLET_PORT + "/ipcamera/"
|
ffmpegMjpeg = new Ffmpeg(this, format, cameraConfig.getFfmpegLocation(), inputOptions, mjpegUri,
|
||||||
+ getThing().getUID().getId() + "/ipcamera.jpg",
|
cameraConfig.getMjpegOptions(), "http://127.0.0.1:" + SERVLET_PORT + "/ipcamera/"
|
||||||
cameraConfig.getUser(), cameraConfig.getPassword());
|
+ getThing().getUID().getId() + "/ipcamera.jpg",
|
||||||
|
cameraConfig.getUser(), cameraConfig.getPassword());
|
||||||
|
} else {
|
||||||
|
ffmpegMjpeg = new Ffmpeg(this, format, cameraConfig.getFfmpegLocation(), inputOptions, rtspUri,
|
||||||
|
cameraConfig.getMjpegOptions(), "http://127.0.0.1:" + SERVLET_PORT + "/ipcamera/"
|
||||||
|
+ getThing().getUID().getId() + "/ipcamera.jpg",
|
||||||
|
cameraConfig.getUser(), cameraConfig.getPassword());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Ffmpeg localMjpeg = ffmpegMjpeg;
|
Ffmpeg localMjpeg = ffmpegMjpeg;
|
||||||
if (localMjpeg != null) {
|
if (localMjpeg != null) {
|
||||||
@ -1649,7 +1656,10 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
threadPool = Executors.newScheduledThreadPool(2);
|
threadPool = Executors.newScheduledThreadPool(2);
|
||||||
mainEventLoopGroup = new NioEventLoopGroup(3);
|
mainEventLoopGroup = new NioEventLoopGroup(3);
|
||||||
snapshotUri = getCorrectUrlFormat(cameraConfig.getSnapshotUrl());
|
snapshotUri = getCorrectUrlFormat(cameraConfig.getSnapshotUrl());
|
||||||
mjpegUri = getCorrectUrlFormat(cameraConfig.getMjpegUrl());
|
mjpegUri = cameraConfig.getMjpegUrl();
|
||||||
|
if (!mjpegUri.toLowerCase().startsWith("rtsp://")) {
|
||||||
|
mjpegUri = getCorrectUrlFormat(mjpegUri);
|
||||||
|
}
|
||||||
rtspUri = cameraConfig.getFfmpegInput();
|
rtspUri = cameraConfig.getFfmpegInput();
|
||||||
if (cameraConfig.getFfmpegOutput().isEmpty()) {
|
if (cameraConfig.getFfmpegOutput().isEmpty()) {
|
||||||
cameraConfig
|
cameraConfig
|
||||||
@ -1739,6 +1749,10 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
rtspUri = "rtsp://" + cameraConfig.getIp() + ":554/h264Preview_0"
|
rtspUri = "rtsp://" + cameraConfig.getIp() + ":554/h264Preview_0"
|
||||||
+ (cameraConfig.getNvrChannel() + 1) + "_main";
|
+ (cameraConfig.getNvrChannel() + 1) + "_main";
|
||||||
}
|
}
|
||||||
|
if (mjpegUri.isEmpty()) {
|
||||||
|
mjpegUri = "rtsp://" + cameraConfig.getIp() + ":554/h264Preview_0"
|
||||||
|
+ (cameraConfig.getNvrChannel() + 1) + "_sub";
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// for poll times 9 seconds and above don't display a warning about the Image channel.
|
// for poll times 9 seconds and above don't display a warning about the Image channel.
|
||||||
@ -1778,7 +1792,7 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
private void keepMjpegRunning() {
|
private void keepMjpegRunning() {
|
||||||
CameraServlet localServlet = servlet;
|
CameraServlet localServlet = servlet;
|
||||||
if (localServlet != null && !localServlet.openStreams.isEmpty()) {
|
if (localServlet != null && !localServlet.openStreams.isEmpty()) {
|
||||||
if (!mjpegUri.isEmpty() && !"ffmpeg".equals(mjpegUri)) {
|
if (!usingRtspForMjpeg()) {
|
||||||
localServlet.openStreams.queueFrame(("--" + localServlet.openStreams.boundary + "\r\n\r\n").getBytes());
|
localServlet.openStreams.queueFrame(("--" + localServlet.openStreams.boundary + "\r\n\r\n").getBytes());
|
||||||
}
|
}
|
||||||
localServlet.openStreams.queueFrame(getSnapshot());
|
localServlet.openStreams.queueFrame(getSnapshot());
|
||||||
@ -1876,6 +1890,10 @@ public class IpCameraHandler extends BaseThingHandler {
|
|||||||
return cameraConfig.getIpWhitelist();
|
return cameraConfig.getIpWhitelist();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean usingRtspForMjpeg() {
|
||||||
|
return (mjpegUri.isEmpty() || "ffmpeg".equals(mjpegUri) || mjpegUri.toLowerCase().startsWith("rtsp://"));
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Collection<Class<? extends ThingHandlerService>> getServices() {
|
public Collection<Class<? extends ThingHandlerService>> getServices() {
|
||||||
return Set.of(IpCameraActions.class);
|
return Set.of(IpCameraActions.class);
|
||||||
|
@ -368,7 +368,7 @@ public class OnvifConnection {
|
|||||||
if (xml != null) {
|
if (xml != null) {
|
||||||
rtspUri = xml;
|
rtspUri = xml;
|
||||||
logger.debug("GetStreamUri: {}", rtspUri);
|
logger.debug("GetStreamUri: {}", rtspUri);
|
||||||
if (ipCameraHandler.cameraConfig.getFfmpegInput().isEmpty()) {
|
if (ipCameraHandler.rtspUri.isEmpty()) {// only use if not hard coded in initialize()
|
||||||
ipCameraHandler.rtspUri = rtspUri;
|
ipCameraHandler.rtspUri = rtspUri;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -184,12 +184,12 @@ public class CameraServlet extends IpCameraServlet {
|
|||||||
if (openStreams.isEmpty()) {
|
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();
|
||||||
if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
|
if (handler.usingRtspForMjpeg()) {
|
||||||
output = new StreamOutput(resp);
|
output = new StreamOutput(resp);
|
||||||
} else {
|
} else {
|
||||||
output = new StreamOutput(resp, handler.mjpegContentType);
|
output = new StreamOutput(resp, handler.mjpegContentType);
|
||||||
}
|
}
|
||||||
} else if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
|
} else if (handler.usingRtspForMjpeg()) {
|
||||||
output = new StreamOutput(resp);
|
output = new StreamOutput(resp);
|
||||||
} else {
|
} else {
|
||||||
ChannelTracking tracker = handler.channelTrackingMap.get(handler.getTinyUrl(handler.mjpegUri));
|
ChannelTracking tracker = handler.channelTrackingMap.get(handler.getTinyUrl(handler.mjpegUri));
|
||||||
|
@ -2419,7 +2419,7 @@
|
|||||||
<label>MJPEG Options</label>
|
<label>MJPEG Options</label>
|
||||||
<description>This gives you direct access to specify your own FFmpeg options to be used for MJPEG streams.
|
<description>This gives you direct access to specify your own FFmpeg options to be used for MJPEG streams.
|
||||||
</description>
|
</description>
|
||||||
<default>-q:v 5 -r 2 -vf scale=640:-2 -update 1</default>
|
<default>-q:v 4 -r 4 -update 1</default>
|
||||||
<advanced>true</advanced>
|
<advanced>true</advanced>
|
||||||
</parameter>
|
</parameter>
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user