Skip to content

A Swiss Army knife for programmatic music library management. Manages both local and music streaming service libraries.

License

Notifications You must be signed in to change notification settings

geo-martino/musify

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Musify

PyPI Version Python Version Documentation
PyPI Downloads Code Size Contributors License
GitHub - Validate GitHub - Deployment GitHub - Documentation

A Swiss Army knife for music library management

Supporting local and music streaming service (remote) libraries.

  • Extract data for all item types from remote libraries, including following/saved items, such as: playlists, tracks, albums, artists, users, podcasts, audiobooks
  • Load local audio files, programmatically manipulate, and save tags/metadata/embedded images
  • Synchronise local tracks metadata with its matching track's metadata on supported music streaming services
  • Synchronise local playlists with playlists on supported music streaming services
  • Backup and restore track tags/metadata and playlists for local and remote libraries
  • Extract and save images from remote tracks or embedded in local tracks

Contents

Note

This readme provides a brief overview of the program. Read the docs for full reference documentation.

Installation

Install through pip using one of the following commands:

pip install musify
python -m pip install musify

There are optional dependencies that you may install for optional functionality. For the current list of optional dependency groups, read the docs

Quick Guides

These quick guides will help you get set up and going with Musify in just a few minutes. For more detailed guides, check out the documentation.

Tip

Set up logging to ensure you can see all info reported by the later operations. Libraries log info about loaded objects to the custom STAT level.

import logging
import sys
from musify.logger import STAT

logging.basicConfig(format="%(message)s", level=STAT, stream=sys.stdout)

Spotify

