Matching short audio segments of speech with each other.
Project description
The speechmatching package allows for calculating similarity scores between short audio fragments containing speech. It then uses these similarity scores to find matches over multiple groups of recordings.
This is especially useful for rare languages from which not enough data exists to train a fully capable ASR model, or when the resources for training such a model are not available.
This package, along with the acoustic binary was created in combination with the thesis on Speech Matching [thesis] by Auke Schuringa for the University of Amsterdam and the Vrije Universiteit in the Netherlands.
Auke Schuringa, “Energy-efficient phonetic matching for spoken words of rare indigenous languages in low-resource environments,” Master’s thesis, Vrije Universiteit Amsterdam and University of Amsterdam, Amsterdam, The Netherlands, Jan. 2025.
Setup
The repository is made up of two parts. The first is the speechmatching package itself, and the second is the acoustic binary located in the acoustic/ directory.
For details on setting up the acoustic package, please see its documentation. Because the acoustic package has specific dependencies and can be somewhat complex to build, it is recommended to run it using Docker.
The speechmatching Python package will by default look for the aukesch/speechmatching Docker image locally, and attempt to pull it if not available. For this, Docker needs to be installed.
Get Docker by installing it:
curl https://get.docker.com/ | sudo sh
Then, grant permissions to the current user so Docker can be run without sudo:
sudo groupadd docker sudo usermod -aG docker $USER newgrp docker
The speechmatching package requires ffmpeg for converting files with an audio stream to a usable format. Using apt, this can be installed with:
sudo apt-get update sudo apt-get install ffmpeg
Finally, install the speechmatching package from PyPI [package]:
pip install speechmatching
Or from the cloned repository:
pip install .
Optionally, create a virtual environment first:
sudo apt-get install python3-venv
python3 -m venv env
source env/bin/activate
pip install .
speechmatching on PyPI https://pypi.org/project/speechmatching/
Development
During development, it may be useful to have a separate Docker image that is not named aukesch/speechmatching, which can then contain changes made during development. To support this, the speechmatching package will look for a Docker image named speechmatching before it looks for the Docker image aukesch/speechmatching. If the first docker image is available, the image under aukesch/ will not be used.
Usage
There are two ways to use the speechmatching package - either locally or from within Docker.
Local
Running locally requires (automatically pulled) the Docker image containing the acoustic binary. This image sets the environment variable ACOUSTIC_RUNNING_IN_DOCKER=1, which is not available when running outside the container.
When running speechmatching and an audio file needs to be processed, the package detects that it is not in Docker, then attempts to start a container from the aukesch/speechmatching Docker image (and attempt to pull it if it is not available locally). It will communicate with this container to process the audio file. When the program stops, the package tries to stop and remove the created container.
In other words, all that is needed is for Docker to be installed and set up correctly, and then the package can be run normally. These steps were explained in the previous section.
Docker
Alternatively, the speechmatching package entirely inside Docker. In this scenario, the package detects that the ACOUSTIC_RUNNING_IN_DOCKER=1 environment variable is present and attempts to interact with the acoustic binary locally within the same container.
To do this, include a Dockerfile in the directory of the code with a structure like:
FROM aukesch/speechmatching
COPY . .
# more code...
CMD ["python3", "main.py"]
After which the Docker image can be built and run.
Example
Most of the functions that are needed to run this on a basic level are in the speechmatching.recording submodule.
In the following example, we assume the user has several audio files:
./audio/speech1.mp3
./audio/speech2.3gp
./audio/speech3.wav
./audio/house1.mp3
./audio/house2.mp4
./audio/tree1.3gp
./audio/tree2.mp4
./audio/tree3.mp3
./audio/tree4.wav
./audio/unknown.mp3
The last file in this list is named unknown.mp3, and it is believed this audio file belongs to one of the three spoken words speech, house, or tree, for which we have several samples available.
A single recording can be loaded and analysed:
>>> from speechmatching.recording import Recording
>>> # load the recording
>>> speech1 = Recording('./audio/speech1.mp3')
>>> # get the transcript
>>> transcript = speech1.transcript
>>> # print the most likely text
>>> print(transcript.text)
>>> # print the top likely texts
>>> print(transcript.probable_texts())
>>> # and calculate similarity scores to other recordings
>>> speech2 = Recording('./audio/speech2.3gp')
>>> print(speech2.similarity(speech1))
>>> # same as
>>> print(speech2.transcript.similarity(speech1.transcript))
These samples are sorted into groups:
>>> from speechmatching.recording import Group, Recording
>>> groups = [
... Group(
... identifier='speech',
... recordings=[
... Recording('./audio/speech1.mp3'),
... Recording('./audio/speech2.3gp'),
... Recording('./audio/speech3.wav')
... ]
... ),
... Group(
... identifier='house',
... recordings=[
... Recording('./audio/house1.mp3'),
... Recording('./audio/house2.mp4')
... ]
... ),
... Group(
... identifier='tree',
... recordings=[
... Recording('./audio/tree1.3gp'),
... Recording('./audio/tree2.mp4'),
... Recording('./audio/tree3.mp3'),
... Recording('./audio/tree4.wav')
... ]
... )
... ]
>>> # load the unknown recording and match it
>>> unknown = Recording('./audio/unknown.mp3')
>>> match = unknown.match(groups)
>>> print(match.identifier)
>>> # best matching group is printed here
There are more possibilities, with much greater control over the process by using various arguments available to the functions around normalization of strings, calculating similarity scores, and strategies for finding the best matching group.
Using the function .match(...) of the Recording instance uses the combination of normalization and matching algorithms that was found to work best in the thesis for which this software was written.
More examples can be found in the examples/ directory.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file speechmatching-1.0.0.tar.gz.
File metadata
- Download URL: speechmatching-1.0.0.tar.gz
- Upload date:
- Size: 30.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2ad912c123ea5c34fb5797a8fd5212971f59a4ea4df4139b1804baf475adeeea
|
|
| MD5 |
9f20961fe551ceca9f570ca3f1ea7bd1
|
|
| BLAKE2b-256 |
c8704f1d6e094759374ddd83355d0b058e374b733c3ec4ce4670858f019c6a11
|
File details
Details for the file speechmatching-1.0.0-py3-none-any.whl.
File metadata
- Download URL: speechmatching-1.0.0-py3-none-any.whl
- Upload date:
- Size: 30.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b4c0108b9b16d6d8e14d044328b1ab66bc781458c13c9657ce1f7495eb8c2be5
|
|
| MD5 |
376f52d10ea8d186cc73bfd53d0a0988
|
|
| BLAKE2b-256 |
90beb84cd5b974f77d35dc21692053050915d5de35d5f6eb414866087a50a7f6
|