Fixes #11170 by introducing an intelligent thread.sleep (getting the duration of the sound, if possible, then wait the appropriate time for letting the sound play). By the way, the method to get the sound duration is not as easy as I thought. Also fix a minor issue with the last volume not propertly saved. And fix some minor warnings by using final local variable. Signed-off-by: Gwendal ROULLEAU <gwendal.roulleau@gmail.com>
This commit is contained in:
parent
56a0449ce1
commit
0a96792a93
|
@ -15,15 +15,20 @@ package org.openhab.binding.pulseaudio.internal;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.net.Socket;
|
import java.net.Socket;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.time.Instant;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ScheduledExecutorService;
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import javazoom.spi.mpeg.sampled.convert.MpegFormatConversionProvider;
|
import javazoom.spi.mpeg.sampled.convert.MpegFormatConversionProvider;
|
||||||
import javazoom.spi.mpeg.sampled.file.MpegAudioFileReader;
|
import javazoom.spi.mpeg.sampled.file.MpegAudioFileReader;
|
||||||
|
|
||||||
|
import javax.sound.sampled.AudioFileFormat;
|
||||||
import javax.sound.sampled.AudioInputStream;
|
import javax.sound.sampled.AudioInputStream;
|
||||||
|
import javax.sound.sampled.AudioSystem;
|
||||||
import javax.sound.sampled.UnsupportedAudioFileException;
|
import javax.sound.sampled.UnsupportedAudioFileException;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
|
@ -38,6 +43,7 @@ import org.openhab.core.audio.UnsupportedAudioStreamException;
|
||||||
import org.openhab.core.library.types.PercentType;
|
import org.openhab.core.library.types.PercentType;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.tritonus.share.sampled.file.TAudioFileFormat;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The audio sink for openhab, implemented by a connection to a pulseaudio sink
|
* The audio sink for openhab, implemented by a connection to a pulseaudio sink
|
||||||
|
@ -87,9 +93,29 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
* @param input
|
* @param input
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
private @Nullable InputStream getPCMStreamFromMp3Stream(InputStream input) {
|
private @Nullable AudioStreamAndDuration getPCMStreamFromMp3Stream(InputStream input) {
|
||||||
try {
|
try {
|
||||||
|
|
||||||
MpegAudioFileReader mpegAudioFileReader = new MpegAudioFileReader();
|
MpegAudioFileReader mpegAudioFileReader = new MpegAudioFileReader();
|
||||||
|
|
||||||
|
int duration = -1;
|
||||||
|
if (input instanceof FixedLengthAudioStream) {
|
||||||
|
final Long audioFileLength = ((FixedLengthAudioStream) input).length();
|
||||||
|
AudioFileFormat audioFileFormat = mpegAudioFileReader.getAudioFileFormat(input);
|
||||||
|
if (audioFileFormat instanceof TAudioFileFormat) {
|
||||||
|
Map<String, Object> taudioFileFormatProperties = ((TAudioFileFormat) audioFileFormat).properties();
|
||||||
|
if (taudioFileFormatProperties.containsKey("mp3.framesize.bytes")
|
||||||
|
&& taudioFileFormatProperties.containsKey("mp3.framerate.fps")) {
|
||||||
|
Integer frameSize = (Integer) taudioFileFormatProperties.get("mp3.framesize.bytes");
|
||||||
|
Float frameRate = (Float) taudioFileFormatProperties.get("mp3.framerate.fps");
|
||||||
|
if (frameSize != null && frameRate != null) {
|
||||||
|
duration = Math.round((audioFileLength / (frameSize * frameRate)) * 1000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
input.reset();
|
||||||
|
}
|
||||||
|
|
||||||
AudioInputStream sourceAIS = mpegAudioFileReader.getAudioInputStream(input);
|
AudioInputStream sourceAIS = mpegAudioFileReader.getAudioInputStream(input);
|
||||||
javax.sound.sampled.AudioFormat sourceFormat = sourceAIS.getFormat();
|
javax.sound.sampled.AudioFormat sourceFormat = sourceAIS.getFormat();
|
||||||
|
|
||||||
|
@ -98,7 +124,8 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
javax.sound.sampled.AudioFormat.Encoding.PCM_SIGNED, sourceFormat.getSampleRate(), 16,
|
javax.sound.sampled.AudioFormat.Encoding.PCM_SIGNED, sourceFormat.getSampleRate(), 16,
|
||||||
sourceFormat.getChannels(), sourceFormat.getChannels() * 2, sourceFormat.getSampleRate(), false);
|
sourceFormat.getChannels(), sourceFormat.getChannels() * 2, sourceFormat.getSampleRate(), false);
|
||||||
|
|
||||||
return mpegconverter.getAudioInputStream(convertFormat, sourceAIS);
|
AudioInputStream audioInputStreamConverted = mpegconverter.getAudioInputStream(convertFormat, sourceAIS);
|
||||||
|
return new AudioStreamAndDuration(audioInputStreamConverted, duration);
|
||||||
|
|
||||||
} catch (IOException | UnsupportedAudioFileException e) {
|
} catch (IOException | UnsupportedAudioFileException e) {
|
||||||
logger.warn("Cannot convert this mp3 stream to pcm stream: {}", e.getMessage());
|
logger.warn("Cannot convert this mp3 stream to pcm stream: {}", e.getMessage());
|
||||||
|
@ -126,10 +153,11 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
* Disconnect the socket to pulseaudio simple protocol
|
* Disconnect the socket to pulseaudio simple protocol
|
||||||
*/
|
*/
|
||||||
public void disconnect() {
|
public void disconnect() {
|
||||||
if (clientSocket != null && isIdle) {
|
final Socket clientSocketLocal = clientSocket;
|
||||||
|
if (clientSocketLocal != null && isIdle) {
|
||||||
logger.debug("Disconnecting");
|
logger.debug("Disconnecting");
|
||||||
try {
|
try {
|
||||||
clientSocket.close();
|
clientSocketLocal.close();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -137,6 +165,23 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private AudioStreamAndDuration getWavAudioAndDuration(AudioStream audioStream) {
|
||||||
|
int duration = -1;
|
||||||
|
if (audioStream instanceof FixedLengthAudioStream) {
|
||||||
|
final Long audioFileLength = ((FixedLengthAudioStream) audioStream).length();
|
||||||
|
try {
|
||||||
|
AudioInputStream audioInputStream = AudioSystem.getAudioInputStream(audioStream);
|
||||||
|
int frameSize = audioInputStream.getFormat().getFrameSize();
|
||||||
|
float frameRate = audioInputStream.getFormat().getFrameRate();
|
||||||
|
float durationInSeconds = (audioFileLength / (frameSize * frameRate));
|
||||||
|
duration = Math.round(durationInSeconds * 1000);
|
||||||
|
} catch (UnsupportedAudioFileException | IOException e) {
|
||||||
|
logger.warn("Error when getting duration information from AudioFile");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new AudioStreamAndDuration(audioStream, duration);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void process(@Nullable AudioStream audioStream)
|
public void process(@Nullable AudioStream audioStream)
|
||||||
throws UnsupportedAudioFormatException, UnsupportedAudioStreamException {
|
throws UnsupportedAudioFormatException, UnsupportedAudioStreamException {
|
||||||
|
@ -145,13 +190,13 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
InputStream audioInputStream = null;
|
AudioStreamAndDuration audioInputStreamAndDuration = null;
|
||||||
try {
|
try {
|
||||||
|
|
||||||
if (AudioFormat.MP3.isCompatible(audioStream.getFormat())) {
|
if (AudioFormat.MP3.isCompatible(audioStream.getFormat())) {
|
||||||
audioInputStream = getPCMStreamFromMp3Stream(audioStream);
|
audioInputStreamAndDuration = getPCMStreamFromMp3Stream(audioStream);
|
||||||
} else if (AudioFormat.WAV.isCompatible(audioStream.getFormat())) {
|
} else if (AudioFormat.WAV.isCompatible(audioStream.getFormat())) {
|
||||||
audioInputStream = audioStream;
|
audioInputStreamAndDuration = getWavAudioAndDuration(audioStream);
|
||||||
} else {
|
} else {
|
||||||
throw new UnsupportedAudioFormatException("pulseaudio audio sink can only play pcm or mp3 stream",
|
throw new UnsupportedAudioFormatException("pulseaudio audio sink can only play pcm or mp3 stream",
|
||||||
audioStream.getFormat());
|
audioStream.getFormat());
|
||||||
|
@ -160,10 +205,23 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
for (int countAttempt = 1; countAttempt <= 2; countAttempt++) { // two attempts allowed
|
for (int countAttempt = 1; countAttempt <= 2; countAttempt++) { // two attempts allowed
|
||||||
try {
|
try {
|
||||||
connectIfNeeded();
|
connectIfNeeded();
|
||||||
if (audioInputStream != null && clientSocket != null) {
|
final Socket clientSocketLocal = clientSocket;
|
||||||
|
if (audioInputStreamAndDuration != null && clientSocketLocal != null) {
|
||||||
// send raw audio to the socket and to pulse audio
|
// send raw audio to the socket and to pulse audio
|
||||||
isIdle = false;
|
isIdle = false;
|
||||||
audioInputStream.transferTo(clientSocket.getOutputStream());
|
Instant start = Instant.now();
|
||||||
|
audioInputStreamAndDuration.inputStream.transferTo(clientSocketLocal.getOutputStream());
|
||||||
|
if (audioInputStreamAndDuration.duration != -1) { // ensure, if the sound has a duration
|
||||||
|
// that we let at least this time for the system to play
|
||||||
|
Instant end = Instant.now();
|
||||||
|
long millisSecondTimedToSendAudioData = Duration.between(start, end).toMillis();
|
||||||
|
if (millisSecondTimedToSendAudioData < audioInputStreamAndDuration.duration) {
|
||||||
|
long timeToSleep = audioInputStreamAndDuration.duration
|
||||||
|
- millisSecondTimedToSendAudioData;
|
||||||
|
logger.debug("Sleep time to let the system play sound : {}", timeToSleep);
|
||||||
|
Thread.sleep(timeToSleep);
|
||||||
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
@ -184,8 +242,8 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
try {
|
try {
|
||||||
if (audioInputStream != null) {
|
if (audioInputStreamAndDuration != null) {
|
||||||
audioInputStream.close();
|
audioInputStreamAndDuration.inputStream.close();
|
||||||
}
|
}
|
||||||
audioStream.close();
|
audioStream.close();
|
||||||
scheduleDisconnect();
|
scheduleDisconnect();
|
||||||
|
@ -219,4 +277,15 @@ public class PulseAudioAudioSink implements AudioSink {
|
||||||
public void setVolume(PercentType volume) {
|
public void setVolume(PercentType volume) {
|
||||||
pulseaudioHandler.setVolume(volume.intValue());
|
pulseaudioHandler.setVolume(volume.intValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class AudioStreamAndDuration {
|
||||||
|
private InputStream inputStream;
|
||||||
|
private int duration;
|
||||||
|
|
||||||
|
public AudioStreamAndDuration(InputStream inputStream, int duration) {
|
||||||
|
super();
|
||||||
|
this.inputStream = inputStream;
|
||||||
|
this.duration = duration + 200; // introduce some delay
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -231,24 +231,28 @@ public class PulseaudioHandler extends BaseThingHandler implements DeviceStatusL
|
||||||
// refresh to get the current volume level
|
// refresh to get the current volume level
|
||||||
bridge.getClient().update();
|
bridge.getClient().update();
|
||||||
device = bridge.getDevice(name);
|
device = bridge.getDevice(name);
|
||||||
savedVolume = device.getVolume();
|
int oldVolume = device.getVolume();
|
||||||
|
int newVolume = oldVolume;
|
||||||
if (command.equals(IncreaseDecreaseType.INCREASE)) {
|
if (command.equals(IncreaseDecreaseType.INCREASE)) {
|
||||||
savedVolume = Math.min(100, savedVolume + 5);
|
newVolume = Math.min(100, oldVolume + 5);
|
||||||
}
|
}
|
||||||
if (command.equals(IncreaseDecreaseType.DECREASE)) {
|
if (command.equals(IncreaseDecreaseType.DECREASE)) {
|
||||||
savedVolume = Math.max(0, savedVolume - 5);
|
newVolume = Math.max(0, oldVolume - 5);
|
||||||
}
|
}
|
||||||
bridge.getClient().setVolumePercent(device, savedVolume);
|
bridge.getClient().setVolumePercent(device, newVolume);
|
||||||
updateState = new PercentType(savedVolume);
|
updateState = new PercentType(newVolume);
|
||||||
|
savedVolume = newVolume;
|
||||||
} else if (command instanceof PercentType) {
|
} else if (command instanceof PercentType) {
|
||||||
DecimalType volume = (DecimalType) command;
|
DecimalType volume = (DecimalType) command;
|
||||||
bridge.getClient().setVolumePercent(device, volume.intValue());
|
bridge.getClient().setVolumePercent(device, volume.intValue());
|
||||||
updateState = (PercentType) command;
|
updateState = (PercentType) command;
|
||||||
|
savedVolume = volume.intValue();
|
||||||
} else if (command instanceof DecimalType) {
|
} else if (command instanceof DecimalType) {
|
||||||
// set volume
|
// set volume
|
||||||
DecimalType volume = (DecimalType) command;
|
DecimalType volume = (DecimalType) command;
|
||||||
bridge.getClient().setVolume(device, volume.intValue());
|
bridge.getClient().setVolume(device, volume.intValue());
|
||||||
updateState = (DecimalType) command;
|
updateState = (DecimalType) command;
|
||||||
|
savedVolume = volume.intValue();
|
||||||
}
|
}
|
||||||
} else if (channelUID.getId().equals(MUTE_CHANNEL)) {
|
} else if (channelUID.getId().equals(MUTE_CHANNEL)) {
|
||||||
if (command instanceof OnOffType) {
|
if (command instanceof OnOffType) {
|
||||||
|
@ -318,6 +322,7 @@ public class PulseaudioHandler extends BaseThingHandler implements DeviceStatusL
|
||||||
AbstractAudioDeviceConfig device = bridge.getDevice(name);
|
AbstractAudioDeviceConfig device = bridge.getDevice(name);
|
||||||
bridge.getClient().setVolumePercent(device, volume);
|
bridge.getClient().setVolumePercent(device, volume);
|
||||||
updateState(VOLUME_CHANNEL, new PercentType(volume));
|
updateState(VOLUME_CHANNEL, new PercentType(volume));
|
||||||
|
savedVolume = volume;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
Loading…
Reference in New Issue