In this example, you will:

  • Authorise access to the Spotify Web API
  • Load your Spotify library
  • Load some other Spotify objects
  • Add some tracks to a playlist
  1. If you don't already have one, create a Spotify for Developers account.

  2. If you don't already have one, create an app. Select "Web API" when asked which APIs you are planning on using. To use this program, you will only need to take note of the client ID and client secret.

  3. Create a SpotifyAPI object and authorise the program access to Spotify data as follows:

    The scopes listed in this example will allow access to read your library data and write to your playlists. See Spotify Web API documentation for more information about scopes

     from musify.libraries.remote.spotify.api import SpotifyAPI
    
     spotify_api = SpotifyAPI(
         client_id="<YOUR CLIENT ID>",
         client_secret="<YOUR CLIENT SECRET>",
         scope=[
             "user-library-read",
             "user-follow-read",
             "playlist-read-collaborative",
             "playlist-read-private",
             "playlist-modify-public",
             "playlist-modify-private"
         ],
         # providing a `token_file_path` will save the generated token to your system 
         # for quicker authorisations in future
         token_file_path="<PATH TO JSON TOKEN>"  
     )
  4. Define helper functions for loading your SpotifyLibrary data:

     from musify.libraries.remote.spotify.library import SpotifyLibrary
    
    
     async def load_library(library: SpotifyLibrary) -> None:
         """Load the objects for a given ``library``. Does not enrich the loaded data."""
         # authorise the program to access your Spotify data in your web browser
         async with library:
             # if you have a very large library, this will take some time...
             await library.load()
    
    
     async def load_library_by_parts(library: SpotifyLibrary) -> None:
         """Load the objects for a given ``library`` by each of its distinct parts.  Does not enrich the loaded data."""
         # authorise the program to access your Spotify data in your web browser
         async with library:
             # load distinct sections of your library
             await library.load_playlists()
             await library.load_tracks()
             await library.load_saved_albums()
             await library.load_saved_artists()
    
    
     async def enrich_library(library: SpotifyLibrary) -> None:
         """Enrich the loaded objects in the given ``library``"""
         # authorise the program to access your Spotify data in your web browser
         async with library:
             # enrich the loaded objects; see each function's docstring for more info on arguments
             # each of these will take some time depending on the size of your library
             await library.enrich_tracks(features=True, analysis=False, albums=False, artists=False)
             await library.enrich_saved_albums()
             await library.enrich_saved_artists(tracks=True, types=("album", "single"))
    
    
     def log_library(library: SpotifyLibrary) -> None:
         """Log stats about the loaded ``library``"""
         library.log_playlists()
         library.log_tracks()
         library.log_albums()
         library.log_artists()
    
         # pretty print an overview of your library
         print(library)
  5. Define helper functions for loading some Spotify objects using any of the supported identifiers:

     from musify.libraries.remote.spotify.object import SpotifyTrack, SpotifyAlbum, SpotifyPlaylist, SpotifyArtist
    
    
     async def load_playlist(api: SpotifyAPI) -> SpotifyPlaylist:
         # authorise the program to access your Spotify data in your web browser
         async with api as a:
             playlist = await SpotifyPlaylist.load("spotify:playlist:37i9dQZF1E4zg1xOOORiP1", api=a, extend_tracks=True)
         return playlist
    
    
     async def load_tracks(api: SpotifyAPI) -> list[SpotifyTrack]:
         tracks = []
    
         # authorise the program to access your Spotify data in your web browser
         async with api as a:
             # load by ID
             tracks.append(await SpotifyTrack.load("6fWoFduMpBem73DMLCOh1Z", api=a))
             # load by URI
             tracks.append(await SpotifyTrack.load("spotify:track:4npv0xZO9fVLBmDS2XP9Bw", api=a))
             # load by open/external style URL
             tracks.append(await SpotifyTrack.load("https://open.spotify.com/track/1TjVbzJUAuOvas1bL00TiH", api=a))
             # load by API style URI
             tracks.append(await SpotifyTrack.load("https://api.spotify.com/v1/tracks/6pmSweeisgfxxsiLINILdJ", api=api))
    
         return tracks
    
    
     async def load_album(api: SpotifyAPI) -> SpotifyAlbum:
         # authorise the program to access your Spotify data in your web browser
         async with api as a:
             album = await SpotifyAlbum.load(
                 "https://open.spotify.com/album/0rAWaAAMfzHzCbYESj4mfx", api=a, extend_tracks=True
             )
         return album
    
    
     async def load_artist(api: SpotifyAPI) -> SpotifyArtist:
         # authorise the program to access your Spotify data in your web browser
         async with api as a:
             artist = await SpotifyArtist.load("1odSzdzUpm3ZEEb74GdyiS", api=a, extend_tracks=True)
         return artist
    
    
     async def load_objects(api: SpotifyAPI) -> None:
         playlist = await load_playlist(api)
         tracks = await load_tracks(api)
         album = await load_album(api)
         artist = await load_artist(api)
    
         # pretty print information about the loaded objects
         print(playlist, *tracks, album, artist, sep="\n")
  6. Define helper function for adding some tracks to a playlist in your library, synchronising with Spotify, and logging the results:

    NOTE: This step will only work if you chose to load either your playlists or your entire library in step 4.

     async def update_playlist(name: str, library: SpotifyLibrary) -> None:
         """Update a playlist with the given ``name`` in the given ``library``"""
         tracks = await load_tracks(library.api)
         album = await load_album(library.api)
         await load_library(library)
    
         my_playlist = library.playlists[name]
    
         # add a track to the playlist
         my_playlist.append(tracks[0])
    
         # add an album to the playlist using either of the following
         my_playlist.extend(album)
         my_playlist += album
    
         # sync the object with Spotify and log the results
         async with library:
             result = await my_playlist.sync(dry_run=False)
         library.log_sync(result)
     
     asyncio.run(update_playlist(spotify_api))
  7. Run the program:

     import asyncio
    
     asyncio.run(load_objects(api))
     asyncio.run(update_playlist("<YOUR PLAYLIST'S NAME>", api))  # case sensitive

Local

