Skip to main content

Output the differences between two MIDI files, using the command-line.

Project description

MIDIDiff

PyPI version Python Versions License: MIT Tests Documentation Status

MIDIDiff compares two MIDI files and produces a third MIDI file containing the notes that are present in only one of the inputs. Notes are matched by pitch, start tick, and duration; velocity differences are ignored.

Documentation

Full documentation is available at Read the Docs (once published).

To build the documentation locally:

cd docs
poetry run sphinx-build -b html . _build/html

Requirements

  • Python 3.11+
  • mido (installed automatically via the project dependencies)
  • rich (optional, for enhanced CLI output)

Installation

For Development (Editable Install)

If you're contributing to MIDIDiff or testing changes, install in editable mode:

Recommended:

poetry install --extras cli

Poetry installs the package in editable mode by default, so changes to the source code are immediately reflected.

Alternative (using pip):

pip install -e ".[cli]"

The -e flag creates an editable install, allowing you to modify the code without reinstalling.

For more details, see the Testing Guide.

Build and install locally

poetry build
pip install dist/*.whl

For CLI functionality with rich formatting, install with CLI extras:

pip install dist/midi_diff-*.whl[cli]

Or install directly from the package:

pip install "midi-diff[cli]"

Core library only (no CLI dependencies)

If you only need the core library without CLI dependencies (e.g., for programmatic use), install without extras:

pip install midi-diff

Usage

Command-line interface

The CLI expects two input MIDI files and an output path for the diff:

midi-diff fileA.mid fileB.mid diff.mid

You can also run the module directly:

python -m midi_diff.cli diff fileA.mid fileB.mid diff.mid

Output behavior

  • If the output file already exists, MIDIDiff will append an incrementing suffix (for example, diff_1.mid) to avoid overwriting.
  • The resulting MIDI file contains only notes that are present in one input but not the other.

Version and environment info

Use -V or --version to print the installed MIDIDiff version, environment details, and the result of an update check against PyPI:

midi-diff --version

Debug information

Use the debug-info subcommand to display comprehensive diagnostic information:

midi-diff debug-info

Shell completions

Generate shell completion scripts for enhanced command-line experience:

midi-diff completion bash        # Bash
midi-diff completion zsh         # Zsh
midi-diff completion fish        # Fish
midi-diff completion powershell  # PowerShell
midi-diff completion cmd         # Command Prompt (prints doskey helper)

Installation: For detailed instructions on installing and configuring shell completions for your specific shell, see the Shell Completions Guide in the documentation.

Programmatic usage

You can also use MIDIDiff as a library in your Python code:

from midi_diff.core import main

# Compare two MIDI files and save the diff
main('fileA.mid', 'fileB.mid', 'diff.mid')

Or work with the lower-level API:

import mido
from midi_diff.midi_utils import extract_notes, notes_to_midi

# Load MIDI files
mid_a = mido.MidiFile('fileA.mid')
mid_b = mido.MidiFile('fileB.mid')

# Extract notes
notes_a = set(extract_notes(mid_a))
notes_b = set(extract_notes(mid_b))

# Compute diff
diff_notes = notes_a.symmetric_difference(notes_b)

# Create output MIDI
diff_mid = notes_to_midi(list(diff_notes), ticks_per_beat=mid_a.ticks_per_beat)
diff_mid.save('diff.mid')

Architecture

CLI Sub-package (v1.0.0+)

Starting with version 1.0.0, the CLI has been separated into its own sub-package (midi_diff.cli) to improve separation of concerns and allow the core library to be used without CLI dependencies.

  • Core library (midi_diff): Contains note extraction, diff logic, and data models. Has minimal dependencies (mido only).
  • CLI sub-package (midi_diff.cli): Contains argument parsing, version reporting, debug info, and environment checks. Requires rich for enhanced output.

Migration note: If you were importing from midi_diff.cli in your code, this continues to work through a backward compatibility shim that will be maintained through the 1.x release series.

Contributing

We welcome contributions! Please see CONTRIBUTING.md for guidelines on:

  • How to contribute code and documentation
  • Changelog update requirements
  • Development setup
  • Pull request process

All contributions with user-facing changes must update the CHANGELOG.md file.

How note matching works

Notes are considered identical when they share the same:

  • Pitch
  • Start tick
  • Duration

Velocity is intentionally ignored so that the diff focuses on musical placement.

API Stability (v1.0.0+)

Starting with version 1.0.0, the following API surface is considered stable and will follow semantic versioning:

  • Core functions: midi_diff.core.main()
  • Data models: midi_diff.midi_utils.NoteEvent
  • Utility functions: midi_diff.midi_utils.extract_notes(), midi_diff.midi_utils.notes_to_midi()
  • CLI entry point: midi-diff command and midi_diff.cli:cli function

Breaking changes to these APIs will result in a major version bump (2.0.0, etc.).

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

midi_diff-1.0.7.tar.gz (16.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

midi_diff-1.0.7-py3-none-any.whl (18.4 kB view details)

Uploaded Python 3

File details

Details for the file midi_diff-1.0.7.tar.gz.

File metadata

  • Download URL: midi_diff-1.0.7.tar.gz
  • Upload date:
  • Size: 16.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.0 CPython/3.13.11 Linux/6.11.0-1018-azure

File hashes

Hashes for midi_diff-1.0.7.tar.gz
Algorithm Hash digest
SHA256 cc8ee709e0340bc37f698d06700750d2f50e53640eb922c4edba6fc3f033239e
MD5 fe9b0f096761b9b744011ed0047f02c9
BLAKE2b-256 dc10074852bf9c2f8293ee5d2268949e4d4ddc5180f9b273fca66cad46d331d0

See more details on using hashes here.

File details

Details for the file midi_diff-1.0.7-py3-none-any.whl.

File metadata

  • Download URL: midi_diff-1.0.7-py3-none-any.whl
  • Upload date:
  • Size: 18.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.0 CPython/3.13.11 Linux/6.11.0-1018-azure

File hashes

Hashes for midi_diff-1.0.7-py3-none-any.whl
Algorithm Hash digest
SHA256 9072959593f6881ce2fdfae7169b6a65cac03ec7313e658349873c47ca36cef1
MD5 e6c9e490b87a944e08356b233bc6219c
BLAKE2b-256 37464f44fbc742364d8d84be5bf74c9f71727743ef2d338a82288c9e57abfda8

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page