You can install the application in different ways. We recommend to use Docker Compose if not the prebuilt image. But if you dont want to use Docker you can also install it directly on your system.
Install packages:
sudo apt-get install -y ca-certificates curl wget gnupg lsb-release git jq vimDownload and add Docker's official GPG key:
mkdir -p /etc/apt/keyrings
curl -fsSL https://download.docker.com/linux/debian/gpg | sudo gpg --dearmor -o /etc/apt/keyrings/docker.gpg
sudo chmod a+r /etc/apt/keyrings/docker.gpgSet up the Docker repository:
echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.gpg] \
https://download.docker.com/linux/debian \
$(lsb_release -cs) stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/nullInstall Docker and Docker Compose:
sudo apt-get update
sudo apt-get install -y docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-pluginDownload the docker-compose.yml and .env.example file from the repository to a folder on your system:
mkdir linux-voice-assistant
cd linux-voice-assistant
LVA_VERSION=$(curl -s https://api.github.com/repos/ohf-voice/linux-voice-assistant/releases/latest | jq -r .tag_name)
echo "Installing version: " $LVA_VERSION
wget https://raw.githubusercontent.com/ohf-voice/linux-voice-assistant/refs/tags/$LVA_VERSION/docker-compose.yml
wget https://raw.githubusercontent.com/ohf-voice/linux-voice-assistant/refs/tags/$LVA_VERSION/.env.example
cp .env.example .env💡 Note: The LVA_VERSION variable downloads the latest available version of LVA. If you want another version, you can specify it with the LVA_VERSION variable.
You can change the used version in the docker-compose.yml.
| Tag | Description | Example |
|---|---|---|
latest |
Latest stable release | ghcr.io/ohf-voice/linux-voice-assistant:latest |
nightly |
Latest development build | ghcr.io/ohf-voice/linux-voice-assistant:nightly |
x.y.z |
Specific version release | ghcr.io/ohf-voice/linux-voice-assistant:1.0.0 |
x.y |
Major.Minor with auto updates | ghcr.io/ohf-voice/linux-voice-assistant:1.0 |
<branch> |
Branch-specific build | ghcr.io/ohf-voice/linux-voice-assistant:my-branch |
Edit the .env file and change the values to your needs:
vim .envYou can change various settings, for example the audio sounds which are played when the wake word is detected or when the timer is finished. You also need to set the correct user and group id, and the pulseaudio server in the .envfile.
💡 Note: You can exit vim with :wq or :q! if you dont want to save the changes.
Start the application:
docker compose up -d💡 Note: If you want to use the application with a different user, you need to change the user in the .env file. Dont forget to change the UID from the user. The docker container will run until you stop it. It will restart autiomatically after a reboot.
Check if the application is running:
docker compose psCheck the logs:
docker compose logs -fStop the service:
docker-compose downDownload the latest image:
docker-compose pullInstall the required packages for the application:
sudo apt update
sudo apt-get install \
avahi-utils \
pulseaudio-utils \
alsa-utils \
pipewire-bin \
pipewire-alsa \
pipewire-pulse \
build-essential \
libmpv-dev \
libasound2-plugins \
ca-certificates \
iproute2 \
procps \
git \
jq \
curl \
wget \
vim \
python3-venv \
python3-devClone the repository:
git clone https://github.com/OHF-Voice/linux-voice-assistant.git
cd linux-voice-assistant
chmod +x docker-entrypoint.shInstall the application:
script/setup💡 Note: For hardware with limited performance (like Raspberry Pi Zero 2W or other SBCs), you may need to adjust compilation parameters to ensure the build process completes successfully. The script supports custom CXXFLAGS and MAKEFLAGS:
--cxxflags: Custom C++ compilation flags (default:-O1 -g0)--makeflags: Custom make flags (default:-j1)
For example, to optimize for low-performance hardware:
script/setup --cxxflags="-O1 -g0" --makeflags="-j1"Create a systemd service for the application:
sudo systemctl edit --force --full linux-voice-assistant.servicePaste the following content into the file:
[Unit]
Description=Linux-Voice-Assistant
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
User=pi
WorkingDirectory=/home/pi/linux-voice-assistant
Environment=PATH=/home/pi/linux-voice-assistant/.venv/bin:/usr/bin:/bin
# Environment=ENABLE_DEBUG="1"
# Environment=LIST_DEVICES="1"
# Environment=CLIENT_NAME="My Voice Assistant Speaker"
Environment=PULSE_SERVER="/run/user/<replace_with_your_user_id>/pulse/native"
Environment=XDG_RUNTIME_DIR="/run/user/<replace_with_your_user_id>"
Environment=PULSE_COOKIE="/home/pi/linux-voice-assistant/tmp_pulse_cookie"
Environment=PREFERENCES_FILE="/home/pi/linux-voice-assistant/preferences.json"
# Environment=NETWORK_INTERFACE="eth0"
# Environment=HOST="0.0.0.0"
# Environment=PORT="6053"
# Environment=AUDIO_INPUT_DEVICE="default"
# Environment=AUDIO_OUTPUT_DEVICE="default"
# Environment=ENABLE_THINKING_SOUND="1"
# Environment=WAKE-MODEL="okay_nabu"
# Environment=REFACTORY_SECONDS="2"
# Environment=WAKEUP_SOUND="sounds/wake_word_triggered.flac"
# Environment=TIMER_FINISHED_SOUND="sounds/timer_finished.flac"
# Environment=PROCESSING_SOUND="sounds/processing.wav"
# Environment=MUTE_SOUND="sounds/mute_switch_on.flac"
# Environment=UNMUTE_SOUND="sounds/mute_switch_off.flac"
ExecStart=/home/pi/linux-voice-assistant/docker-entrypoint.sh
# ExecStart=/home/pi/linux-voice-assistant/docker-entrypoint.sh --additional-parameter-if-you-want
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target💡 Note: We are using the docker-entrypoint.sh script to start the application. This script is located in the root of the repository. But there is no docker used. Only the start script is used.
💡 Note: Replace pi with your actual user that you want to run the voice assistant. You need to run Pipewire and LVA with the same user in order to provide access to the audio socket. You can also add the group audio to the user which LVA is running on. sudo usermod -a -G audio pi
Reload the systemd daemon and start the service:
sudo systemctl daemon-reexec
sudo systemctl daemon-reload
sudo systemctl enable linux-voice-assistant
sudo systemctl start linux-voice-assistant💡 Note: If you want to change settings you can do this with editing the Environment in the service file.
If you want to check if the service is running you can use the following command:
sudo systemctl status linux-voice-assistant- In Home Assistant, go to "Settings" -> "Device & services"
- Click the "Add integration" button
- Choose "ESPHome" and then "Set up another instance of ESPHome"
- Enter the IP address of your voice satellite with port 6053
- Click "Submit"
All optional settings except the audio sounds are disabled by default. You can change them in the UI and they will be saved in the PREFERENCES_FILE. If you set the configuration in the service file, the corresponding settings will be overwritten and you are no longer able to change them in the UI.
If you want to change the audio sounds you need to change the paths in the service file.
The following variables can be configured in the .env or in the service file:
| Variable | Default | Description |
|---|---|---|
LVA_USER_ID |
1000 |
User ID for the container (usually 1000 for the first user) |
LVA_USER_GROUP |
1000 |
GROUP ID for the container (usually 1000 for the first users group) |
CLIENT_NAME |
(optional) | Custom name for this voice assistant instance |
LVA_PULSE_SERVER |
/run/user/${LVA_USER_ID}/pulse/native |
Path to the PulseAudio/PipeWire socket (In some cases a :unixinfront of the path is needed) |
LVA_XDG_RUNTIME_DIR |
/run/user/${LVA_USER_ID} |
XDG runtime directory |
LVA_PULSE_COOKIE |
/app/configuration/tmp_pulse_cookie |
Cookie file for PulseAudio if you use encryption. By default disabled. We use a tmp file to avoid errors if the file is not found |
ENABLE_DEBUG |
(optional) | Set to "1" to enable debug mode |
LIST_DEVICES |
(optional) | Set to "1" to list audio devices instead of starting |
PREFERENCES_FILE |
(optional) | Path to a custom preferences JSON file |
NETWORK_INTERFACE |
Autodetected | network card for server |
HOST |
Autodetected | API server IP-Address, can be 0.0.0.0 for all interfaces, but only one network card works for MAC-ADDRESS and ESP protocol |
PORT |
6053 |
API server port |
AUDIO_INPUT_DEVICE |
Autodetected | Audio input device name |
AUDIO_OUTPUT_DEVICE |
Autodetected | Audio output device name |
ENABLE_THINKING_SOUND |
false | Set to "1" to enable thinking sound |
WAKE_WORD_DIR |
app/wakewords |
Path to the wake word directory |
WAKE_MODEL |
okay_nabu |
Wake word model to use |
REFACTORY_SECONDS |
2 |
Refractory period in seconds after wake word |
WAKEUP_SOUND |
sounds/wake_word_triggered.flac |
Sound file for wake word triggered |
TIMER_FINISHED_SOUND |
sounds/timer_finished.flac |
Sound file for timer finished |
PROCESSING_SOUND |
sounds/processing.wav |
Sound file for processing state |
MUTE_SOUND |
sounds/mute_switch_on.flac |
Sound file for mute on |
UNMUTE_SOUND |
sounds/mute_switch_off.flac |
Sound file for Configure Audio Devices |
💡 Note: For the systemd installation some variables set in the service need to be without LVA_ prefix.
If you want to use your own sounds, you can add them to the sounds/custom aka /var/lib/docker/volumes/lva_sounds_custom/_data directory and reference them in the .env file.
The following wake word models are available:
okay_nabu- Default wake wordalexa- Alexa wake wordhey_jarvis- Jarvis wake wordhey_mycroft- Mycroft wake wordhey_luna- Luna wake wordhey_home_assistant- Home Assistant wake wordstop- Stop wake wordokay_computer- Okay Computer wake wordchoo_choo_homie- Choo Choo Homie wake word
If you want to use your own wakewords, you can add them to the wakewords/custom aka /var/lib/docker/volumes/lva_wakeword_custom/_data directory and reference them in the .env file.
Change the default wake word with WAKE_MODEL <id> where <id> is the name of a model in the wakewords directory. For example, WAKE_MODEL hey_jarvis will load app/wakewords/hey_jarvis.tflite by default.
You can include more wakeword directories by adding WAKE_WORD_DIR <DIR> where <DIR> contains either [microWakeWord][] or [openWakeWord][] config files and .tflite models. For example, WAKE_WORD_DIR app/wakewords/openWakeWord will include the default wake words for openWakeWord.
If you want to add [other wakewords][wakewords-collection], make sure to create a small JSON config file to identify it as an openWakeWord model. For example, download the [GLaDOS][glados] model to glados.tflite and create glados.json with:
{
"type": "openWakeWord",
"wake_word": "GLaDOS",
"model": "glados.tflite"
}Add WAKE_WORD_DIR <DIR> with the directory containing glados.tflite and glados.json to your command-line.
More wakewords can be found in the wakewords-collection repository.