From 417c5b00fb7511f19a6f373e38e945a405ede19a Mon Sep 17 00:00:00 2001 From: Laurent Garnier Date: Sat, 8 Jul 2023 13:43:45 +0200 Subject: [PATCH] [freeboxos] Align audio sink to freebox binding Related to #15113 Signed-off-by: Laurent Garnier --- .../internal/handler/AirMediaSink.java | 97 ++++++++++++------- 1 file changed, 61 insertions(+), 36 deletions(-) diff --git a/bundles/org.openhab.binding.freeboxos/src/main/java/org/openhab/binding/freeboxos/internal/handler/AirMediaSink.java b/bundles/org.openhab.binding.freeboxos/src/main/java/org/openhab/binding/freeboxos/internal/handler/AirMediaSink.java index 7d0d480011166..deaed073e65ca 100644 --- a/bundles/org.openhab.binding.freeboxos/src/main/java/org/openhab/binding/freeboxos/internal/handler/AirMediaSink.java +++ b/bundles/org.openhab.binding.freeboxos/src/main/java/org/openhab/binding/freeboxos/internal/handler/AirMediaSink.java @@ -15,6 +15,7 @@ import static org.openhab.core.audio.AudioFormat.*; import java.io.IOException; +import java.io.InputStream; import java.util.HashSet; import java.util.Locale; import java.util.Set; @@ -47,7 +48,8 @@ @NonNullByDefault public class AirMediaSink extends AudioSinkAsync { private static final Set> SUPPORTED_STREAMS = Set.of(AudioStream.class); - private static final Set BASIC_FORMATS = Set.of(WAV, OGG); + // OGG seems to not be properly supported (tested with a file produced by VoiceRSS) + private static final Set BASIC_FORMATS = Set.of(WAV/* , OGG */); private static final Set ALL_MP3_FORMATS = Set.of( new AudioFormat(CONTAINER_NONE, CODEC_MP3, null, null, 96000, null), new AudioFormat(CONTAINER_NONE, CODEC_MP3, null, null, 112000, null), @@ -111,46 +113,69 @@ public String getId() { @Override protected void processAsynchronously(@Nullable AudioStream audioStream) throws UnsupportedAudioFormatException, UnsupportedAudioStreamException { - if (thingHandler.getThing().getStatus() == ThingStatus.ONLINE) { + if (thingHandler.getThing().getStatus() != ThingStatus.ONLINE) { + tryClose(audioStream); + return; + } + + if (audioStream == null) { + stopMedia(); + return; + } + + String url; + if (audioStream instanceof URLAudioStream urlAudioStream) { + // it is an external URL, we can access it directly + url = urlAudioStream.getURL(); + tryClose(audioStream); + } else { + // we serve it on our own HTTP server + logger.debug("audioStream {} {}", audioStream.getClass().getSimpleName(), audioStream.getFormat()); + StreamServed streamServed; try { - MediaReceiverManager manager = thingHandler.getManager(MediaReceiverManager.class); - if (audioStream == null) { - manager.sendToReceiver(playerName, password, Action.STOP, MediaType.VIDEO); - return; - } - - if (audioStream instanceof URLAudioStream urlAudioStream) { - // it is an external URL, we can access it directly - logger.debug("AirPlay audio sink: process url {}", urlAudioStream.getURL()); - playMedia(manager, urlAudioStream.getURL()); - return; - } - // we serve it on our own HTTP server - StreamServed streamServed; - try { - streamServed = audioHTTPServer.serve(audioStream, 5, true); - } catch (IOException e) { - try { - audioStream.close(); - } catch (IOException ex) { - logger.debug("Exception while closing audioStream"); - } - throw new UnsupportedAudioStreamException( - "AirPlay device was not able to handle the audio stream (cache on disk failed).", - audioStream.getClass(), e); - } - streamServed.playEnd().thenRun(() -> this.playbackFinished(audioStream)); - logger.debug("AirPlay audio sink: process url {}", callbackUrl + streamServed.url()); - playMedia(manager, callbackUrl + streamServed.url()); - } catch (FreeboxException e) { - logger.warn("Audio stream playback failed: {}", e.getMessage()); + streamServed = audioHTTPServer.serve(audioStream, 5, true); + } catch (IOException e) { + tryClose(audioStream); + throw new UnsupportedAudioStreamException( + "AirPlay device was not able to handle the audio stream (cache on disk failed).", + audioStream.getClass(), e); } + url = callbackUrl + streamServed.url(); + streamServed.playEnd().thenRun(() -> { + stopMedia(); + this.playbackFinished(audioStream); + }); } + logger.debug("AirPlay audio sink: process url {}", url); + playMedia(url); } - private void playMedia(MediaReceiverManager manager, String url) throws FreeboxException { - manager.sendToReceiver(playerName, password, Action.STOP, MediaType.VIDEO); - manager.sendToReceiver(playerName, password, Action.START, MediaType.VIDEO, url); + private void tryClose(@Nullable InputStream is) { + if (is != null) { + try { + is.close(); + } catch (IOException ignored) { + } + } + } + + private void playMedia(String url) { + try { + MediaReceiverManager manager = thingHandler.getManager(MediaReceiverManager.class); + manager.sendToReceiver(playerName, password, Action.STOP, MediaType.VIDEO); + manager.sendToReceiver(playerName, password, Action.START, MediaType.VIDEO, url); + } catch (FreeboxException e) { + logger.warn("Playing media failed: {}", e.getMessage()); + } + } + + private void stopMedia() { + try { + MediaReceiverManager manager = thingHandler.getManager(MediaReceiverManager.class); + manager.sendToReceiver(playerName, password, Action.STOP, MediaType.VIDEO); + } catch (FreeboxException e) { + logger.warn("Stopping media failed: {}", e.getMessage()); + } } @Override