Skip to content

A repository for my MSc thesis in Data Science & Machine Learning @ NTUA. A deep learning approach to audio fingerprinting for recognizing songs on real time through the microphone.

License

Notifications You must be signed in to change notification settings

ChrisNick92/deep-audio-fingerprinting

Repository files navigation

deep-audio-fingerprinting

A repository for my MSc thesis in Data Science & Machine Learning @ NTUA. A deep learning approach to audio fingerprinting for recognizing songs on real time through the microphone.

Thesis pdf

The pdf file of the thesis can be found via the following link: Audio Signal Processing and Audio Fingerprinting: Implementing a Music Recognition System

Live Demonstration

Below you can see a live demonstration of the music recognition system in a query length corresponding to 5 sec. We use "sequence search" to predict the time offset of the query also. The database consists of ~26K songs and ~2M fingerprints in this case.

query_5_sec.mp4

A Note

In the upcoming paragraphs I briefly explain how the code of this repository works by providing some directions to train and validate your own model. The code its not updated frequently so you may experience some unxpected errors. In the future, I am planning to refactor the repository and create a more detailed doc with reproducibility instructions. Until then, I am always accessible to provide any help in the issues sections. Thanks for using, enjoy! 😃😄

Overview

This repository is a PyTorch implementation of the paper Neural Audio Fingerprint for High-specific Audio Retrieval based on Contrastive Learning . In short, the goal is to train a deep neural network to extract relevant information from short audio fragments. This approach, offers an alternative way of handling the song identification problem as opposed to all Shazam-like algorithms, where they heavily relied on sophisticated feature extraction techniques.

In this repo we apply a series of data augmentations and train the neural network with contrastive loss to make it as robust as possible against to signal distortions that may arise in a realistic scenario. Below we describe:

  1. How to train a deep neural network.

  2. How to create and index the database.

  3. How to evaluate and test the music recognition system through your microphone.

Before you do anything clone this repo by typing

git clone https://github.com/ChrisNick92/deep-audio-fingerprinting.git

and then install the requirements by

pip install -r requirements.txt

Table of Contents

  1. Training the deep neural network
  2. Creating the Database
  3. Evaluate on Microphone
  4. Performance

1. Training the deep neural network

To start training the deep neural network you'll need to have a collections of songs splitted into train/val sets. All files should be in wav format with 8KHz sampling rate with one channel. If you have a collection of songs that you want to convert you can type

python utils/convert_songs.py -src <path_to_source_dir> -dst <path_to_destination_dir>

to convert a collection of mp3 files into .wav files in 8KHz with 1 channel. To handle other extensions you there is an option -ex indicating the extension of the input files. For example, if you enter -ex wav then the input files are expecting to be in a wav format. To convert the files you'll need to have ffmpeg installed on your machine.

A well-known and publicly available dataset utilized for such MIR tasks is the FMA dataset. Then, you'll need to have a collection of background noises and splitted into background_train, background_test sets. These noises must capture all the noise that the model will face in a realistic scenario. Examples can be found on youtube from background noises from cars, vehicles, people talking, etc. To simulate the reverbaration effects, you'll need to download a collection of impulse responses again splitted into impulse_train, impulse_val sets. Then, place all these folders inside deep-audio-fingerprinting/data and create a configuration file (training_config.json) for training like the one below

{
    "epochs": 120,
    "patience": 15,
    "batch size": 512,
    "model name": "fingerprinter",
    "optimizer": "Lamb",
    "output_path": "data/pretrained_models/",
    "background_noise_train": "data/background_train",
    "background_noise_val": "data/background_val",
    "impulse_responses_train": "data/impulse_responses",
    "impulse_responses_val": "data/impulse_responses_val",
    "train_set": ["data/train"],
    "val_set": ["data/val"],
    "loss":{
        "loss": "NTXent_Loss"
    }
}

Then, run

python training/trainer.py --config training/training_config.json

This will output fingerprinter.pt in data/pretrained_models/, containing the model's weights. Of course, you can experiment with the previous hyperparameters.

Note that the quality of the model is highly dependent on the quality of the background noises & impulse responses.

2. Creating the database

