Symusic("Sybolic Music") is a cross-platform note level
midi decoding library with lightening speed, which is hundreds of times faster (100x to 1000x depending on your file size) than mido, the main midi parsing library in python.
The library is written in cpp and based on minimidi. It offers a python binding using pybind11.
Here, we have added a tutorial.ipynb for you to learn about how to use the library.
And a document is also available. Not complete though.
- You can just read a midi file like
score = symusic.Score("path to midi", ttype="tick")
- Writing back to midi is now supported!
score.dump_midi("path")
- Multiple
time unit (ttype)
is now supported (currentlytick
andquarter
) - The tempo attribute in the tempo event represents quarter per minute (qpm)
- We offer some batch operation functions for both
Score
andTrack
class:- shift_pitch(offset: int)
- shift_velocity(offset: int)
- shift_time(offset: float)
- sort(key, reverse)
- You can operate each note just like you did before in python (like PrettyMidi)
- Extremely fast
pickle
is now supported .numpy()
method for getting SoA data.filter(func, inplace)
method for filtering objets in all the "List" in symusic- A new synthesizer is now available! It comes from our another project prestosynth. Find usages in our document. Note that we will support more features of soundfont for synthesizing in the future.
Currently, synthesizing audio with symusic on linux gives a probability of hearing strange noise at the end of the audio. This phenomenon cannot be reproduced consistently and does not occur on windows.
You can temporarily fix this by adding a few seconds of notes at the end and removing those seconds of audio at the end.
It is important to note that the audio synthesis functionality was implemented by us rather than using an off-the-shelf project, and it is still experimental at this time. Feel free to give us feedback on any bugs you find.
pip install symusic
Make sure that your system has cmake and c++ compilers
You could clone from github and install it by pip
git clone --recursive https://github.com/Yikai-Liao/symusic
pip install ./symusic
Or you could install the source distribution from pypi
pip install symusic --no-binary symusic
For debugging purpose, you could pass -Ccmake.define.MEM_LEAK_WARNING=True
to pip install
to enable the memory leak warning from nanobind
.
pip install -Ccmake.define.MEM_LEAK_WARNING=True ./symusic
- test using mahler.mid from minimidi/example on my laptop (i7-10875H, 32GB 2666MHz DDR4 RAM, Linux 6.1.69-1-lts)
- Note that mahler.mid is quite a large midi file. So this benchmark mainly reelects the
parsing time
(the percentage of time of loading file gets more significant when file is smaller) midifile
is writen in cpp, and could parse midi files to bothevent level
andnote level
. It is slow mostly because ofiostream
.mido
is writen in pure python, and only parses midi files toevent level
pretty_midi
andmiditoolkit
is based onmido
, and parse midi files tonote level
- For libraries written in python or with python bindings, we use
timeit
to measure the time cost of parsing the midi file.nanobench
for cpp libraries andBenchmarkTools
for julia libraries.
library | level | absolute time | relative time |
---|---|---|---|
minimidi | event | 2.86 ms | 1.0x |
symusic | note | 3.47 ms ± 113 µs | 1.2x |
midifile | event | 44.0 ms | 15.4x |
midifile | note | 45.6 ms | 15.9x |
MIDI.jl | note | 109.707 ms | 38.4x |
mido | event | 2.92 s ± 42.7 ms | 1021.0x |
miditoolkit | note | 3.15 s ± 38.2 ms | 1101.4x |
pretty_midi | note | 3.16 s ± 9.56 ms | 1104.9x |
music21 | note | 4.23 s ± 34.5 ms | 1479.0x |
- minimidi : A fast and lightweight midi parsing library written in cpp, which is the foundation of this project.
- prestosynth : A new fast soundfont synthesizer written in cpp, which is the foundation of the synthesizer in this project.
- nanobind : A efficient and lightweight library for binding C++ to Python, which is significantly faster than pybind11.
- zpp_bits : An extraordinary fast and lightweight single header library for serialization and deserialization. I use it to support pickle.
- geek_time_cpp The example code of the book "Modern C++ Programming Practice". We use the metamacro.h in it for shortening the code.
- utfcpp An easy to use and portable library for handling utf8 string in C++.