[ipcamera] Make sure created Servlet supports async (#14552)

Fixes the folowing errors:

```
HTTP ERROR 500 java.lang.IllegalStateException: !asyncSupported: NotAsync:org.ops4j.pax.web.service.spi.servlet.OsgiInitializedServlet@536b0858
URI:	/ipcamera/192168493/ipcamera.jpg
STATUS:	500
MESSAGE:	java.lang.IllegalStateException: !asyncSupported: NotAsync:org.ops4j.pax.web.service.spi.servlet.OsgiInitializedServlet@536b0858
SERVLET:	org.openhab.binding.ipcamera.internal.servlet.CameraServlet
CAUSED BY:	java.lang.IllegalStateException: !asyncSupported: NotAsync:org.ops4j.pax.web.service.spi.servlet.OsgiInitializedServlet@536b0858
Powered by Jetty:// 9.4.50.v20221201
```

See:

* https://community.openhab.org/t/openhab-4-0-snapshot-discussion/142322/226
* https://groups.google.com/g/ops4j/c/E9p7tPydPmo
* https://github.com/ops4j/org.ops4j.pax.web/issues/1767

Signed-off-by: Wouter Born <github@maindrain.net>
This commit is contained in:
Wouter Born 2023-03-08 17:20:08 +01:00 committed by GitHub
parent 5c0b5711a8
commit 50cdd02447
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 23 additions and 11 deletions

View File

@ -17,6 +17,9 @@ import static org.openhab.binding.ipcamera.internal.IpCameraBindingConstants.HLS
import java.io.IOException; import java.io.IOException;
import java.time.Duration; import java.time.Duration;
import java.time.Instant; import java.time.Instant;
import java.util.Dictionary;
import java.util.Hashtable;
import java.util.Map;
import javax.servlet.AsyncContext; import javax.servlet.AsyncContext;
import javax.servlet.ServletInputStream; import javax.servlet.ServletInputStream;
@ -41,13 +44,16 @@ import org.osgi.service.http.HttpService;
@NonNullByDefault @NonNullByDefault
public class CameraServlet extends IpCameraServlet { public class CameraServlet extends IpCameraServlet {
private static final long serialVersionUID = -134658667574L; private static final long serialVersionUID = -134658667574L;
private static final Dictionary<Object, Object> INIT_PARAMETERS = new Hashtable<>(
Map.of("async-supported", "true"));
private final IpCameraHandler handler; private final IpCameraHandler handler;
public OpenStreams openStreams = new OpenStreams(); public OpenStreams openStreams = new OpenStreams();
private OpenStreams openSnapshotStreams = new OpenStreams(); private OpenStreams openSnapshotStreams = new OpenStreams();
private OpenStreams openAutoFpsStreams = new OpenStreams(); private OpenStreams openAutoFpsStreams = new OpenStreams();
public CameraServlet(IpCameraHandler handler, HttpService httpService) { public CameraServlet(IpCameraHandler handler, HttpService httpService) {
super(handler, httpService); super(handler, httpService, INIT_PARAMETERS);
this.handler = handler; this.handler = handler;
} }
@ -183,17 +189,15 @@ public class CameraServlet extends IpCameraServlet {
} else { } else {
output = new StreamOutput(resp, handler.mjpegContentType); output = new StreamOutput(resp, handler.mjpegContentType);
} }
} else if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) {
output = new StreamOutput(resp);
} else { } else {
if (handler.mjpegUri.isEmpty() || "ffmpeg".equals(handler.mjpegUri)) { ChannelTracking tracker = handler.channelTrackingMap.get(handler.getTinyUrl(handler.mjpegUri));
output = new StreamOutput(resp); if (tracker == null || !tracker.getChannel().isOpen()) {
} else { logger.debug("Not the first stream requested but the stream from camera was closed");
ChannelTracking tracker = handler.channelTrackingMap.get(handler.getTinyUrl(handler.mjpegUri)); handler.openCamerasStream();
if (tracker == null || !tracker.getChannel().isOpen()) {
logger.debug("Not the first stream requested but the stream from camera was closed");
handler.openCamerasStream();
}
output = new StreamOutput(resp, handler.mjpegContentType);
} }
output = new StreamOutput(resp, handler.mjpegContentType);
} }
openStreams.addStream(output); openStreams.addStream(output);
do { do {

View File

@ -17,6 +17,8 @@ import java.io.BufferedOutputStream;
import java.io.File; import java.io.File;
import java.io.FileInputStream; import java.io.FileInputStream;
import java.io.IOException; import java.io.IOException;
import java.util.Dictionary;
import java.util.Properties;
import javax.servlet.ServletOutputStream; import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet; import javax.servlet.annotation.WebServlet;
@ -42,16 +44,22 @@ public abstract class IpCameraServlet extends HttpServlet {
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
protected final ThingHandler handler; protected final ThingHandler handler;
protected final HttpService httpService; protected final HttpService httpService;
protected final Dictionary<Object, Object> initParameters;
public IpCameraServlet(ThingHandler handler, HttpService httpService) { public IpCameraServlet(ThingHandler handler, HttpService httpService) {
this(handler, httpService, new Properties());
}
public IpCameraServlet(ThingHandler handler, HttpService httpService, Dictionary<Object, Object> initParameters) {
this.handler = handler; this.handler = handler;
this.httpService = httpService; this.httpService = httpService;
this.initParameters = initParameters;
startListening(); startListening();
} }
public void startListening() { public void startListening() {
try { try {
httpService.registerServlet("/ipcamera/" + handler.getThing().getUID().getId(), this, null, httpService.registerServlet("/ipcamera/" + handler.getThing().getUID().getId(), this, initParameters,
httpService.createDefaultHttpContext()); httpService.createDefaultHttpContext());
} catch (Exception e) { } catch (Exception e) {
logger.warn("Registering servlet failed:{}", e.getMessage()); logger.warn("Registering servlet failed:{}", e.getMessage());