Metadata-Version: 2.1
Name: media-tools
Version: 0.2.5
Summary: Creating and managing playlists, and managing the filenames and directory structure for large numbers of music files.
Home-page: https://gitlab.com/lilacashes/music-library-tools
License: MIT
Keywords: music,audio
Author: Lene Preuss
Author-email: lene.preuss@gmail.com
Requires-Python: >=3.7,<4.0
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Requires-Dist: audioread (>=2.1.9)
Requires-Dist: coloredlogs (>=15.0,<16.0)
Requires-Dist: lxml (>=4.6.2)
Requires-Dist: mutagen (>=1.45.1)
Requires-Dist: pydub (>=0.24.1)
Requires-Dist: pylast (>=4.0.0)
Requires-Dist: python-magic (>=0.4.18)
Requires-Dist: requests (>=2.25.0)
Requires-Dist: tinytag (>=1.5.0)
Project-URL: Repository, https://gitlab.com/lilacashes/music-library-tools.git
Description-Content-Type: text/markdown

-   [Installation](#installation)
-   [Usage](#usage)
    -   [`mixcloud_upload`](#mixcloud_upload)
        -   [Authorization](#authorization)
        -   [Configuration options](#configuration-options)
    -   [`clean_filenames`](#clean_filenames)
        -   [Checking results](#checking-results)
    -   [Audacious playlist tools](#audacious-playlist-tools)
        -   [`copy_from_playlist`](#copy_from_playlist)
    -   [`buy_most_played`](#buy_most_played)
    -   [`backup_lastfm_data`](#backup_lastfm_data)
    -   [`print_length`](#print_length)
-   [Development](#development)
    -   [Test suite](#test-suite)
    -   [Generating a Table of Contents to
        `README.md`](#generating-a-table-of-contents-to-readme.md)

Toolbox for helping creating and managing playlists, and managing the filenames and directory
structure for large numbers of music files.

This is mostly tailored to my own usage patterns, although it may be useful to others.

Home page: https://gitlab.com/lilacashes/music-library-tools

PyPI page: https://pypi.org/project/media-tools

Installation
============

``` {.bash}
$ pip install media-tools
```

Usage
=====

`mixcloud_upload`
-----------------

Generate a mix on [Mixcloud](mixcloud.com) from the contents of a local directory.

Assumes that the given directory contains a number of audio files as well as a picture file (JPEG or
PNG format), one file called `description.txt` which contains the description which is printed along
with the mix on Mixcloud, and a file called `tags.txt` which contains a newline-separated list of
tags for the mix. The mix is created as an audio file called `mix.mp3` in the given directory and
then uploaded to Mixcloud. The title of the mix on Mixcloud will be the name of the directory given,
minus any leading numbers.

Example: You have a directory structured like this:

    10 - My awesome mix
    |-- 01 - Awesome track.mp3
    |-- 02 - Also awesome track.ogg
    |-- 03 - Another pretty cool track.m4a
    |-- 04 - This track rocks.flac
    |-- cover.png
    |-- description.txt
    |-- tags.txt

``` {.shell}
$ mixcloud_upload -d "10 - My awesome mix"
```

will upload this mix to Mixcloud under the name "My awesome mix" with track 01 to 04 in that order.

### Authorization

You will need to generate an application on Mixcloud for your account and get an access token, as
described in the [Mixcloud API Documentation](https://www.mixcloud.com/developers/#authorization).
Store this access token as `.mixcloud_access_token` in either the current directory, your user's
`$HOME` directory or under `$HOME/.config/media-tools`.

### Configuration options

-   `-d DIRECTORY`, `--directory DIRECTORY`: Specify the folder containing the mix, as seen above
-   `-e EXTENSIONS [EXTENSIONS ...]`, `--extensions EXTENSIONS [EXTENSIONS ...]`: List of file
    extensions to consider as audio files for the mix. Per default, files ending in `.mp3`, `.MP3`,
    `.flac`, `.m4a` and `.ogg` are treated as audio and added to the mix. Change this list of some
    of your audio files have extensions that are not part of this list, or if you want to exclude
    some of these extensions.
-   `-q`, `--quiet`: Only errors are printed. Usually you want to omit `-q`, since the entire
    generating and uploading process takes several minutes, so you want to be sure what is going on.
-   `-s`, `--strict`: Do not upload the mix if any of these conditions are met:
    -   There is no picture, no `descriptions.txt` or no `tags.txt` in the upload folder
    -   Any of the audio tracks in the folder is missing the ID3 tag for title or artist. These tags
        are needed to generate the tracklist for the mix.
-   `-c CROSSFADE_MS`, `--crossfade-ms CROSSFADE_MS`: Number of milliseconds to use for crossfading
    between two tracks. Currently, there can only be one global value which is used between all
    tracks.
-   `-r MAX_RETRY`, `--max-retry MAX_RETRY`: Maximum number of retries for failing uploads. Uploads
    can fail for a number of reasons, such as rate limiting on the Mixcloud side, or because of a
    bad internet connection. The default number of retries is set to 100, set this to a lower number
    if you want to give up earlier.
-   `-a`, `--auth-token-file`: explicitly specify the file containing the Mixcloud authorization
    token.
-   `-t`, `--auth-token-string`: explicitly specify the Mixcloud authorization token as a string.

`clean_filenames`
-----------------

In general: run a script with the `-v` option first to see what it would change. If satisfied, then
re-run it with the `-f` option to effect those changes.

Removing useless duplicate strings from filenames:

``` {.bash}
$ clean_filenames -v clean-filenames --recurse .
$ clean_filenames -f clean-filenames --recurse .
```

Changing filenames from different numbering schemes to the scheme `01 - filename.ext`:

``` {.bash}
$ clean_filenames -v clean-numbering .
$ clean_filenames -f clean-numbering .
```

Removing stray junk, such as underscores, stray dashes, and stray `[]` and `()` from filenames:

``` {.bash}
$ clean_filenames -v clean-junk .
$ clean_filenames -f clean-junk .
```

Undoing renamings done with this script, limited to a specified directory and its subfolders, or a
single file name:

``` {.bash}
$ clean_filenames -v undo .
$ clean_filenames -f undo .
# OR
$ clean_filenames -f undo ./subdir/file_name.mp3
```

Fixing symlinks to files which have been renamed by any of the previous commands:

``` {.bash}
$ clean_filenames -v fix-symlinks .
$ clean_filenames -f fix-symlinks .
```

### Checking results

Finding mp3 files which do not conform to the numbering scheme in general:

``` {.bash}
$ find . -name \*.mp3 | grep -vE '[[:digit:]]+ - .+\.mp3'
```

Finding mp3 files which have a number in their filename but do not conform to the numbering scheme,
excluding some more common use cases:

``` {.bash}
$ find . -name \*.mp3 | \
    grep -E '[[:digit:]]+[^/]+\.mp3' | \
    grep -vE '[[:digit:]]+ - .+\.mp3' | \
    grep -vE '[[:digit:]]{2}\.mp3'
```

Audacious playlist tools
------------------------

Tools for making and repairing playlists containing physical music files from audacious playlists
and the latest music files. The argument to `--playlist` defaults to the playlist currently playing
in audacious.

### `copy_from_playlist`

Copying files from the current or specified playlist (as its name in the `playlists` subdir in the
audacious configuration folder) to a specified target folder, optionally limiting the number of
files to copy to the first NUM, and optionally renaming the files to reflect the position of the
song in the playlist:

``` {.bash}
$ copy_from_playlist [-v] copy \
    [--playlist PLAYLIST_ID] \
    [--number NUM] \
    [--renumber] \
    TARGET_DIR
```

Try to find files in the current playlist which are unavailable because they have been moved, and
move them back to the place in the filesystem which is noted on the playlist (does not appear to
work currently):

``` {.bash}
$ copy_from_playlist [-v] restore \
    [--playlist PLAYLIST_ID]
```

Copy the newest files to a specified target:

``` {.bash}
$ copy_from_playlist [-v] copy-newest \
    --max-age NUM_DAYS \
    --source SOURCE_DIR \
    --target TARGET_DIR
```

`buy_most_played`
-----------------

Attempt to buy your most played tracks (in a given period) from several platforms.

...

`backup_lastfm_data`
--------------------

Back up Last.FM scrobbles.

...

`print_length`
--------------

Given a directory of audio files, print the total playing length of that directory and all
directories below.

...

Development
===========

After cloning, it is recommended to set up the git hook that runs the test suite before every
`git push`:

``` {.shell}
$ cd .git/hooks
$ ln -s ../../.git_hooks/pre-push .
```

Test suite
----------

Run all in one:

``` {.shell}
$ .git_hooks/pre-push
```

Run separately:

``` {.shell}
$ poetry run mypy .
$ poetry run flake8 .
$ poetry run pylint media_tools
$ poetry run nosetests --with-coverage --cover-package=media_tools.util tests/unit
$ poetry run nosetests tests/integration
```

Generating a Table of Contents to `README.md`
---------------------------------------------

``` {.shell}
$ git add README.md
$ git commit -m 'Before ToC generation'
$ pandoc -s --columns 100 --toc --toc-depth=4 README.md -o README_toc.md
$ mv README_toc.md README.md
$ git add README.md
$ git commit -m 'After ToC generation'
```

