mirror of
https://github.com/danieldemus/openhab-core.git
synced 2025-01-10 21:31:53 +01:00
[voice] Add console commands for register/unregister dialogs and list them (#3459)
* [voice] Add voice commands for register/unregister dialogs and list dialogs and dialog registrations Signed-off-by: Miguel Álvarez <miguelwork92@gmail.com>
This commit is contained in:
parent
3ec1457583
commit
6b914162bf
@ -135,6 +135,11 @@ public interface VoiceManager {
|
||||
@Nullable
|
||||
DialogContext getLastDialogContext();
|
||||
|
||||
/**
|
||||
* Returns a list with the contexts of all running dialogs.
|
||||
*/
|
||||
List<DialogContext> getDialogsContexts();
|
||||
|
||||
/**
|
||||
* Starts an infinite dialog sequence: keyword spotting on the audio source, audio source listening to retrieve
|
||||
* a question or a command (Speech to Text service), interpretation and handling of the command, and finally
|
||||
|
@ -223,6 +223,13 @@ public class DialogProcessor implements KSListener, STTListener {
|
||||
eventListener.onDialogStopped(dialogContext);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the dialog context used to start this processor.
|
||||
*/
|
||||
public DialogContext getContext() {
|
||||
return dialogContext;
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates if voice recognition is running.
|
||||
*/
|
||||
|
@ -20,6 +20,7 @@ import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Locale;
|
||||
import java.util.Objects;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
@ -33,6 +34,7 @@ import org.openhab.core.items.ItemNotFoundException;
|
||||
import org.openhab.core.items.ItemNotUniqueException;
|
||||
import org.openhab.core.items.ItemRegistry;
|
||||
import org.openhab.core.voice.DialogContext;
|
||||
import org.openhab.core.voice.DialogRegistration;
|
||||
import org.openhab.core.voice.KSService;
|
||||
import org.openhab.core.voice.STTService;
|
||||
import org.openhab.core.voice.TTSService;
|
||||
@ -60,7 +62,11 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
private static final String SUBCMD_VOICES = "voices";
|
||||
private static final String SUBCMD_START_DIALOG = "startdialog";
|
||||
private static final String SUBCMD_STOP_DIALOG = "stopdialog";
|
||||
private static final String SUBCMD_REGISTER_DIALOG = "registerdialog";
|
||||
private static final String SUBCMD_UNREGISTER_DIALOG = "unregisterdialog";
|
||||
private static final String SUBCMD_LISTEN_ANSWER = "listenandanswer";
|
||||
private static final String SUBCMD_DIALOGS = "dialogs";
|
||||
private static final String SUBCMD_DIALOG_REGS = "dialogregs";
|
||||
private static final String SUBCMD_INTERPRETERS = "interpreters";
|
||||
private static final String SUBCMD_KEYWORD_SPOTTERS = "keywordspotters";
|
||||
private static final String SUBCMD_STT_SERVICES = "sttservices";
|
||||
@ -87,13 +93,21 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
return List.of(buildCommandUsage(SUBCMD_SAY + " <text>", "speaks a text"),
|
||||
buildCommandUsage(SUBCMD_INTERPRET + " <command>", "interprets a human language command"),
|
||||
buildCommandUsage(SUBCMD_VOICES, "lists available voices of the TTS services"),
|
||||
buildCommandUsage(SUBCMD_DIALOGS, "lists the running dialog and their audio/voice services"),
|
||||
buildCommandUsage(SUBCMD_DIALOG_REGS,
|
||||
"lists the existing dialog registrations and their selected audio/voice services"),
|
||||
buildCommandUsage(SUBCMD_REGISTER_DIALOG
|
||||
+ " [--source <source>] [--sink <sink>] [--hlis <comma,separated,interpreters>] [--tts <tts> [--voice <voice>]] [--stt <stt>] [--ks ks [--keyword <ks>]] [--listening-item <listeningItem>]",
|
||||
"register a new dialog processing using the default services or the services identified with provided arguments, it will be persisted and keep running whenever is possible."),
|
||||
buildCommandUsage(SUBCMD_UNREGISTER_DIALOG + " [source]",
|
||||
"unregister the dialog processing for the default audio source or the audio source identified with provided argument, stopping it if started"),
|
||||
buildCommandUsage(SUBCMD_START_DIALOG
|
||||
+ " [--source <source>] [--sink <sink>] [--interpreters <comma,separated,interpreters>] [--tts <tts> [--voice <voice>]] [--stt <stt>] [--ks ks [--keyword <ks>]] [--listening-item <listeningItem>]",
|
||||
+ " [--source <source>] [--sink <sink>] [--hlis <comma,separated,interpreters>] [--tts <tts> [--voice <voice>]] [--stt <stt>] [--ks ks [--keyword <ks>]] [--listening-item <listeningItem>]",
|
||||
"start a new dialog processing using the default services or the services identified with provided arguments"),
|
||||
buildCommandUsage(SUBCMD_STOP_DIALOG + " [<source>]",
|
||||
"stop the dialog processing for the default audio source or the audio source identified with provided argument"),
|
||||
buildCommandUsage(SUBCMD_LISTEN_ANSWER
|
||||
+ " [--source <source>] [--sink <sink>] [--interpreters <comma,separated,interpreters>] [--tts <tts> [--voice <voice>]] [--stt <stt>] [--listening-item <listeningItem>]",
|
||||
+ " [--source <source>] [--sink <sink>] [--hlis <comma,separated,interpreters>] [--tts <tts> [--voice <voice>]] [--stt <stt>] [--listening-item <listeningItem>]",
|
||||
"Execute a simple dialog sequence without keyword spotting using the default services or the services identified with provided arguments"),
|
||||
buildCommandUsage(SUBCMD_INTERPRETERS, "lists the interpreters"),
|
||||
buildCommandUsage(SUBCMD_KEYWORD_SPOTTERS, "lists the keyword spotters"),
|
||||
@ -135,10 +149,41 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
}
|
||||
return;
|
||||
}
|
||||
case SUBCMD_REGISTER_DIALOG -> {
|
||||
DialogRegistration dialogRegistration;
|
||||
try {
|
||||
dialogRegistration = parseDialogRegistration(args);
|
||||
} catch (IllegalStateException e) {
|
||||
console.println(Objects.requireNonNullElse(e.getMessage(),
|
||||
"An error occurred while parsing the dialog options"));
|
||||
break;
|
||||
}
|
||||
try {
|
||||
voiceManager.registerDialog(dialogRegistration);
|
||||
} catch (IllegalStateException e) {
|
||||
console.println(Objects.requireNonNullElse(e.getMessage(),
|
||||
"An error occurred while registering the dialog"));
|
||||
}
|
||||
return;
|
||||
}
|
||||
case SUBCMD_UNREGISTER_DIALOG -> {
|
||||
try {
|
||||
var sourceId = args.length < 2 ? audioManager.getSourceId() : args[1];
|
||||
if (sourceId == null) {
|
||||
console.println("No source provided nor default source available");
|
||||
break;
|
||||
}
|
||||
voiceManager.unregisterDialog(sourceId);
|
||||
} catch (IllegalStateException e) {
|
||||
console.println(Objects.requireNonNullElse(e.getMessage(),
|
||||
"An error occurred while stopping the dialog"));
|
||||
}
|
||||
return;
|
||||
}
|
||||
case SUBCMD_START_DIALOG -> {
|
||||
DialogContext.Builder dialogContextBuilder;
|
||||
try {
|
||||
dialogContextBuilder = parseDialogParameters(args);
|
||||
dialogContextBuilder = parseDialogContext(args);
|
||||
} catch (IllegalStateException e) {
|
||||
console.println(Objects.requireNonNullElse(e.getMessage(),
|
||||
"An error occurred while parsing the dialog options"));
|
||||
@ -164,7 +209,7 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
case SUBCMD_LISTEN_ANSWER -> {
|
||||
DialogContext.Builder dialogContextBuilder;
|
||||
try {
|
||||
dialogContextBuilder = parseDialogParameters(args);
|
||||
dialogContextBuilder = parseDialogContext(args);
|
||||
} catch (IllegalStateException e) {
|
||||
console.println(Objects.requireNonNullElse(e.getMessage(),
|
||||
"An error occurred while parsing the dialog options"));
|
||||
@ -178,6 +223,14 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
}
|
||||
return;
|
||||
}
|
||||
case SUBCMD_DIALOGS -> {
|
||||
listDialogs(console);
|
||||
return;
|
||||
}
|
||||
case SUBCMD_DIALOG_REGS -> {
|
||||
listDialogRegistrations(console);
|
||||
return;
|
||||
}
|
||||
case SUBCMD_INTERPRETERS -> {
|
||||
listInterpreters(console);
|
||||
return;
|
||||
@ -252,6 +305,42 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
voiceManager.say(msg.toString());
|
||||
}
|
||||
|
||||
private void listDialogRegistrations(Console console) {
|
||||
Collection<DialogRegistration> registrations = voiceManager.getDialogRegistrations();
|
||||
if (!registrations.isEmpty()) {
|
||||
registrations.stream().sorted(comparing(dr -> dr.sourceId)).forEach(dr -> {
|
||||
console.println(
|
||||
String.format(" Source: %s - Sink: %s (STT: %s, TTS: %s, HLIs: %s, KS: %s, Keyword: %s)",
|
||||
dr.sourceId, dr.sinkId, getOrDefault(dr.sttId), getOrDefault(dr.ttsId),
|
||||
dr.hliIds.isEmpty() ? getOrDefault(null) : String.join("->", dr.hliIds),
|
||||
getOrDefault(dr.ksId), getOrDefault(dr.keyword)));
|
||||
});
|
||||
} else {
|
||||
console.println("No dialog registrations.");
|
||||
}
|
||||
}
|
||||
|
||||
private String getOrDefault(@Nullable String value) {
|
||||
return value != null && !value.isBlank() ? value : "**Default**";
|
||||
}
|
||||
|
||||
private void listDialogs(Console console) {
|
||||
Collection<DialogContext> dialogContexts = voiceManager.getDialogsContexts();
|
||||
if (!dialogContexts.isEmpty()) {
|
||||
dialogContexts.stream().sorted(comparing(s -> s.source().getId())).forEach(c -> {
|
||||
var ks = c.ks();
|
||||
String ksText = ks != null ? String.format(", KS: %s, Keyword: %s", ks.getId(), c.keyword()) : "";
|
||||
console.println(
|
||||
String.format(" Source: %s - Sink: %s (STT: %s, TTS: %s, HLIs: %s%s)", c.source().getId(),
|
||||
c.sink().getId(), c.stt().getId(), c.tts().getId(), c.hlis().stream()
|
||||
.map(HumanLanguageInterpreter::getId).collect(Collectors.joining("->")),
|
||||
ksText));
|
||||
});
|
||||
} else {
|
||||
console.println("No running dialogs.");
|
||||
}
|
||||
}
|
||||
|
||||
private void listInterpreters(Console console) {
|
||||
Collection<HumanLanguageInterpreter> interpreters = voiceManager.getHLIs();
|
||||
if (!interpreters.isEmpty()) {
|
||||
@ -314,11 +403,7 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
.orElse(null);
|
||||
}
|
||||
|
||||
private DialogContext.Builder parseDialogParameters(String[] args) {
|
||||
var dialogContextBuilder = voiceManager.getDialogContextBuilder();
|
||||
if (args.length < 2) {
|
||||
return dialogContextBuilder;
|
||||
}
|
||||
private HashMap<String, String> parseDialogParameters(String[] args) {
|
||||
var parameters = new HashMap<String, String>();
|
||||
for (int i = 1; i < args.length; i++) {
|
||||
var arg = args[i].trim();
|
||||
@ -333,6 +418,15 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
throw new IllegalStateException("Argument name should start by -- " + arg);
|
||||
}
|
||||
}
|
||||
return parameters;
|
||||
}
|
||||
|
||||
private DialogContext.Builder parseDialogContext(String[] args) {
|
||||
var dialogContextBuilder = voiceManager.getDialogContextBuilder();
|
||||
if (args.length < 2) {
|
||||
return dialogContextBuilder;
|
||||
}
|
||||
var parameters = parseDialogParameters(args);
|
||||
String sourceId = parameters.remove("source");
|
||||
if (sourceId != null) {
|
||||
var source = audioManager.getSource(sourceId);
|
||||
@ -363,4 +457,40 @@ public class VoiceConsoleCommandExtension extends AbstractConsoleCommandExtensio
|
||||
}
|
||||
return dialogContextBuilder;
|
||||
}
|
||||
|
||||
private DialogRegistration parseDialogRegistration(String[] args) {
|
||||
var parameters = parseDialogParameters(args);
|
||||
@Nullable
|
||||
String sourceId = parameters.remove("source");
|
||||
if (sourceId == null) {
|
||||
sourceId = audioManager.getSourceId();
|
||||
}
|
||||
if (sourceId == null) {
|
||||
throw new IllegalStateException("A source is required if the default is not configured");
|
||||
}
|
||||
@Nullable
|
||||
String sinkId = parameters.remove("sink");
|
||||
if (sinkId == null) {
|
||||
sinkId = audioManager.getSinkId();
|
||||
}
|
||||
if (sinkId == null) {
|
||||
throw new IllegalStateException("A sink is required if the default is not configured");
|
||||
}
|
||||
var dr = new DialogRegistration(sourceId, sinkId);
|
||||
dr.ksId = parameters.remove("ks");
|
||||
dr.keyword = parameters.remove("keyword");
|
||||
dr.sttId = parameters.remove("stt");
|
||||
dr.ttsId = parameters.remove("tts");
|
||||
dr.voiceId = parameters.remove("voice");
|
||||
dr.listeningItem = parameters.remove("listening-item");
|
||||
String hliIds = parameters.remove("hlis");
|
||||
if (hliIds != null) {
|
||||
dr.hliIds = Arrays.stream(hliIds.split(",")).map(String::trim).collect(Collectors.toList());
|
||||
}
|
||||
if (!parameters.isEmpty()) {
|
||||
throw new IllegalStateException(
|
||||
"Argument " + parameters.keySet().stream().findAny().orElse("") + " is not supported");
|
||||
}
|
||||
return dr;
|
||||
}
|
||||
}
|
||||
|
@ -495,6 +495,11 @@ public class VoiceManagerImpl implements VoiceManager, ConfigOptionProvider, Dia
|
||||
.withListeningItem(listeningItem);
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<DialogContext> getDialogsContexts() {
|
||||
return dialogProcessors.values().stream().map(DialogProcessor::getContext).collect(Collectors.toList());
|
||||
}
|
||||
|
||||
@Override
|
||||
public @Nullable DialogContext getLastDialogContext() {
|
||||
return lastDialogContext;
|
||||
|
Loading…
Reference in New Issue
Block a user