Skip to content

DiNoScore/OMR-MeasureRecognition

 
 

Repository files navigation

About this fork

This repository is focused on using the results of the original project. If you are interested in training and evaluation of your own data set, please look upstream. Notable changes made:

  • Harnessed all dependencies in Nix
    • Type nix-shell to get a ready to work environment with all packages and dependencies
    • All dependencies are pinned so this will keep working forever™ (until the download links rot)
    • Direnv users should type direnv allow and feel at home
  • Added a standalone inference script and inference server script
  • Added a NixOS module to easily host the server in a sandboxed environment
  • Minor fixes and cleanups

Important: the detectron2 code must not run on the main thread, or it might never return. Especially this means that you must instruct your Python server to use multiple threads if you run into it.

CLI usage

In the Nix shell (or some other suitable development environment), run:

python ./Python/inference_cli.py --models-dir=$(nix-build models.nix) example-images/* -o output.json

The result is a JSON list containing a list of staves for every input page:

[
	{
		"width": 3493,
		"height": 2002,
		"staves": [
			{
				"left": 205,
				"top": 1701,
				"right": 3357,
				"bottom": 1820,
			},
			
		],
	},
	
]

Server usage

Start the server with Flask:

export MODELS_DIR=$(nix-build models.nix)
export FLASK_APP=Python/inference_server.py
flask run --port=8000

Or in production mode:

export MODELS_DIR=$(nix-build models.nix)
export FLASK_ENV = "production"
export PYTHONPATH=./Python
gunicorn -b localhost:8000 inference_server:app --log-level=debug --timeout=300 --workers=3 --threads=3

Alternatively, you can simply import server.nix into your NixOS configuration and start the service. The systemd unit is pre-configured with the correct input data and also hardened.

Upload some images with

import requests
staves = requests.post('http://localhost:8000/upload', files=[('file', open('image.jpg', 'rb'))]).json()

You'll find a more complex example in ./Python/inference_server_test.py.

Common HTTP error codes:

  • 200: The result is what you want
  • 400: You didn't upload an image
  • 413: You uploaded too many or too large images (Max 50 images per request, 8 MiB per image, 2048×2048 pixels)
  • 500: Python exception 🤡
  • 504: Service down

A public server is hosted at https://inference.piegames.de/public/upload, it'll serve you on a "best effort" basis.

Nix overview

  • models.nix: Fetches the pre-trained models into the Nix store.
  • shell.nix: Contains all runtime + dev dependencies to give you a working development environment.
  • server_app.nix: Some ugly packaging of the inference server as a Python module. You only need this if you are starting the service in NixOS, see server.nix.
  • server.nix: A prototype NixOS module that'll start a systemd service for the server. For it to work, you need to apply the overlay into your system.

About this repo

This repo has a live app running Open in Streamlit

This repository has been created as part of my master's thesis. The thesis focuses on detecting system measures, stave measures and staves in music pages with the help of deep learning. I used two open-source datasets for this thesis. The handwritten MUSCIMA++ dataset, which has all three categories and the typeset bounding box annotations of musical measures dataset, which only contains system measures. I first enhanced the typeset dataset by using a deep learning framework called Detectron2. This has been achieved by training a model on only the stave measures from the MUSCIMA++ dataset. The trained model generalized well on the typeset data, so that I could predict stave measures from it. I took the stave measures from the predictions, created staves from them and intersected them with the existing ground truth system measures to generate more precise stave measures. After manually correcting the generated stave measure annotations, I generated stave lines from the system measures and stave measures. These have also been manually corrected where needed. With both datasets now containing all three categories, I trained several models, one for each category, one for system measures and staves, and one for all three categories combined. I also used three different backbones to compare results between them. Our results show that the models generalize well for typeset music pages, but several flaws for handwritten pages have been observed. A web-based tool has been developed and deployed to use our models which is publicly available.

With this work, I achieved remarkable results for object detection in music notes. They are state-of-the-art results and the models produced can be used by anyone by visiting the live Streamlit application or cloning, installing all dependencies and running the application locally on their own pc. To run locally, follow the installation instructions below. Running locally will download all our pre-trained models which are 8GB large.

All scripts used for this work are in the Python folder. The training, inference and evaluation scripts are probably the most important ones. The scripts demonstrate how I trained all the networks, used inference to manually check their accuracy on data and how I evaluated them based on the COCO-metric. The training process how we enhanced the typeset dataset is also available as a Jupyter Notebook in the AudioLabsEnhancement file. Other scripts are used as classes to bundle functionality together. The Streamlit application runs from the streamlit_app file.

There are two different requirement files because the requirements.txt file is used by the Streamlit application.
It is different from the local_requirements.txt file in that it does not use the integrated submodule because Streamlit does not yet support submodules.
This will be changed ASAP when the support for submodules is implemented by the Streamlit framework.

Results

Faster R-CNN with ResNet-50 backbone

Category Name Iterations mAP AP75 AP50 system measures mAP staves mAP stave measures mAP
System measures 12600 95.828 98.785 98.982 - - -
Stave measures 12900 87.639 97.582 98.933 - - -
Staves 16500 92.578 99.003 99.010 - - -
System measures and Staves 14100 88.190 95.423 95.519 93.668 82.711 -
System measures, Stave measures and Staves 3600 75.970 85.549 86.422 83.366 79.535 65.010

Faster R-CNN with ResNet-101 backbone

Category Name Iterations mAP AP75 AP50 system measures mAP staves mAP stave measures mAP
System measures 15600 95.996 98.823 98.988 - - -
Stave measures 12600 88.882 97.515 98.938 - - -
Staves 19200 93.650 100.00 100.00 - - -
System measures and Staves 5400 88.886 96.962 97.018 93.651 84.122 -
System measures, Stave measures and Staves 3000 75.041 85.297 86.713 85.676 78.454 60.992

Faster R-CNN with ResNeXt-101-32x8d backbone

Category Name Iterations mAP AP75 AP50 system measures mAP staves mAP stave measures mAP
System measures 8400 95.907 98.931 99.008 - - -
Stave measures 15300 89.625 97.785 99.001 - - -
Staves 10800 93.457 99.009 100.00 - - -
System measures and Staves 16800 88.941 95.319 95.693 93.792 84.091 -
System measures, Stave measures and Staves 1800 75.922 86.017 87.059 90.096 77.275 60.393

Cloning this repository

This repository uses Detectron2 as submodule.
In order to clone the submodule correctly, you will need to use:

git clone --recurse-submodules https://github.com/MarcKletz/OMR-MeasureRecognition

If you already cloned the project and forgot --recurse-submodules,
you can combine the git submodule init and git submodule update steps by running

git submodule update --init

Installation Setup

Requirements before starting:
Python >= 3.6
to run training and testing you need a CUDA capable device and the CUDA Toolkit 10.1
you can run the streamlit app which does inference without CUDA

For Linux:

Step 1:
You will require some build / development tools, install them by running:

sudo yum groupinstall "Development Tools"
or
sudo apt-get install build-essential

Step 2:
Install python development version.

sudo yum install python36-devel
or
sudo apt-get install python3-dev

Step 3 (OS DEPENDENT):
CentOS, Amazon Linux AMI, Red Hat Enterprise Linux:
Needs cython before running the requirements install:
pip3 install cython
This is needed for pycocotools because pip apparently builds all packages first, before attempting to install them.
(ノ☉ヮ⚆)ノ ┻━┻

Ubuntu:
There are no wheels available for opencv-python-headless on some ubuntu distributions.
Instead of building it on your own, I recommend to install it with the following command.
sudo apt install python3-opencv
Dont forget to remove the opencv-python-headless requirement from the local_requirements.txt if you did this!

Debian:
Skip to Step 4

Step 4:
install all the required python libraries from this repository:
sudo pip3 install -r local_requirements.txt [-v]
This might take a while! So be patient, you may add the -v tag to see installation progress.

Step 5:
Install the Detectron2 submodule as python library by running
sudo python setup.py install
from within the Detectron2 folder.

For Windows:

Requirements:
Windows SDK
C++14.0 build tools
Microsoft Visual C++ Redistributable
can all be installed with the Visual Studio installer.

Step 1:
install all the required python libraries from the OMR-MeasureRecognition repo.
pip install -r local_requirements.txt
(Requires admin privileges!)

Step 2:
Install the Detectron2 submodule as python library by running
python setup.py install
from within the Detectron2 folder.
(Requires admin privileges, so run cmd as admin!)

Possible step 3:
If step 3 fails with an error message about an nms_rotated_cuda.cu file, try this.
add the following line in detectron2\detectron2\layers\csrc\nms_rotated\nms_rotated_cuda.cu before #ifdef WITH_HIP:
#define WITH_HIP
Repead step 2.

Run the Streamlit app:

Make sure that the python package installation location is added to path, so that you can run streamlit. If the streamlit command fails with "command not found" you will need to add the following to your path:
export PATH="$HOME/.local/bin:$PATH"

Complete the installation instructions and then run:
streamlit run Python/streamlit_app.py
from the OMR-MeasureRecognition repository

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Python 96.0%
  • Nix 4.0%