Metadata-Version: 2.4
Name: mopidy-local
Version: 4.0.0rc1
Summary: Mopidy extension for playing music from your local music archive
Author-email: Stein Magnus Jodal <stein.magnus@jodal.no>
License: Apache-2.0
Project-URL: Homepage, https://github.com/mopidy/mopidy-local
Classifier: Environment :: No Input/Output (Daemon)
Classifier: Intended Audience :: End Users/Desktop
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Operating System :: OS Independent
Classifier: Topic :: Multimedia :: Sound/Audio :: Players
Requires-Python: >=3.13
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: mopidy>=4.0.0rc2
Requires-Dist: pykka>=4.1
Requires-Dist: uritools>=4.0.3
Dynamic: license-file

# mopidy-local

[![Latest PyPI version](https://img.shields.io/pypi/v/mopidy-local)](https://pypi.org/p/mopidy-local)
[![CI build status](https://img.shields.io/github/actions/workflow/status/mopidy/mopidy-local/ci.yml)](https://github.com/mopidy/mopidy-local/actions/workflows/ci.yml)
[![Test coverage](https://img.shields.io/codecov/c/gh/mopidy/mopidy-local)](https://codecov.io/gh/mopidy/mopidy-local)

[Mopidy](https://mopidy.com/) extension for playing music from your local music
archive.

## Table of contents

- [Maintainer wanted](#maintainer-wanted)
- [Installation](#installation)
- [Configuration](#configuration)
- [Usage](#usage)
  - [Generating a library](#generating-a-library)
  - [Updating the library](#updating-the-library)
  - [Clearing the library](#clearing-the-library)
  - [Library layout](#library-layout)
- [Project resources](#project-resources)
- [Development](#development)
- [Credits](#credits)

## Maintainer wanted

Mopidy-Local is currently kept on life support by the Mopidy core developers.
It is in need of a more dedicated maintainer.

If you want to be the maintainer of Mopidy-Local, please:

1. Make 2-3 good pull requests improving any part of the project.

2. Read and get familiar with all of the project's open issues.

3. Send a pull request removing this section and adding yourself as the
   "Current maintainer" in the "Credits" section below. In the pull request
   description, please refer to the previous pull requests and state that
   you've familiarized yourself with the open issues.

   As a maintainer, you'll be given push access to the repo and the authority
   to make releases to PyPI when you see fit.

## Installation

Install by running:

```sh
python3 -m pip install mopidy-local
```

See https://mopidy.com/ext/local/ for alternative installation methods.

## Configuration

Before starting Mopidy, you must add configuration for Mopidy-Local to
your Mopidy configuration file:

```ini
[local]
media_dir = /path/to/your/music/archive
```

The following configuration values are available:

- `local/enabled`: If the local extension should be enabled or not.
  Defaults to `true`.
- `local/media_dir`: Path to directory with local media files.
- `local/max_search_results`: Number of search results that should be
  returned. Default is 100.
- `local/scan_timeout`: Number of milliseconds before giving up
  scanning a file and moving on to the next file.
- `local/scan_follow_symlinks`: If we should follow symlinks found in
  `local/media_dir`.
- `local/scan_flush_threshold`: Number of tracks to wait before
  telling library it should try and store its progress so far. Some
  libraries might not respect this setting. Set this to zero to
  disable flushing.
- `local/included_file_extensions`: File extensions to include when
  scanning the media directory. Values should be separated by either
  comma or newline. Each file extension should start with a dot, .e.g.
  `.flac`. Setting any values here will override the existence of
  `local/excluded_file_extensions`.
- `local/excluded_file_extensions`: File extensions to exclude when
  scanning the media directory. Values should be separated by either
  comma or newline. Each file extension should start with a dot, .e.g.
  `.html`. Defaults to a list of common non-audio file extensions
  often found in music collections. This config value has no effect if
  `local/included_file_extensions` is set.
- `local/directories`: List of top-level directory names and URIs for
  browsing. See below.
- `local/timeout`: Database connection timeout in seconds.
- `local/use_artist_sortname`: Whether to use the sortname field for
  ordering artist browse results. Disabled by default, since this may
  give confusing results if not all artists in the library have proper
  sortnames.
- `local/album_art_files`: List of file names to check for when
  searching for external album art. These may contain UNIX shell
  patterns, i.e. `*`, `?`, etc

# Usage

## Generating a library

The command `mopidy local scan` will scan the path set in the
`local/media_dir` config value for any audio files and build a library
of metadata.

To make a local library for your music available for Mopidy:

1.  Ensure that the `local/media_dir` config value points to where your
    music is located. Check the current setting by running:

        mopidy config

2.  Scan your media library.:

        mopidy local scan

3.  Start Mopidy, find the music library in a client, and play some
    local music!

## Updating the library

When you've added or removed music in your collection and want to
update Mopidy's index of your local library, you need to rescan:

```sh
mopidy local scan
```

Options can be specified to control the behavior of the scan command:

- `--force`: Force rescan of all media files
- `--limit <number>`: Maximum number of tracks to scan

Example:

```sh
mopidy local scan --limit 50
```

## Clearing the library

To delete your local images and clear your local library:

```sh
mopidy local clear
```

A prompt will ask you to confirm this irreversible operation.

## Library layout

The exposed library has a root directory and nine top-level directories
defined under the root directory:

- Albums
- Artists
- Composers
- Genres
- Performers
- Release Years
- Tracks
- Last Week's Updates
- Last Month's Updates

This can be configured through the `directories` setting. It's expected
to be a list of space separated name and URI supported for browsing, eg:

```ini
[local]
# ... other config...
directories =
    Albums                  local:directory?type=album
    Artists                 local:directory?type=artist
    Composers               local:directory?type=artist&role=composer
    Tracks                  local:directory?type=track
    Last Week's Updates     local:directory?max-age=604800
```

### URIs supported for browsing

_Remember that URIs are opaque values that neither Mopidy's core layer
or Mopidy frontends should attempt to derive any meaning from._ That
said, it's necessary to have a sufficient knowledge of Mopidy-Local
URIs to customize the `directories` setting properly:

- Browsing URIs starting with `local:artist` returns references to albums
  and tracks with the given artist.
- Browsing URIs starting with
  `local:album` returns references to the album tracks.
- Browsing URIs
  starting with `local:track` is not supported.

Other URIs supported for browsing start with `local:directory`. The
returned references are specified through "query parameters":

- `local:directory`: References to the top levels directories.
- `local:directory?type=tracks`: References all tracks. Multiple
  parameters can be added to filter the referenced tracks: `album`,
  `albumartist`, `artist`, `composer`, `date`, `genre`, `performer`,
  and `max-age`.
- `local:directory?type=date`: References to directories grouping
  tracks by date and album. Dates are transformed according to the
  optional parameter `FORMAT` which default to `%Y-%m-%d`. The URIs of
  the references start with `local:directory?date=`.
- `local:directory?type=genre`: References to directories named after
  genres found among all tracks. Their URIs start with
  `local:directory?genre=`.
- `local:directory?type=album`: References to all albums.
- `local:directory?type=album&PARAM=VALUE`: References to directories
  grouping tracks matching the given criteria. `PARAM` must be one of
  `albumartist`, `artist`, `composer`, `date`, `genre`, `performer`,
  `max-age`. The referenced directories group the selected tracks by
  album; Their URIs start with
  `local:directory?PARAM=VALUE&type=track&album=local:album:`.
- `local:directory?type=artist`: References to all artists.
- `local:directory?type=artist&role=ROLE`: References to directories
  with URIs `local:directory?ROLE=URI` where `URI` varies among all
  URIs starting with `local:artist` build from all tracks tag
  corresponding to `ROLE`. `ROLE` is one of `albumartist`, `artist`,
  `composer`, or `performer`.
- `local:directory?album=URI`: A reference to a directory grouping the
  tracks of the album with given URI. Its URI starts with
  `local:directory?album=URI&type=track`.
- `local:directory?albumartist=URI`: References to directories
  grouping tracks whose albumartist tag has given URI. The referenced
  directories group the selected tracks by album; Their URIs start
  with
  `local:directory?albumartist=URI&type=track&album=local:album:`.
- `local:directory?artist=URI`: References to directories grouping
  tracks whose artist has given URI. The referenced directories group
  the selected tracks by album; Their URIs start with
  `local:directory?artist=URI&type=track&album=local:album:`.
- `local:directory?composer=URI`: References to directories grouping
  tracks whose composer has given URI. The referenced directories
  group the selected tracks by album; Their URIs start with
  `local:directory?composer=URI&type=track&album=local:album:`.
- `local:directory?date=DATE`: References to directories grouping
  tracks whose date match DATE. The referenced directories group the
  selected tracks by album; Their URIs start with
  `local:directory?date=DATE&type=track&album=local:album:`. The match
  is to be interpreted as in a `LIKE` SQL statement.
- `local:directory?genre=GENRE`: References to directories grouping
  tracks whose genre is GENRE. The referenced directories group the
  selected tracks by album; Their URIs start with
  `local:directory?genre=GENRE&type=track&album=local:album:`.
- `local:directory?performer=URI`: References to directories grouping
  tracks whose performer has given URI. The referenced directories
  group the selected tracks by album; Their URIs start with
  `local:directory?performer=URI&type=track&album=local:album:`.
- `local:directory?max-age=SECONDS`: References to directories
  grouping tracks whose \"last modified\" date is newer than SECONDS
  seconds. The referenced directories group the selected tracks by
  album; Their URIs start with
  `local:directory?max-age=SECONDS&type=track&album=local:album:`.

## Project resources

- [Source code](https://github.com/mopidy/mopidy-local)
- [Issues](https://github.com/mopidy/mopidy-local/issues)
- [Releases](https://github.com/mopidy/mopidy-local/releases)

## Development

### Set up development environment

Clone the repo using, e.g. using [gh](https://cli.github.com/):

```sh
gh repo clone mopidy/mopidy-local
```

Enter the directory, and install dependencies using [uv](https://docs.astral.sh/uv/):

```sh
cd mopidy-local/
uv sync
```

### Running tests

To run all tests and linters in isolated environments, use
[tox](https://tox.wiki/):

```sh
tox
```

To only run tests, use [pytest](https://pytest.org/):

```sh
pytest
```

To format the code, use [ruff](https://docs.astral.sh/ruff/):

```sh
ruff format .
```

To check for lints with ruff, run:

```sh
ruff check .
```

To check for type errors, use [pyright](https://microsoft.github.io/pyright/):

```sh
pyright .
```

### Making a release

To make a release to PyPI, go to the project's [GitHub releases
page](https://github.com/mopidy/mopidy-local/releases)
and click the "Draft a new release" button.

In the "choose a tag" dropdown, select the tag you want to release or create a
new tag, e.g. `v0.1.0`. Add a title, e.g. `v0.1.0`, and a description of the changes.

Decide if the release is a pre-release (alpha, beta, or release candidate) or
should be marked as the latest release, and click "Publish release".

Once the release is created, the `release.yml` GitHub Action will automatically
build and publish the release to
[PyPI](https://pypi.org/project/mopidy-local/).

## Credits

- Original authors: [Stein Magnus Jodal](https://github.com/jodal) and
  [Thomas Adamcik](https://github.com/adamcik) for the Mopidy-Local
  extension in Mopidy core. [Thomas Kemmer](https://github.com/tkem)
  for the SQLite storage and support for embedded album art.
- Current maintainer: None. Maintainer wanted, see section above.
- [Contributors](https://github.com/mopidy/mopidy-local/graphs/contributors)