The next step is to extract the fingerprints using the pretrained model. The features are extracted for each 1 sec audio duration with a hop length corresponding to 0.5 sec for each song to placed on the database. To extract the fingerprints create an empty folder in deep-audio-fingerprinting/data to place the fingerprints (e.g. fingerprints). Create a configuration file (generation_config.json) like on the one below

{
    "SR": 8000,
    "HOP SIZE": 4000,
    "input dirs": ["data/train", "data/val"],
    "batch size": 30,
    "weights": "data/pretrained_models/fingerprinter.pt",
    "attention": false,
    "output dir": "data/fingerprints"
}

Then generate the fingerprints by typing

python generation/generate_fingerprints.py --config generation/generation_config.json

This will output the fingerprints from the songs in the folders train,val in data/fingerprints. The fingerprints are in the form of numpy arrays, corresponding to 128-dimensional vectors. To reduce the size of the fingerprints and to achieve quick retrievals we use product quantization with a non exhaustive search based on inverted lists. We use the faiss library to implement these techniques. The database in this case, is represented by a faiss index (.index file), and by a .json file. The index contains the quntized representations and the index contains the song names corresponding to each index on the database. To generate the index + json files, first create a configuration file like the one below

{
    "input_dir": "data/fingerprints",
    "output_dir": "data/faiss_indexes/",
    "name": "ivf20_pq64",
    "index": "IVF20,PQ64",
    "d": 128
}

The above configuration file creates a an index using 20 centroids with a product quantization with 64 subquantizers. You can experiments with the faiss indexes, the choice is dependent on the number of fingerprints to be placed on the database. In this example, the database I created consists of about ~ 2M fingerprints. You can look at the wiki of faiss for more information about which index is appropriate depending on the database size. To create the index + json file type

python generation/create_faiss_index.py --config generation/faiss_config.json

This will output two files ivf20_pq64.index, ivf20_pq64.json, in data/faiss_indexes/.

3. Evaluate on Microphone

To test the overall performance of the system with your microphone, create a configuration file (online_config.json) of the form

{
    "SR": 8000,
    "Hop size": 4000,
    "FMB": 4000,
    "duration": 10,
    "nprobes": 5,
    "search algorithm": "sequence search",
    "weights": "data/pretrained_models/fingerprinter.pt",
    "index": "data/faiss_indexes/fingerprinter/ivf20_pq64.index",
    "json": "data/faiss_indexes/fingerprinter/ivf20_pq64.json",
    "device": "cpu",
    "neighbors": 4,
    "attention": false
}

The parameter "duration" corresponds to the length of the query duration in seconds. The higher the query duration, the better the results with a cost of inference time. Using the search algorithm "sequence search" we can predict the time offset of the query relative the true recording. The other option is "majority vote" which does not predict the time offset but performs slightly better. To test the performance, start playing music and type

python evaluation/online_inference.py --config evaluation/online_config.json

4. Performance

Below you can a see the results of this approach on a database comprising about 26K songs. 25K come from the FMA dataset corresponding to 30 sec audio clips, while the rest are downloaded from you tube and are full length songs. The database consists about ~2M fingerprints. Below you see the results on a recorded audio corresponding to 15 songs played sequentially one after the other. You can see a model without applying low/high pass filters and a model that is trained on such degradations. The comparison is the with the dejavu open source library.

Performance (Accuracy) of the Music Recognition Systems on Microphone Experiment.
Metrics Query length (s) low mid high
Deep Audio 85.03% 62.65% 41.92%
Deep Audio (No Filters) 2 67.22% 49.02% 19.47%
Dejavu 12.63% 6.50% 10.73%
Deep Audio 92.66% 80.06% 62.52%
Deep Audio (No Filters) 5 80.38% 65.39% 31.10%
Dejavu 59.21% 40.94% 47.24%
Deep Audio 94.52% 90.32% 74.19%
Deep Audio (No Filters) 10 84.19% 78.39% 41.29%
Dejavu 83.60% 70.98% 71.29%
Deep Audio 97.56% 90.24% 80.98%
Deep Audio (No Filters) 15 86.34% 80.98% 48.29%
Dejavu 93.33% 77.14% 83.33%

About

A repository for my MSc thesis in Data Science & Machine Learning @ NTUA. A deep learning approach to audio fingerprinting for recognizing songs on real time through the microphone.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published