In this example, you will:

  • Load a local library
  • Modify the tags of some local tracks and save them
  • Modify a local playlist and save it
  1. Create one of the following supported LocalLibrary objects:

    Generic local library

    from musify.libraries.local.library import LocalLibrary
    
    library = LocalLibrary(
        library_folders=["<PATH TO YOUR LIBRARY FOLDER>", ...],
        playlist_folder="<PATH TO YOUR PLAYLIST FOLDER",
    )

    MusicBee

    You will need to install the musicbee optional dependency to work with MusicBee objects. Read the docs for more info.

    from musify.libraries.local.library import MusicBee
    
    library = MusicBee(musicbee_folder="<PATH TO YOUR MUSICBEE FOLDER>")
  2. Load your library:

    # if you have a very large library, this will take some time...
    library.load()
    
    # ...or you may also just load distinct sections of your library
    library.load_tracks()
    library.load_playlists()
    
    # optionally log stats about these sections
    library.log_tracks()
    library.log_playlists()
    
    # pretty print an overview of your library
    print(library)
  3. Get collections from your library:

    playlist = library.playlists["<NAME OF YOUR PLAYLIST>"]  # case sensitive
    album = next(album for album in library.albums if album.name == "<ALBUM NAME>")
    artist = next(artist for artist in library.artists if artist.name == "<ARTIST NAME>")
    folder = next(folder for folder in library.folders if folder.name == "<FOLDER NAME>")
    genre = next(genre for genre in library.genres if genre.name == "<GENRE NAME>")
    
    # pretty print information about the loaded objects
    print(playlist, album, artist, folder, genre, sep="\n")
  4. Get a track from your library using any of the following identifiers:

    # get a track via its title
    # if multiple tracks have the same title, the first matching one if returned
    track = library["<TRACK TITLE>"]
    
    # get a track via its path
    track = library["<PATH TO YOUR TRACK>"]  # must be an absolute path
    
    # get a track according to a specific tag
    track = next(track for track in library if track.artist == "<ARTIST NAME>")
    track = next(track for track in library if "<GENRE>" in (track.genres or []))
    
    # pretty print information about this track
    print(track)
  5. Change some tags:

    from datetime import date
    
    track.title = "new title"
    track.artist = "new artist"
    track.album = "new album"
    track.track_number = 200
    track.genres = ["super cool genre", "awesome genre"]
    track.key = "C#"
    track.bpm = 120.5
    track.date = date(year=2024, month=1, day=1)
    track.compilation = True
    track.image_links.update({
         "cover front": "https://i.scdn.co/image/ab67616d0000b2737f0918f1560fc4b40b967dd4",
         "cover back": "<PATH TO AN IMAGE ON YOUR LOCAL DRIVE>"
    })
    
    # see the updated information
    print(track)
  6. Save the tags to the file:

    from musify.libraries.local.track.field import LocalTrackField
    
    # you don't have to save all the tags you just modified
    # select which you wish to save first like so
    tags = [
         LocalTrackField.TITLE,
         LocalTrackField.GENRES,
         LocalTrackField.KEY,
         LocalTrackField.BPM,
         LocalTrackField.DATE,
         LocalTrackField.COMPILATION,
         LocalTrackField.IMAGES
    ]
    
    track.save(tags=tags, replace=True, dry_run=False)
  7. Add some tracks to one of your playlists and save it:

    my_playlist = library.playlists["<NAME OF YOUR PLAYLIST>"]  # case sensitive
    
    # add a track to the playlist
    my_playlist.append(track)
    
    # add album's and artist's tracks to the playlist using either of the following
    my_playlist.extend(album)
    my_playlist += artist
    
    result = my_playlist.save(dry_run=False)
    print(result)

Currently Supported

  • Music Streaming Services: Spotify
  • Audio filetypes: .flac .m4a .mp3 .wma
  • Local playlist filetypes: .m3u .xautopf
  • Local Libraries: MusicBee

Motivation and Aims

The key aim of this package is to provide a seamless framework for interoperability between all types of music libraries whether local or remote.

This framework should allow for the following key functionality between libraries:

  • Synchronise saved user data including:
    • playlists data (e.g. name, description, tracks)
    • saved tracks/albums/artists etc.
  • Synchronise metadata by allowing users to pull metadata from music streaming services and save this to local tracks
  • Provide tools to allow users to move from music streaming services to a local library by semi-automating the process of purchasing songs.

With this functionality, user's should then have the freedom to:

  • Switch between music streaming services with a few simple commands
  • Share local playlists and other local library data with friends over music streaming services without ever having to use them personally
  • Easily maintain a high-quality local library with complete metadata

Users should have the freedom to choose how and where they want to listen to their favourite artists.

Given the near non-existence of income these services provide to artists, user's should have the choice to compensate their favourite artists fairly for their work, choosing to switch to other services that do and/or choosing not to use music streaming services altogether because of this. Hopefully, by reintroducing this choice to users, the music industry will be forced to re-evaluate their complete devaluing of creative work in the rush to chase profits, and instead return to a culture of nurturing talent by providing artists with a basic income to survive on the work of their craft. One can dream.

Release History

For change and release history, check out the documentation.

Contributing and Reporting Issues

If you have any suggestions, wish to contribute, or have any issues to report, please do let me know via the issues tab or make a new pull request with your new feature for review.

For more info on how to contribute to Musify, check out the documentation.

Author notes

I initially developed this program for my own use so that I can share my local playlists with friends online. I have always maintained my own local library well and never saw the need to switch to music streaming services after their release. However, as an artist who has released music on all streaming services and after listening to the concerns many of the artists I follow have regarding these services, I started to refocus this project to be one that aims to break down the barriers between listening experiences for users. The ultimate aim being to make managing a local library as easy as any of the major music streaming services, allowing users the same conveniences while compensating artists fairly for their work.

I hope you enjoy using Musify!