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.
The memory management logic for vectors in c++ is not the same as for python lists, so currently holding a reference to an element (e.g. a Note) inside a Score for an extended period of time poses a potential dangling pointer risk. Find more information here.
I'm trying to implement a container that is different from a c++
std::vector
to solve this problem, which would be similar to the stable_vector
in boost
.
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
git clone --recursive https://github.com/Yikai-Liao/symusic
pip install ./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++.