Compare commits
5 Commits
3702bc54a8
...
42e4aa02b6
Author | SHA1 | Date | |
---|---|---|---|
42e4aa02b6 | |||
d060a30c45 | |||
79df83d1b1 | |||
c1f8066688 | |||
0fe55ba06d |
@@ -1,3 +1,30 @@
|
|||||||
|
# List artists
|
||||||
|
#
|
||||||
|
# parameter file_local_artists: This is an optional parameter with the path
|
||||||
|
# to a file with a MusicBrainz artist ID per
|
||||||
|
# line.
|
||||||
|
# parameter format_person: This is the format string for single person
|
||||||
|
# artists, which includes the <<name>>
|
||||||
|
# placeholder.
|
||||||
|
# parameter format_disambiguation: This is the format string for the
|
||||||
|
# disambiguation part of the artist, with a
|
||||||
|
# placeholder <<disambiguation>>.
|
||||||
|
# parameter format_group: This is as format_person, but for music
|
||||||
|
# groups.
|
||||||
|
# parameter format_local: String to indicate that there is some music
|
||||||
|
# locally available of an artist
|
||||||
|
#
|
||||||
|
# This awk program takes as input a sequence of lines where the first item is
|
||||||
|
# the MusicBrainz artist ID, the second item is the type of the artist
|
||||||
|
# ("Person" or "Group"), the third item is the name, and the forth item is a
|
||||||
|
# disambiguation string.
|
||||||
|
#
|
||||||
|
# The output of this script is a sequence of tab-delimited lines where the
|
||||||
|
# first item is the format_local string, if some music of that artist is
|
||||||
|
# locally accessible, and the empty string otherwise, the second item is the
|
||||||
|
# formatted artist string (formatted according to format_person or
|
||||||
|
# format_group), the third item is the constant string "0" indicating the
|
||||||
|
# parent MusicBrainz ID, and the last item is the MusicBrainz artist ID.
|
||||||
BEGIN {
|
BEGIN {
|
||||||
OFS="\t"
|
OFS="\t"
|
||||||
local_artists[0] = 0
|
local_artists[0] = 0
|
||||||
|
@@ -1,3 +1,39 @@
|
|||||||
|
# List recordings
|
||||||
|
#
|
||||||
|
# parameter file_local_recordings: This is an optional parameter with the path
|
||||||
|
# to a file with a MusicBrainz recording ID
|
||||||
|
# per line.
|
||||||
|
# parameter format: The format of a recording line including the
|
||||||
|
# placeholders <<med>> for medium number,
|
||||||
|
# <<nr> for for track number within a medium,
|
||||||
|
# <<title>> for the title, <<artist>> for the
|
||||||
|
# artist string, and <<duration>> for the
|
||||||
|
# track duration.
|
||||||
|
# parameter format_local: String to indicate that the track is locally
|
||||||
|
# available
|
||||||
|
# parameter format_current: String to indicate that the track is
|
||||||
|
# "currently playing"
|
||||||
|
# parameter current_id: MusicBrainz track ID of a track to be marked
|
||||||
|
# as "currently playing"
|
||||||
|
#
|
||||||
|
# The input to this awk program is a sequence of lines containing the following fields:
|
||||||
|
# Field 1: The MusicBrainz ID of the release this track belongs to
|
||||||
|
# Field 2: MusicBrainz ID of this track
|
||||||
|
# Field 3: Medium number of this track within the release
|
||||||
|
# Field 4: Track number of this track within the medium
|
||||||
|
# Field 5: Duration of this track in miliseconds
|
||||||
|
# Field 6: Title of this track
|
||||||
|
# Field 7: Artist of this track
|
||||||
|
# Field 8: Path to decoratoin file of this release
|
||||||
|
#
|
||||||
|
# The output is a sequence of tab-delimited lines containing the following fields:
|
||||||
|
# Field 1: Sort value (to sort the track within the release)
|
||||||
|
# Field 2: The string `format_local` if the track is locally available
|
||||||
|
# Field 3: The string `format_current` if the track has MusicBrainz ID `current_id`
|
||||||
|
# Field 4: The track line to be displayed according to `format`
|
||||||
|
# Field 5: The MusicBrainz ID of the release this track belongs to
|
||||||
|
# Field 6: The MusicBrainz ID of this track ":" separated from the path to the
|
||||||
|
# decoration file of this release
|
||||||
BEGIN {
|
BEGIN {
|
||||||
OFS="\t"
|
OFS="\t"
|
||||||
local_recordings[0] = 0
|
local_recordings[0] = 0
|
||||||
@@ -9,7 +45,6 @@ BEGIN {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
gsub("&", "\\\\&")
|
|
||||||
parentid = $1
|
parentid = $1
|
||||||
id = $2
|
id = $2
|
||||||
med = $3
|
med = $3
|
||||||
@@ -18,6 +53,8 @@ BEGIN {
|
|||||||
title = $6
|
title = $6
|
||||||
artist = $7
|
artist = $7
|
||||||
deco = local_recordings[id] ? $8 : ""
|
deco = local_recordings[id] ? $8 : ""
|
||||||
|
gsub("&", "\\\\&", title)
|
||||||
|
gsub("&", "\\\\&", artist)
|
||||||
# Parse duration
|
# Parse duration
|
||||||
if (dur) {
|
if (dur) {
|
||||||
dur = int(dur / 1000)
|
dur = int(dur / 1000)
|
||||||
|
@@ -1,3 +1,69 @@
|
|||||||
|
# List release groups
|
||||||
|
#
|
||||||
|
# parameter file_local_releasegroups: This is an optional parameter with the
|
||||||
|
# path to a file with a MusicBrainz
|
||||||
|
# release-group ID per line.
|
||||||
|
# parameter format_release: Format for the release title with a
|
||||||
|
# <<title>> placeholder.
|
||||||
|
# parameter format_release_w_artist: Same as `format_release` but with an
|
||||||
|
# additional <<artist>> placeholder.
|
||||||
|
# parameter format_year: Format string for the year part, with a
|
||||||
|
# <<year>> placeholder.
|
||||||
|
# parameter format_local: String to indicate that the track is
|
||||||
|
# locally available.
|
||||||
|
# parameter artist: Artist name to compare release-groups
|
||||||
|
# artist names against. If the names
|
||||||
|
# differ, then the format with <<artist>>
|
||||||
|
# placeholder is used (optional)
|
||||||
|
# parameter artistid MusicBrainz ID of the artist (optional).
|
||||||
|
# Then, there are several format strings that indicate the type of a release
|
||||||
|
# group. The types are implicit from the parameter names:
|
||||||
|
# parameter format_album
|
||||||
|
# parameter format_single
|
||||||
|
# parameter format_ep
|
||||||
|
# parameter format_broadcast
|
||||||
|
# parameter format_other
|
||||||
|
# Some release groups have also a secondary type. The presence of a secondary
|
||||||
|
# type is formatted using
|
||||||
|
# parameter format_secondary.
|
||||||
|
# The list of all secondary types (implicit from their variable names) are
|
||||||
|
# formatted using
|
||||||
|
# parameter format_secondary.
|
||||||
|
# Each of the secondary types is specified with (the type is implicit from the
|
||||||
|
# variable names)
|
||||||
|
# parameter format_compilation
|
||||||
|
# parameter format_soundtrack
|
||||||
|
# parameter format_spokenword
|
||||||
|
# parameter format_interview
|
||||||
|
# parameter format_audiobook
|
||||||
|
# parameter format_audiodrama
|
||||||
|
# parameter format_live
|
||||||
|
# parameter format_remix
|
||||||
|
# parameter format_djmix
|
||||||
|
# parameter format_mixtape
|
||||||
|
# parameter format_demo
|
||||||
|
# parameter format_fieldrec
|
||||||
|
#
|
||||||
|
# The input to this awk program is a sequence of lines containing the following
|
||||||
|
# fields:
|
||||||
|
# Field 1: The MusicBrainz ID of the release group
|
||||||
|
# Field 2: The primary type
|
||||||
|
# Field 3: A ;-delimited string of secondary types
|
||||||
|
# Field 4: The original release year
|
||||||
|
# Field 5: Title of the release group
|
||||||
|
# Field 6: The artist as credited
|
||||||
|
#
|
||||||
|
# The output is a sequence of tab-delimited lines with the fields:
|
||||||
|
# Field 1: Sort value to sort release groups
|
||||||
|
# Field 2: The flag `format_local` if the release group is accessible locally,
|
||||||
|
# and "" else.
|
||||||
|
# Field 3: Release-group type
|
||||||
|
# Field 4: Release-group string
|
||||||
|
# Field 5: Release-group year
|
||||||
|
# Field 6: Secondary types
|
||||||
|
# Field 7: MusicBrainz artist ID of the release group artist, if there is one,
|
||||||
|
# else "0"
|
||||||
|
# Field 8: MusicBrainz release-group ID
|
||||||
BEGIN {
|
BEGIN {
|
||||||
OFS="\t"
|
OFS="\t"
|
||||||
local_releasegroups[0] = 0
|
local_releasegroups[0] = 0
|
||||||
@@ -9,9 +75,9 @@ BEGIN {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
line_type=""
|
line_type = ""
|
||||||
line_sectype=""
|
line_sectype = ""
|
||||||
line_year=""
|
line_year = ""
|
||||||
gsub("&", "\\\\&")
|
gsub("&", "\\\\&")
|
||||||
id = $1
|
id = $1
|
||||||
type = $2
|
type = $2
|
||||||
|
@@ -1,3 +1,57 @@
|
|||||||
|
# List releases
|
||||||
|
#
|
||||||
|
# parameter file_local_releases: This is an optional parameter with the
|
||||||
|
# path to a file with a MusicBrainz
|
||||||
|
# release ID per line.
|
||||||
|
# parameter format_release: Format for the release with the
|
||||||
|
# placeholders <<status>>, <<tracks>>,
|
||||||
|
# <<media>>, <<year>>, <<country>>, and
|
||||||
|
# <<label>>.
|
||||||
|
# parameter format_release_title_artist: Format to specify title and artist,
|
||||||
|
# with the placeholders <<title>> and
|
||||||
|
# <<artist>>.
|
||||||
|
# parameter format_release_title: Format to specify the release title
|
||||||
|
# with a placeholder <<title>>.
|
||||||
|
# parameter format_release_artist: Format to specify the release artist
|
||||||
|
# with a placeholder <<artist>>.
|
||||||
|
# parameter format_local: String to indicate that the track is
|
||||||
|
# locally available.
|
||||||
|
# parameter rg_artist: Artist name of release group
|
||||||
|
# (optional)
|
||||||
|
# parameter rg_title: Title of release group (optional)
|
||||||
|
# parameter rgid: MusicBrainz release-group ID
|
||||||
|
# (optional)
|
||||||
|
# Then, there are several format strings that indicate the status of a release.
|
||||||
|
# The status are implicit from the parameter names:
|
||||||
|
# parameter release_official
|
||||||
|
# parameter release_promotion
|
||||||
|
# parameter release_bootleg
|
||||||
|
# parameter release_pseudo
|
||||||
|
# parameter release_withdrawn
|
||||||
|
# parameter release_expunged
|
||||||
|
# parameter release_cancelled
|
||||||
|
#
|
||||||
|
# The input to this awk program is a sequence of lines containing the following
|
||||||
|
# fields:
|
||||||
|
# Field 1: MusicBrainz ID of the release
|
||||||
|
# Field 2: Release status
|
||||||
|
# Field 3: Release date
|
||||||
|
# Field 4: Number of cover-art images
|
||||||
|
# Field 5: Label string (', '-delimited)
|
||||||
|
# Field 6: Total number of tracks
|
||||||
|
# Field 7: Format (', '-delimited)
|
||||||
|
# Field 8: Release country
|
||||||
|
# Field 9: Release title
|
||||||
|
# Field 10: Artist as credited
|
||||||
|
#
|
||||||
|
# The output is a sequence of tab-delimited lines with the fields:
|
||||||
|
# Field 1: Sort value to sort release groups
|
||||||
|
# Field 2: The flag `format_local` if the release is accessible locally, and ""
|
||||||
|
# else.
|
||||||
|
# Field 3: Release line
|
||||||
|
# Field 4: MusicBrainz release-group ID if present, else "0"
|
||||||
|
# Field 5: MusicBrainz release ID followed by ":" and then a path to the
|
||||||
|
# decoration file (if it exists)
|
||||||
BEGIN {
|
BEGIN {
|
||||||
OFS="\t"
|
OFS="\t"
|
||||||
local_releases[0] = 0
|
local_releases[0] = 0
|
||||||
@@ -30,13 +84,13 @@ BEGIN {
|
|||||||
case "Cancelled": line_status = release_cancelled; break
|
case "Cancelled": line_status = release_cancelled; break
|
||||||
default: line_status = ""
|
default: line_status = ""
|
||||||
}
|
}
|
||||||
line = release_format
|
line = format_release
|
||||||
if (artist != rg_artist && title != rg_title)
|
if (artist != rg_artist && title != rg_title)
|
||||||
line = line "\t" release_format_title_artist
|
line = line "\t" format_release_title_artist
|
||||||
else if (artist != rg_artist && title == rg_title)
|
else if (artist != rg_artist && title == rg_title)
|
||||||
line = line "\t" release_format_artist
|
line = line "\t" format_release_artist
|
||||||
else if (artist == rg_artist && title != rg_title)
|
else if (artist == rg_artist && title != rg_title)
|
||||||
line = line "\t" release_format_title
|
line = line "\t" format_release_title
|
||||||
else
|
else
|
||||||
line = line "\t"
|
line = line "\t"
|
||||||
|
|
||||||
|
@@ -110,7 +110,7 @@ case "${1:-}" in
|
|||||||
"$VIEW_LIST_ARTISTS") list_local_artists ;;
|
"$VIEW_LIST_ARTISTS") list_local_artists ;;
|
||||||
"$VIEW_LIST_ALBUMS") list_local_releasegroups ;;
|
"$VIEW_LIST_ALBUMS") list_local_releasegroups ;;
|
||||||
"$VIEW_PLAYLIST") list_playlist ;;
|
"$VIEW_PLAYLIST") list_playlist ;;
|
||||||
"$VIEW_SEARCH_ARTIST" | "$VIEW_SEARCH_ALBUM") fzf_reload_after_change ;;
|
"$VIEW_SEARCH_ARTIST" | "$VIEW_SEARCH_ALBUM") mb_results_async ;;
|
||||||
esac
|
esac
|
||||||
exit 0
|
exit 0
|
||||||
;;
|
;;
|
||||||
@@ -298,7 +298,7 @@ case "${1:-}" in
|
|||||||
# This stops any search being executed and initiates a new query through the
|
# This stops any search being executed and initiates a new query through the
|
||||||
# MusicBrainz API. The results will be made available through the ``--lines
|
# MusicBrainz API. The results will be made available through the ``--lines
|
||||||
# <view>`` command.
|
# <view>`` command.
|
||||||
fzf_handle_change "$2"
|
mb_search_async "$2"
|
||||||
exit 0
|
exit 0
|
||||||
;;
|
;;
|
||||||
"--preview-artist")
|
"--preview-artist")
|
||||||
@@ -308,7 +308,7 @@ case "${1:-}" in
|
|||||||
#
|
#
|
||||||
# This prints the text to be displayed in the preview window for the
|
# This prints the text to be displayed in the preview window for the
|
||||||
# specified artist.
|
# specified artist.
|
||||||
__preview_artist "$2"
|
preview_artist "$2"
|
||||||
exit 0
|
exit 0
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
@@ -365,7 +365,7 @@ case "${1:-}" in
|
|||||||
Usage: $0 [OPTION]
|
Usage: $0 [OPTION]
|
||||||
|
|
||||||
GENERAL OPTIONS:
|
GENERAL OPTIONS:
|
||||||
--help Show this help and exit.
|
--help Show this help and exit
|
||||||
--artists Default options, list artists of local music
|
--artists Default options, list artists of local music
|
||||||
--albums List albums of local music
|
--albums List albums of local music
|
||||||
--search-artist Search artist on MusicBrainz
|
--search-artist Search artist on MusicBrainz
|
||||||
|
@@ -1,3 +1,10 @@
|
|||||||
|
# This file provides the methods for access to several APIs
|
||||||
|
#
|
||||||
|
# APIs:
|
||||||
|
# - MusicBrainz
|
||||||
|
# - Discogs
|
||||||
|
# - Wikidata
|
||||||
|
# - Wikipedia
|
||||||
if [ ! "${API_LOADED:-}" ]; then
|
if [ ! "${API_LOADED:-}" ]; then
|
||||||
MB_MAX_RETRIES=10
|
MB_MAX_RETRIES=10
|
||||||
MB_BROWSE_STEPS=100
|
MB_BROWSE_STEPS=100
|
||||||
@@ -8,6 +15,16 @@ if [ ! "${API_LOADED:-}" ]; then
|
|||||||
export API_LOADED=1
|
export API_LOADED=1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Internal method for MusicBrainz API access
|
||||||
|
#
|
||||||
|
# @argument $1: entity (see `case` below)
|
||||||
|
# @argument $2: MusicBrainz ID
|
||||||
|
# @argument $3: offset (optional, but mandatory for browse requests)
|
||||||
|
#
|
||||||
|
# If the API access fails, then the error message is logged, and at most
|
||||||
|
# `MB_MAX_RETRIES` retries are made. If browse requests are made, then at most
|
||||||
|
# `MB_BROWSE_STEPS` number of entries are requested per call. The offset in
|
||||||
|
# browse request must be specified.
|
||||||
__api_mb() {
|
__api_mb() {
|
||||||
tmpout=$(mktemp)
|
tmpout=$(mktemp)
|
||||||
for _ in $(seq "$MB_MAX_RETRIES"); do
|
for _ in $(seq "$MB_MAX_RETRIES"); do
|
||||||
@@ -98,35 +115,62 @@ __api_mb() {
|
|||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# The interface to MusicBrainz API.
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz artist information
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
api_mb_artist() {
|
api_mb_artist() {
|
||||||
__api_mb "artist" "$1"
|
__api_mb "artist" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz release-group information
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
api_mb_releasegroup() {
|
api_mb_releasegroup() {
|
||||||
__api_mb "releasegroup" "$1"
|
__api_mb "releasegroup" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz release information
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
api_mb_release() {
|
api_mb_release() {
|
||||||
__api_mb "release" "$1"
|
__api_mb "release" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz release-groups for given artist
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
|
# @argument $2: offset (defaults to 0)
|
||||||
api_mb_browse_artist_releasegroups() {
|
api_mb_browse_artist_releasegroups() {
|
||||||
__api_mb "browse-artist-releasegroups" "$1" "${2:-0}"
|
__api_mb "browse-artist-releasegroups" "$1" "${2:-0}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz releases in given release group
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
|
# @argument $2: offset (defaults to 0)
|
||||||
api_mb_browse_releasegroup_releases() {
|
api_mb_browse_releasegroup_releases() {
|
||||||
__api_mb "browse-releasegroup-releases" "$1" "${2:-0}"
|
__api_mb "browse-releasegroup-releases" "$1" "${2:-0}"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Argument: Search string
|
# Search MusicBrainz database for given artist
|
||||||
|
#
|
||||||
|
# @argument $1: query
|
||||||
api_mb_search_artist() {
|
api_mb_search_artist() {
|
||||||
__api_mb "search-artist" "$1"
|
__api_mb "search-artist" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Search MusicBrainz database for given release group
|
||||||
|
#
|
||||||
|
# @argument $1: query
|
||||||
api_mb_search_releasegroup() {
|
api_mb_search_releasegroup() {
|
||||||
__api_mb "search-releasegroup" "$1"
|
__api_mb "search-releasegroup" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve Discogs artist information
|
||||||
|
#
|
||||||
|
# @argument $1: Discogs artist ID
|
||||||
api_discogs_artist() {
|
api_discogs_artist() {
|
||||||
$CURL \
|
$CURL \
|
||||||
--get \
|
--get \
|
||||||
@@ -134,6 +178,9 @@ api_discogs_artist() {
|
|||||||
"https://api.discogs.com/artists/$1"
|
"https://api.discogs.com/artists/$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve sitelinks from wikidata
|
||||||
|
#
|
||||||
|
# @argument $1: Wikidata ID
|
||||||
api_wikidata_sitelinks() {
|
api_wikidata_sitelinks() {
|
||||||
$CURL \
|
$CURL \
|
||||||
--get \
|
--get \
|
||||||
@@ -141,6 +188,9 @@ api_wikidata_sitelinks() {
|
|||||||
"https://www.wikidata.org/w/rest.php/wikibase/v1/entities/items/$1/sitelinks"
|
"https://www.wikidata.org/w/rest.php/wikibase/v1/entities/items/$1/sitelinks"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve summary from Wikipedia page
|
||||||
|
#
|
||||||
|
# @argument $1: Wikipedia page name
|
||||||
api_wikipedia_en_summary() {
|
api_wikipedia_en_summary() {
|
||||||
$CURL \
|
$CURL \
|
||||||
--get \
|
--get \
|
||||||
|
@@ -1,3 +1,5 @@
|
|||||||
|
# The code below is used together with `scripts/build.sh`to internalize the awk
|
||||||
|
# scripts. See the awk sources for more information.
|
||||||
if [ ! "${AWK_LOADED:-}" ]; then
|
if [ ! "${AWK_LOADED:-}" ]; then
|
||||||
AWK_ARTISTS=$(
|
AWK_ARTISTS=$(
|
||||||
cat <<'EOF'
|
cat <<'EOF'
|
||||||
|
127
src/sh/cache.sh
127
src/sh/cache.sh
@@ -1,17 +1,27 @@
|
|||||||
# Caching structure
|
# This implements the caching functionalities. The cache is stored under
|
||||||
#
|
# `CACHEDIR` defined below, and organized as follows (all paths relative to
|
||||||
# ./artist/radix(uuid)/musicbrainz.json # Artist information
|
# `CAHCEDIR`) ./<type>/radix(mbid)/<file>. Here, type is one of `TYPE_ARTIST`,
|
||||||
# ./artist/radix(uuid)/releasegroups.json # List of all release groups
|
# `TYPE_RELEASEGROUP`, or `TYPE_RELEASE`. The string `radix(mbid)` is the radix
|
||||||
# ./artist/radix(uuid)/... # Any other artist information
|
# encoded MusicBrainz ID of given type (see method below). Finally <file> is a
|
||||||
# ./releasegroup/radix(uuid)/musicbrainz.json # Release group information
|
# filename to hold the respective data in the json format. Currently, the data
|
||||||
# ./releasegroup/radix(uuid)/releases.json # List of all releases in release group
|
# is stored as follows:
|
||||||
# ./release/radix(uuid)/musicbrainz.json # Release information with tracklist etc.
|
# ./artist/radix(mbid)/musicbrainz.json MusicBrainz artist data
|
||||||
|
# ./artist/radix(mbid)/discogs.json Discogs artist data
|
||||||
|
# ./artist/radix(mbid)/wikidata.json Wikidata artist data
|
||||||
|
# ./artist/radix(mbid)/enwikipedia.json Wikipedia artist data
|
||||||
|
# ./artist/radix(mbid)/releasegroups.json Release groups of artist
|
||||||
|
# ./releasegroup/radix(mbid)/musicbrainz.json MusicBrainz release-group data
|
||||||
|
# ./releasegroup/radix(mbid)/releases.json Releases in release group
|
||||||
|
# ./release/radix(mbid)/musicbrainz.json MusicBrainz release data
|
||||||
|
|
||||||
if [ ! "${CACHE_LOADED:-}" ]; then
|
if [ ! "${CACHE_LOADED:-}" ]; then
|
||||||
|
# Base path for cache
|
||||||
CACHEDIR="$HOME/.cache/$APP_NAME"
|
CACHEDIR="$HOME/.cache/$APP_NAME"
|
||||||
|
# Directory names for cache types
|
||||||
TYPE_ARTIST="artist"
|
TYPE_ARTIST="artist"
|
||||||
TYPE_RELEASEGROUP="releasegroup"
|
TYPE_RELEASEGROUP="releasegroup"
|
||||||
TYPE_RELEASE="release"
|
TYPE_RELEASE="release"
|
||||||
|
# Filenames for cache entries
|
||||||
ARTIST_FILENAME="musicbrainz.json"
|
ARTIST_FILENAME="musicbrainz.json"
|
||||||
ARTIST_RELEASEROUPS_FILENAME="releasegroups.json"
|
ARTIST_RELEASEROUPS_FILENAME="releasegroups.json"
|
||||||
ARTIST_DISCOGS_FILENAME="discogs.json"
|
ARTIST_DISCOGS_FILENAME="discogs.json"
|
||||||
@@ -28,17 +38,22 @@ if [ ! "${CACHE_LOADED:-}" ]; then
|
|||||||
export CACHE_LOADED=1
|
export CACHE_LOADED=1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Radix transform directory name
|
# Radix transform string
|
||||||
|
#
|
||||||
|
# @argument $1: some string
|
||||||
__radix() {
|
__radix() {
|
||||||
echo "$1" | awk -F "" '{ print $1$2$3$4"/"$5$6$7$8"/"$0 }'
|
echo "$1" | awk -F "" '{ print $1$2$3$4"/"$5$6$7$8"/"$0 }'
|
||||||
}
|
}
|
||||||
|
|
||||||
# Radix transform directory names from stdin
|
# Radix transform strings (batch)
|
||||||
|
#
|
||||||
|
# Here, the input is read line-by-line from stdin.
|
||||||
__radix_batch() {
|
__radix_batch() {
|
||||||
cat | awk -F "" '{ print $1$2$3$4"/"$5$6$7$8"/"$0 }'
|
cat | awk -F "" '{ print $1$2$3$4"/"$5$6$7$8"/"$0 }'
|
||||||
}
|
}
|
||||||
|
|
||||||
# Super wrapper
|
# Super wrapper to print json data from cache
|
||||||
|
#
|
||||||
# argument $1: type
|
# argument $1: type
|
||||||
# argument $2: MusicBrainz ID
|
# argument $2: MusicBrainz ID
|
||||||
# argument $3: Filename of json file
|
# argument $3: Filename of json file
|
||||||
@@ -48,7 +63,8 @@ __get_json() {
|
|||||||
cat "$f"
|
cat "$f"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Super wrapper
|
# Super wrapper to store json data in cache
|
||||||
|
#
|
||||||
# argument $1: type
|
# argument $1: type
|
||||||
# argument $2: MusicBrainz ID
|
# argument $2: MusicBrainz ID
|
||||||
# argument $3: Filename of json file
|
# argument $3: Filename of json file
|
||||||
@@ -61,35 +77,64 @@ __put_json() {
|
|||||||
[ -s "$tmpf" ] && mv "$tmpf" "$f" || printf "{}" >"$f"
|
[ -s "$tmpf" ] && mv "$tmpf" "$f" || printf "{}" >"$f"
|
||||||
}
|
}
|
||||||
|
|
||||||
## Artist
|
# Print MusicBrainz data of given artist from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_get_artist() {
|
cache_get_artist() {
|
||||||
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_FILENAME"
|
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print release groups (MusicBrainz) of given artist from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_get_artist_releasegroups() {
|
cache_get_artist_releasegroups() {
|
||||||
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_RELEASEROUPS_FILENAME"
|
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_RELEASEROUPS_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print Discogs data of given artist from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_get_artist_discogs() {
|
cache_get_artist_discogs() {
|
||||||
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_DISCOGS_FILENAME"
|
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_DISCOGS_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print Wikipedia data of given artist from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_get_artist_enwikipedia() {
|
cache_get_artist_enwikipedia() {
|
||||||
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_ENWIKIPEDIA_FILENAME"
|
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_ENWIKIPEDIA_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print Wikidata data of given artist from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_get_artist_wikidata() {
|
cache_get_artist_wikidata() {
|
||||||
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_WIKIDATA_FILENAME"
|
__get_json "$TYPE_ARTIST" "$1" "$ARTIST_WIKIDATA_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store MusicBrainz data of given artist in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
|
#
|
||||||
|
# This methods reads the data to be stored from stdin.
|
||||||
cache_put_artist() {
|
cache_put_artist() {
|
||||||
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_FILENAME"
|
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store release groups (MusicBrainz) of given artist in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
|
#
|
||||||
|
# This methods reads the data to be stored from stdin.
|
||||||
cache_put_artist_releasegroups() {
|
cache_put_artist_releasegroups() {
|
||||||
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_RELEASEROUPS_FILENAME"
|
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_RELEASEROUPS_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Append release groups (MusicBrainz) of given artist to existing file in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
|
#
|
||||||
|
# This methods reads the data to be stored from stdin.
|
||||||
cache_append_artist_releasegroups() {
|
cache_append_artist_releasegroups() {
|
||||||
tmpf=$(mktemp)
|
tmpf=$(mktemp)
|
||||||
cat >"$tmpf"
|
cat >"$tmpf"
|
||||||
@@ -99,35 +144,59 @@ cache_append_artist_releasegroups() {
|
|||||||
rm -f "$tmpf"
|
rm -f "$tmpf"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store Discogs data of given artist to cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_put_artist_discogs() {
|
cache_put_artist_discogs() {
|
||||||
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_DISCOGS_FILENAME"
|
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_DISCOGS_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store Wikipedia data of given artist to cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_put_artist_enwikipedia() {
|
cache_put_artist_enwikipedia() {
|
||||||
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_ENWIKIPEDIA_FILENAME"
|
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_ENWIKIPEDIA_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store Wikidata data of given artist to cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
cache_put_artist_wikidata() {
|
cache_put_artist_wikidata() {
|
||||||
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_WIKIDATA_FILENAME"
|
cat | __put_json "$TYPE_ARTIST" "$1" "$ARTIST_WIKIDATA_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
## Release group
|
# Print MusicBrainz data of given release group from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
cache_get_releasegroup() {
|
cache_get_releasegroup() {
|
||||||
__get_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_FILENAME"
|
__get_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print releases (MusicBrainz) in release group from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
cache_get_releasegroup_releases() {
|
cache_get_releasegroup_releases() {
|
||||||
__get_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_RELEASES_FILENAME"
|
__get_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_RELEASES_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store MusicBrainz data of given release group in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
cache_put_releasegroup() {
|
cache_put_releasegroup() {
|
||||||
cat | __put_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_FILENAME"
|
cat | __put_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store releases (MusicBrainz) of given release group in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
cache_put_releasegroup_releases() {
|
cache_put_releasegroup_releases() {
|
||||||
cat | __put_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_RELEASES_FILENAME"
|
cat | __put_json "$TYPE_RELEASEGROUP" "$1" "$RELEASEGROUP_RELEASES_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Append releases (MusicBrainz) of given release group to existing file in
|
||||||
|
# cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
cache_append_releasegroup_releases() {
|
cache_append_releasegroup_releases() {
|
||||||
tmpf=$(mktemp)
|
tmpf=$(mktemp)
|
||||||
cat >"$tmpf"
|
cat >"$tmpf"
|
||||||
@@ -137,40 +206,26 @@ cache_append_releasegroup_releases() {
|
|||||||
rm -f "$tmpf"
|
rm -f "$tmpf"
|
||||||
}
|
}
|
||||||
|
|
||||||
## Release
|
# Print MusicBrainz data of given release from cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
cache_get_release() {
|
cache_get_release() {
|
||||||
__get_json "$TYPE_RELEASE" "$1" "$RELEASE_FILENAME"
|
__get_json "$TYPE_RELEASE" "$1" "$RELEASE_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Store MusicBrainz data of given release in cache
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
cache_put_release() {
|
cache_put_release() {
|
||||||
cat | __put_json "$TYPE_RELEASE" "$1" "$RELEASE_FILENAME"
|
cat | __put_json "$TYPE_RELEASE" "$1" "$RELEASE_FILENAME"
|
||||||
}
|
}
|
||||||
|
|
||||||
## Cache deletion
|
# Print all MusicBrainz cache paths to the files specified by their IDs
|
||||||
cache_delete_artist() {
|
|
||||||
# Get release groups
|
|
||||||
echo "NOT IMPLEMENTED" >/dev/stderr
|
|
||||||
}
|
|
||||||
|
|
||||||
# Check if main items are in cache
|
|
||||||
# argument $1: type
|
|
||||||
# argument $2: MusicBrainz ID
|
|
||||||
in_cache() {
|
|
||||||
case "$1" in
|
|
||||||
"$TYPE_ARTIST") fn="$ARTIST_FILENAME" ;;
|
|
||||||
"$TYPE_RELEASEGROUP") fn="$RELEASEGROUP_FILENAME" ;;
|
|
||||||
"$TYPE_RELEASE") fn="$RELEASE_FILENAME" ;;
|
|
||||||
*) return 1 ;;
|
|
||||||
esac
|
|
||||||
[ "$(__get_json "$1" "$2" "$fn")" ] && return 0 || return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
# Print all cache paths to the files specified by their IDs
|
|
||||||
#
|
#
|
||||||
# @argument $1: type
|
# @argument $1: type
|
||||||
#
|
#
|
||||||
# This method reads from stdin any number of MusicBrainz IDs of objects of the
|
# This method reads from stdin any number of MusicBrainz IDs of objects of the
|
||||||
# specified type, and prints the file pahts.
|
# specified type, and prints the file paths.
|
||||||
cache_get_file_batch() {
|
cache_get_file_batch() {
|
||||||
case "$1" in
|
case "$1" in
|
||||||
"$TYPE_ARTIST") fn="$ARTIST_FILENAME" ;;
|
"$TYPE_ARTIST") fn="$ARTIST_FILENAME" ;;
|
||||||
|
@@ -1,4 +1,16 @@
|
|||||||
# Configuration capabilities
|
# Main application configuration. This application does not require a
|
||||||
|
# configuration file. However, a configuration file may be stored as
|
||||||
|
# `CONFIGFILE_DEFAULT`. If that file exists, it will be sourced. The path to
|
||||||
|
# the file may be overwritten by specifying the environment variable
|
||||||
|
# `CONFIGFILE`. If a configuration file is specified, then it must also exist.
|
||||||
|
# A configuration file comprises the specification of environment variables
|
||||||
|
# that are` allowed to be set.
|
||||||
|
#
|
||||||
|
# Currently, the following files hold variables that are configurable:
|
||||||
|
# - `src/sh/filter.sh`: Configuration of filters that can be triggered with
|
||||||
|
# the respective key bindings.
|
||||||
|
# - `src/sh/keys.sh`: Configuration of key bindings to certain actions
|
||||||
|
# - `src/sh/theme.sh`: Configuration of theme
|
||||||
CONFIGFILE_DEFAULT="$HOME/.config/$APP_NAME/config"
|
CONFIGFILE_DEFAULT="$HOME/.config/$APP_NAME/config"
|
||||||
CONFIGFILE="${CONFIGFILE:-"$CONFIGFILE_DEFAULT"}"
|
CONFIGFILE="${CONFIGFILE:-"$CONFIGFILE_DEFAULT"}"
|
||||||
[ "$CONFIGFILE" != "$CONFIGFILE_DEFAULT" ] && [ ! -f "$CONFIGFILE" ] && err "The configuration file manually specified with the environment variable CONFIGFILE=($CONFIGFILE) does not exist." && exit 1
|
[ "$CONFIGFILE" != "$CONFIGFILE_DEFAULT" ] && [ ! -f "$CONFIGFILE" ] && err "The configuration file manually specified with the environment variable CONFIGFILE=($CONFIGFILE) does not exist." && exit 1
|
||||||
|
@@ -1,7 +1,14 @@
|
|||||||
# Preset filters
|
# Preset filters for different views. These filters are associated to key
|
||||||
#
|
# bindings (see `src/sh/keys.sh`), and are configurable through a configuration
|
||||||
# See `src/sh/query.sh` for details and the associated methods.
|
# file (see `src/sh/config.sh`).
|
||||||
|
|
||||||
|
# The `QUERY_LOCAL` filter is associated with the keys `KEYS_FILTER_LOCAL`. It
|
||||||
|
# is used to hide all entries that are not available locally (see
|
||||||
|
# `src/sh/query.sh` for details and the relevant methods)
|
||||||
QUERY_LOCAL="${QUERY_LOCAL:-"$(printf "'%s'" "$FORMAT_LOCAL" | __clean_filter)"}"
|
QUERY_LOCAL="${QUERY_LOCAL:-"$(printf "'%s'" "$FORMAT_LOCAL" | __clean_filter)"}"
|
||||||
|
|
||||||
|
# The following variables store preset strings derived from the theme (see
|
||||||
|
# `src/sh/theme.sh`), and used in the assignment of the default filters.
|
||||||
q_has_seconary="$(printf "$FORMAT_TYPE_HAS_SECONDARY" "" | __clean_filter)"
|
q_has_seconary="$(printf "$FORMAT_TYPE_HAS_SECONDARY" "" | __clean_filter)"
|
||||||
q_album="$(printf "%s" "$FORMAT_TYPE_ALBUM" | __clean_filter)"
|
q_album="$(printf "%s" "$FORMAT_TYPE_ALBUM" | __clean_filter)"
|
||||||
q_ep=$(printf "%s" "$FORMAT_TYPE_EP" | sed "s/${ESC}\[[0-9;]*[mK]//g" | sed "s/ /\\\ /g")
|
q_ep=$(printf "%s" "$FORMAT_TYPE_EP" | sed "s/${ESC}\[[0-9;]*[mK]//g" | sed "s/ /\\\ /g")
|
||||||
@@ -11,6 +18,9 @@ if printf "$RV_FORMAT" | grep -q "<<status>>"; then
|
|||||||
fi
|
fi
|
||||||
export QUERY_LOCAL
|
export QUERY_LOCAL
|
||||||
|
|
||||||
|
# Here starts the list of all filters (grouped per view) that are associated to
|
||||||
|
# the keys `KEYS_FILTER_0` - `KEYS_FILTER_9`. The filters in the `F_1_<view>`
|
||||||
|
# variable are automatically applied whenever the given view is entered.
|
||||||
F_1_VIEW_ARTIST="${F_1_VIEW_ARTIST:-"!'$q_has_seconary'"}"
|
F_1_VIEW_ARTIST="${F_1_VIEW_ARTIST:-"!'$q_has_seconary'"}"
|
||||||
F_2_VIEW_ARTIST="${F_2_VIEW_ARTIST:-"'$q_album'"}"
|
F_2_VIEW_ARTIST="${F_2_VIEW_ARTIST:-"'$q_album'"}"
|
||||||
F_3_VIEW_ARTIST="${F_3_VIEW_ARTIST:-"'$q_ep'"}"
|
F_3_VIEW_ARTIST="${F_3_VIEW_ARTIST:-"'$q_ep'"}"
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
# Print the command that sets the header.
|
# Print the fzf instructions that sets the header
|
||||||
|
#
|
||||||
# @argument $1: view
|
# @argument $1: view
|
||||||
# @argument $2: mbid
|
# @argument $2: mbid
|
||||||
fzf_command_set_header() {
|
fzf_command_set_header() {
|
||||||
@@ -30,93 +31,3 @@ fzf_command_set_header() {
|
|||||||
esac
|
esac
|
||||||
printf "+change-header(%s)" "${header:-"???"}"
|
printf "+change-header(%s)" "${header:-"???"}"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Reload hook that is used after change in query
|
|
||||||
fzf_reload_after_change() {
|
|
||||||
# Wait for async. process to terminate
|
|
||||||
sleep 1
|
|
||||||
while [ -f "$LOCKFILE" ]; do
|
|
||||||
sleep 1
|
|
||||||
done
|
|
||||||
# Show results
|
|
||||||
column -t -s "$(printf '\t')" "$RESULTS" |
|
|
||||||
sed 's| \+\([0-9a-f-]\+\) \+\([0-9a-f-]\+\)$|\t\1\t\2|'
|
|
||||||
}
|
|
||||||
|
|
||||||
# Handle change in query
|
|
||||||
fzf_handle_change() {
|
|
||||||
view="$1"
|
|
||||||
# Kill any running search
|
|
||||||
if [ -f "$PIDFILE" ]; then
|
|
||||||
pid=$(cat "$PIDFILE")
|
|
||||||
rm -f "$PIDFILE"
|
|
||||||
kill -9 "$pid" >/dev/null 2>&1 || true
|
|
||||||
fi
|
|
||||||
# Stop, if no search string is given
|
|
||||||
[ "$FZF_QUERY" ] || exit 0
|
|
||||||
# Store PID of current process
|
|
||||||
echo "$$" >"$PIDFILE"
|
|
||||||
touch "$LOCKFILE"
|
|
||||||
sleep 1
|
|
||||||
if [ "$view" = "$VIEW_SEARCH_ARTIST" ]; then
|
|
||||||
api_mb_search_artist "$FZF_QUERY" |
|
|
||||||
$JQ '.artists[] | [
|
|
||||||
.id,
|
|
||||||
.type,
|
|
||||||
.name,
|
|
||||||
.disambiguation,
|
|
||||||
.["life-span"].begin,
|
|
||||||
.["life-span"].end
|
|
||||||
] | join("\t")' |
|
|
||||||
awk \
|
|
||||||
-F "\t" \
|
|
||||||
-v file_local_artists="${LOCALDATA_ARTISTS:-}" \
|
|
||||||
-v format_person="$AV_PERSON" \
|
|
||||||
-v format_group="$AV_GROUP" \
|
|
||||||
-v format_disambiguation="$AV_DISAMBIGUATION" \
|
|
||||||
-v format_local="$FORMAT_LOCAL" \
|
|
||||||
"$AWK_ARTISTS" >"$RESULTS" ||
|
|
||||||
true
|
|
||||||
else
|
|
||||||
api_mb_search_releasegroup "$FZF_QUERY" |
|
|
||||||
$JQ '."release-groups"[] | [
|
|
||||||
.id,
|
|
||||||
."primary-type",
|
|
||||||
(."secondary-types" // []|join(";")),
|
|
||||||
."first-release-date",
|
|
||||||
.title,
|
|
||||||
(."artist-credit" | map(([.name, .joinphrase]|join(""))) | join(""))
|
|
||||||
] | join("\t")' |
|
|
||||||
awk \
|
|
||||||
-F "\t" \
|
|
||||||
-v file_local_releasegroups="${LOCALDATA_RELEASEGROUPS:-}" \
|
|
||||||
-v format_release="$RGV_RELEASE" \
|
|
||||||
-v format_release_w_artist="$RGV_RELEASE_W_ARTIST" \
|
|
||||||
-v format_year="$RGV_YEAR" \
|
|
||||||
-v format_album="$FORMAT_TYPE_ALBUM" \
|
|
||||||
-v format_single="$FORMAT_TYPE_SINGLE" \
|
|
||||||
-v format_ep="$FORMAT_TYPE_EP" \
|
|
||||||
-v format_broadcast="$FORMAT_TYPE_BROADCAST" \
|
|
||||||
-v format_other="$FORMAT_TYPE_OTHER" \
|
|
||||||
-v format_has_secondary="$FORMAT_TYPE_HAS_SECONDARY" \
|
|
||||||
-v format_secondary="$FORMAT_TYPE_SECONDARY" \
|
|
||||||
-v format_compilation="$FORMAT_TYPE_SECONDARY_COMPILATION" \
|
|
||||||
-v format_soundtrack="$FORMAT_TYPE_SECONDARY_SOUNDTRACK" \
|
|
||||||
-v format_spokenword="$FORMAT_TYPE_SECONDARY_SPOKENWORD" \
|
|
||||||
-v format_interview="$FORMAT_TYPE_SECONDARY_INTERVIEW" \
|
|
||||||
-v format_audiobook="$FORMAT_TYPE_SECONDARY_AUDIOBOOK" \
|
|
||||||
-v format_audiodrama="$FORMAT_TYPE_SECONDARY_AUDIODRAMA" \
|
|
||||||
-v format_live="$FORMAT_TYPE_SECONDARY_LIVE" \
|
|
||||||
-v format_remix="$FORMAT_TYPE_SECONDARY_REMIX" \
|
|
||||||
-v format_djmix="$FORMAT_TYPE_SECONDARY_DJMIX" \
|
|
||||||
-v format_mixtape="$FORMAT_TYPE_SECONDARY_MIXTAPE" \
|
|
||||||
-v format_demo="$FORMAT_TYPE_SECONDARY_DEMO" \
|
|
||||||
-v format_fieldrec="$FORMAT_TYPE_SECONDARY_FIELDREC" \
|
|
||||||
-v format_local="$FORMAT_LOCAL" \
|
|
||||||
"$AWK_RELEASEGROUPS" |
|
|
||||||
cut -d "$(printf '\t')" -f 2- >"$RESULTS" ||
|
|
||||||
true
|
|
||||||
fi
|
|
||||||
# Process ends now: Display and quit
|
|
||||||
rm -f "$LOCKFILE" "$PIDFILE"
|
|
||||||
}
|
|
||||||
|
@@ -1,5 +1,8 @@
|
|||||||
|
# These methods generate lists that are used as input to FZF.
|
||||||
|
|
||||||
# List release groups of given artist
|
# List release groups of given artist
|
||||||
# argument $1: MB artist id
|
#
|
||||||
|
# argument $1: MusicBrainz artist ID
|
||||||
list_releasegroups() {
|
list_releasegroups() {
|
||||||
name=$(mb_artist "$1" | $JQ '.name')
|
name=$(mb_artist "$1" | $JQ '.name')
|
||||||
mb_artist_releasegroups "$1" |
|
mb_artist_releasegroups "$1" |
|
||||||
@@ -47,7 +50,8 @@ list_releasegroups() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# List releases in given relese group
|
# List releases in given relese group
|
||||||
# argument $1: MB release-group id
|
#
|
||||||
|
# argument $1: MusicBrainz release-group ID
|
||||||
list_releases() {
|
list_releases() {
|
||||||
title="$(mb_releasegroup "$1" |
|
title="$(mb_releasegroup "$1" |
|
||||||
$JQ '.title')"
|
$JQ '.title')"
|
||||||
@@ -76,10 +80,10 @@ list_releases() {
|
|||||||
-v release_withdrawn="$FORMAT_STATUS_WITHDRAWN" \
|
-v release_withdrawn="$FORMAT_STATUS_WITHDRAWN" \
|
||||||
-v release_expunged="$FORMAT_STATUS_EXPUNGED" \
|
-v release_expunged="$FORMAT_STATUS_EXPUNGED" \
|
||||||
-v release_cancelled="$FORMAT_STATUS_CANCELLED" \
|
-v release_cancelled="$FORMAT_STATUS_CANCELLED" \
|
||||||
-v release_format="$RV_FORMAT" \
|
-v format_release="$RV_FORMAT" \
|
||||||
-v release_format_title_artist="$RV_TITLE_ARTIST" \
|
-v format_release_title_artist="$RV_TITLE_ARTIST" \
|
||||||
-v release_format_title="$RV_TITLE" \
|
-v format_release_title="$RV_TITLE" \
|
||||||
-v release_format_artist="$RV_ARTIST" \
|
-v format_release_artist="$RV_ARTIST" \
|
||||||
-v rg_artist="$artist" \
|
-v rg_artist="$artist" \
|
||||||
-v rg_title="$title" \
|
-v rg_title="$title" \
|
||||||
-v rgid="$1" \
|
-v rgid="$1" \
|
||||||
@@ -92,7 +96,8 @@ list_releases() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# List recordings of given release
|
# List recordings of given release
|
||||||
# argument $1: MB release id
|
#
|
||||||
|
# argument $1: MusicBrainz release ID
|
||||||
list_recordings() {
|
list_recordings() {
|
||||||
deco="$(grep "$1" "$LOCALDATA_RELEASES" | cut -d "$(printf '\t')" -f 2)"
|
deco="$(grep "$1" "$LOCALDATA_RELEASES" | cut -d "$(printf '\t')" -f 2)"
|
||||||
if [ "$deco" ]; then
|
if [ "$deco" ]; then
|
||||||
@@ -131,22 +136,19 @@ list_recordings() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# List artists (local)
|
# List artists available locally
|
||||||
list_local_artists() {
|
list_local_artists() {
|
||||||
cat "$LOCALDATA_ARTISTS_VIEW" 2>/dev/null
|
cat "$LOCALDATA_ARTISTS_VIEW" 2>/dev/null
|
||||||
}
|
}
|
||||||
|
|
||||||
# List release groups (local)
|
# List release groups vailable locally
|
||||||
list_local_releasegroups() {
|
list_local_releasegroups() {
|
||||||
cat "$LOCALDATA_RELEASEGROUPS_VIEW" 2>/dev/null
|
cat "$LOCALDATA_RELEASEGROUPS_VIEW" 2>/dev/null
|
||||||
}
|
}
|
||||||
|
|
||||||
# List releases (local)
|
# List artist from input json data
|
||||||
list_local_releases() {
|
#
|
||||||
cat "$LOCALDATA_RELEASES_VIEW" 2>/dev/null
|
# The input is read from stdin
|
||||||
}
|
|
||||||
|
|
||||||
# Generate artist list from JSON
|
|
||||||
list_artists_from_json() {
|
list_artists_from_json() {
|
||||||
cat |
|
cat |
|
||||||
$JQ 'map([.artist.id, .artist.type, .name] | join("\t")) | join("\n")' |
|
$JQ 'map([.artist.id, .artist.type, .name] | join("\t")) | join("\n")' |
|
||||||
@@ -161,7 +163,7 @@ list_artists_from_json() {
|
|||||||
column -t -s "$(printf '\t')" -l 2
|
column -t -s "$(printf '\t')" -l 2
|
||||||
}
|
}
|
||||||
|
|
||||||
# Generate playlist view
|
# Print playlist currently loaded
|
||||||
list_playlist() {
|
list_playlist() {
|
||||||
count=$(mpv_playlist_count)
|
count=$(mpv_playlist_count)
|
||||||
[ "$count" -eq 0 ] && return 0
|
[ "$count" -eq 0 ] && return 0
|
||||||
|
@@ -1,3 +1,8 @@
|
|||||||
|
# Database functionality to support local music.
|
||||||
|
#
|
||||||
|
# All local data is stored in the directory `LOCALDATADIR`. In the future, we
|
||||||
|
# will also use the methods here, and modifications thereof, to support
|
||||||
|
# MusicBainz collections.
|
||||||
if [ ! "${LOCAL_LOADED:-}" ]; then
|
if [ ! "${LOCAL_LOADED:-}" ]; then
|
||||||
LOCALDATADIR="$HOME/.cache/$APP_NAME/local"
|
LOCALDATADIR="$HOME/.cache/$APP_NAME/local"
|
||||||
LOCALDATA_ARTISTS="$LOCALDATADIR/artists"
|
LOCALDATA_ARTISTS="$LOCALDATADIR/artists"
|
||||||
@@ -19,7 +24,13 @@ if [ ! "${LOCAL_LOADED:-}" ]; then
|
|||||||
export LOCAL_LOADED=1
|
export LOCAL_LOADED=1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
gettags() {
|
# Retrieve tags as json object from music file
|
||||||
|
#
|
||||||
|
# @argument $1: path to music file
|
||||||
|
#
|
||||||
|
# The tags retrieved are the MusicBrainz release ID and the MusicBrainz track
|
||||||
|
# ID
|
||||||
|
__gettags() {
|
||||||
ffprobe -v error -show_entries format_tags -print_format json "$1" |
|
ffprobe -v error -show_entries format_tags -print_format json "$1" |
|
||||||
$JQ '.format.tags | {
|
$JQ '.format.tags | {
|
||||||
trackid: (."MusicBrainz Release Track Id" // ."MUSICBRAINZ_RELEASETRACKID" // ."MusicBrainz/Release Track Id" // ""),
|
trackid: (."MusicBrainz Release Track Id" // ."MUSICBRAINZ_RELEASETRACKID" // ."MusicBrainz/Release Track Id" // ""),
|
||||||
@@ -27,9 +38,14 @@ gettags() {
|
|||||||
}'
|
}'
|
||||||
}
|
}
|
||||||
|
|
||||||
# Read music files in specified directory and create json file that points to
|
# Decorate locally available music
|
||||||
# all relevant MusicBrainz IDs.
|
#
|
||||||
# @input $1: Path to directory with album
|
# @input $1: Path to directory with album
|
||||||
|
#
|
||||||
|
# This methods reads the music files in the specified directory and writes a
|
||||||
|
# json file that points to all relevant MusicBrainz IDs. If the directory
|
||||||
|
# contains untagged files, or files of different releases, then the decoration
|
||||||
|
# process will fail, and an error is printed.
|
||||||
decorate() {
|
decorate() {
|
||||||
if [ -f "$1/$DECORATION_FILENAME" ]; then
|
if [ -f "$1/$DECORATION_FILENAME" ]; then
|
||||||
info "Directory $1 has already been decorated (skipping)"
|
info "Directory $1 has already been decorated (skipping)"
|
||||||
@@ -39,7 +55,7 @@ decorate() {
|
|||||||
tmpf=$(mktemp)
|
tmpf=$(mktemp)
|
||||||
(cd "$1" && find . -type f -iname '*.mp3' -o -iname '*.mp4' -o -iname '*.flac' -o -iname '*.m4a') >"$tmpf"
|
(cd "$1" && find . -type f -iname '*.mp3' -o -iname '*.mp4' -o -iname '*.flac' -o -iname '*.m4a') >"$tmpf"
|
||||||
while IFS= read -r f; do
|
while IFS= read -r f; do
|
||||||
mbid=$(gettags "$1/$f")
|
mbid=$(__gettags "$1/$f")
|
||||||
rid=$(echo "$mbid" | $JQ '.releaseid')
|
rid=$(echo "$mbid" | $JQ '.releaseid')
|
||||||
tid=$(echo "$mbid" | $JQ '.trackid')
|
tid=$(echo "$mbid" | $JQ '.trackid')
|
||||||
if [ ! "$rid" ] || [ ! "$tid" ]; then
|
if [ ! "$rid" ] || [ ! "$tid" ]; then
|
||||||
@@ -105,7 +121,13 @@ __batch_load_missing() {
|
|||||||
rm -f "$tmpf"
|
rm -f "$tmpf"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Precompute views
|
# Precompute lists
|
||||||
|
#
|
||||||
|
# The main views (VIEW_ARTIST and TYPE_RELEASEGROUP) for locally available
|
||||||
|
# music are theme dependent. These views are generated from the lists that are
|
||||||
|
# produced with the present method. It contains all essential data, but in a
|
||||||
|
# theme-independent fashion. The lists are stored in the files
|
||||||
|
# `LOCALDATA_ARTISTS_LIST` and `LOCALDATA_RELEASEGROUPS_LIST`.
|
||||||
__precompute_lists() {
|
__precompute_lists() {
|
||||||
cache_get_file_batch "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS" | xargs \
|
cache_get_file_batch "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS" | xargs \
|
||||||
$JQ '[
|
$JQ '[
|
||||||
@@ -125,23 +147,13 @@ __precompute_lists() {
|
|||||||
.title,
|
.title,
|
||||||
(."artist-credit" | map(([.name, .joinphrase]|join(""))) | join(""))
|
(."artist-credit" | map(([.name, .joinphrase]|join(""))) | join(""))
|
||||||
] | join("\t")' >"$LOCALDATA_RELEASEGROUPS_LIST"
|
] | join("\t")' >"$LOCALDATA_RELEASEGROUPS_LIST"
|
||||||
# cache_get_file_batch "$TYPE_RELEASE" <"$LOCALDATA_RELEASES" | xargs \
|
|
||||||
# $JQ '[
|
|
||||||
# .id,
|
|
||||||
# .status,
|
|
||||||
# .date,
|
|
||||||
# ."cover-art-archive".count,
|
|
||||||
# (."label-info" | map(.label.name) | unique | join(", ")),
|
|
||||||
# (.media | map(."track-count") | add),
|
|
||||||
# (.media | map(.format) | unique | join(", ")),
|
|
||||||
# .country,
|
|
||||||
# .title,
|
|
||||||
# (."artist-credit" | map(([.name, .joinphrase]|join(""))) | join(""))
|
|
||||||
# ] | join("\t")' >"$LOCALDATA_RELEASES_LIST" &
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# Precompute views
|
# Precompute views
|
||||||
# TODO: The sed opperations take too long, improve!
|
#
|
||||||
|
# This method injects the theme elements to the lists from `precompute_lists`.
|
||||||
|
# The resulting views are stored in the files `LOCALDATA_ARTISTS_VIEW` and
|
||||||
|
# `LOCALDATA_RELEASEGROUPS_VIEW`.
|
||||||
precompute_views() {
|
precompute_views() {
|
||||||
awk \
|
awk \
|
||||||
-F "\t" \
|
-F "\t" \
|
||||||
@@ -193,7 +205,9 @@ precompute_views() {
|
|||||||
# argument $1: path to decorated music files
|
# argument $1: path to decorated music files
|
||||||
#
|
#
|
||||||
# This method parses all decorations and generates a line-by-line database of
|
# This method parses all decorations and generates a line-by-line database of
|
||||||
# locally available artists, releases, and release groups.
|
# locally available artists, releases, and release groups. This data is stored
|
||||||
|
# in the files `LOCALDATA_ARTISTS`, `LOCALDATA_RELEASES`, and
|
||||||
|
# `LOCALDATA_RELEASEGROUPS`.
|
||||||
reloaddb() {
|
reloaddb() {
|
||||||
rm -rf "$LOCALDATADIR"
|
rm -rf "$LOCALDATADIR"
|
||||||
mkdir -p "$LOCALDATADIR"
|
mkdir -p "$LOCALDATADIR"
|
||||||
@@ -225,16 +239,24 @@ reloaddb() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# Check if necessary cache files are present or not
|
# Check if necessary cache files are present or not
|
||||||
|
#
|
||||||
|
# This method returns a non-zero value if some cached file is required to exist
|
||||||
|
# for the computation of the lists (and views). This does not include the
|
||||||
|
# derivation of the MusicBrainz artist IDs and MusicBrainz release-group IDs
|
||||||
|
# from the MusicBrainz releases (see the `reloaddb` method above).
|
||||||
local_files_present() {
|
local_files_present() {
|
||||||
cache_get_file_batch "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS" | xargs ls >/dev/null 2>&1 || return 1
|
cache_get_file_batch "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS" | xargs ls >/dev/null 2>&1 || return 1
|
||||||
cache_get_file_batch "$TYPE_RELEASEGROUP" <"$LOCALDATA_RELEASEGROUPS" | xargs ls >/dev/null 2>&1 || return 1
|
cache_get_file_batch "$TYPE_RELEASEGROUP" <"$LOCALDATA_RELEASEGROUPS" | xargs ls >/dev/null 2>&1 || return 1
|
||||||
cut -d "$(printf '\t')" -f 1 "$LOCALDATA_RELEASES" | cache_get_file_batch "$TYPE_RELEASEGROUP" | xargs ls >/dev/null 2>&1 || return 1
|
#cut -d "$(printf '\t')" -f 1 "$LOCALDATA_RELEASES" | cache_get_file_batch "$TYPE_RELEASE" | xargs ls >/dev/null 2>&1 || return 1
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
# Load missing files
|
# Load missing files
|
||||||
|
#
|
||||||
|
# If missing files were detected with `local_files_present`, then these missing
|
||||||
|
# files may be cached using the present method.
|
||||||
load_missing_files() {
|
load_missing_files() {
|
||||||
__batch_load_missing "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS"
|
__batch_load_missing "$TYPE_ARTIST" <"$LOCALDATA_ARTISTS"
|
||||||
__batch_load_missing "$TYPE_RELEASEGROUP" <"$LOCALDATA_RELEASEGROUPS"
|
__batch_load_missing "$TYPE_RELEASEGROUP" <"$LOCALDATA_RELEASEGROUPS"
|
||||||
cut -d "$(printf '\t')" -f 1 "$LOCALDATA_RELEASES" | __batch_load_missing "$TYPE_RELEASE"
|
#cut -d "$(printf '\t')" -f 1 "$LOCALDATA_RELEASES" | __batch_load_missing "$TYPE_RELEASE"
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +1,7 @@
|
|||||||
# Logging methods
|
# Logging methods
|
||||||
|
#
|
||||||
|
# The default log file is `LOGFILE`. In the future, this file may become
|
||||||
|
# configurable.
|
||||||
if [ ! "${LOG_LOADED:-}" ]; then
|
if [ ! "${LOG_LOADED:-}" ]; then
|
||||||
ERR="\033[38;5;196m"
|
ERR="\033[38;5;196m"
|
||||||
INFO="\033[38;5;75m"
|
INFO="\033[38;5;75m"
|
||||||
@@ -11,10 +14,14 @@ if [ ! "${LOG_LOADED:-}" ]; then
|
|||||||
export LOG_LOADED=1
|
export LOG_LOADED=1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Print an error message to stderr and log it incuding the time stamp and PID
|
||||||
|
# to the log file.
|
||||||
err() {
|
err() {
|
||||||
echo "$(date) [$$]>${ERR}ERROR:${OFF} ${1:-}" | tee -a "$LOGFILE" | cut -d ">" -f 2- >/dev/stderr
|
echo "$(date) [$$]>${ERR}ERROR:${OFF} ${1:-}" | tee -a "$LOGFILE" | cut -d ">" -f 2- >/dev/stderr
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Print information to stderr and log it incuding the time stamp and PID to the
|
||||||
|
# log file.
|
||||||
info() {
|
info() {
|
||||||
echo "$(date) [$$]>${INFO}Info:${OFF} ${1:-}" | tee -a "$LOGFILE" | cut -d ">" -f 2- >/dev/stderr
|
echo "$(date) [$$]>${INFO}Info:${OFF} ${1:-}" | tee -a "$LOGFILE" | cut -d ">" -f 2- >/dev/stderr
|
||||||
}
|
}
|
||||||
|
167
src/sh/mb.sh
167
src/sh/mb.sh
@@ -1,7 +1,13 @@
|
|||||||
# The only IDs uses here are MusicBrainz IDs
|
# This files provides a high-level access to the MusicBrainz databse. The only
|
||||||
|
# IDs used here are MusicBrainz IDs
|
||||||
|
|
||||||
# Helper methods to retrieve from cache, if it exists, and otherwise populate
|
# The following methods are local methods that combines the MusicBrainz API
|
||||||
# cache and retrieve
|
# with the caching methods.
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz data for artist from cache (if it exists), and otherwise
|
||||||
|
# download it using the MusicBrainz API.
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
__mb_artist_cache_or_fetch() {
|
__mb_artist_cache_or_fetch() {
|
||||||
if ! cache_get_artist "$1"; then
|
if ! cache_get_artist "$1"; then
|
||||||
api_mb_artist "$1" | cache_put_artist "$1"
|
api_mb_artist "$1" | cache_put_artist "$1"
|
||||||
@@ -9,6 +15,10 @@ __mb_artist_cache_or_fetch() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz data for release group from cache (if it exists), and
|
||||||
|
# otherwise download it using the MusicBrainz API.
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
__mb_releasegroup_cache_or_fetch() {
|
__mb_releasegroup_cache_or_fetch() {
|
||||||
if ! cache_get_releasegroup "$1"; then
|
if ! cache_get_releasegroup "$1"; then
|
||||||
api_mb_releasegroup "$1" | cache_put_releasegroup "$1"
|
api_mb_releasegroup "$1" | cache_put_releasegroup "$1"
|
||||||
@@ -16,6 +26,10 @@ __mb_releasegroup_cache_or_fetch() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz data for release from cache (if it exists), and
|
||||||
|
# otherwise download it using the MusicBrainz API.
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
__mb_release_cache_or_fetch() {
|
__mb_release_cache_or_fetch() {
|
||||||
if ! cache_get_release "$1"; then
|
if ! cache_get_release "$1"; then
|
||||||
api_mb_release "$1" | cache_put_release "$1"
|
api_mb_release "$1" | cache_put_release "$1"
|
||||||
@@ -23,6 +37,10 @@ __mb_release_cache_or_fetch() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz data for release groups of given artist from cache (if
|
||||||
|
# it exists), and otherwise download it using the MusicBrainz API.
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
__mb_artist_cache_or_fetch_releasegroups() {
|
__mb_artist_cache_or_fetch_releasegroups() {
|
||||||
if ! cache_get_artist_releasegroups "$1"; then
|
if ! cache_get_artist_releasegroups "$1"; then
|
||||||
api_mb_browse_artist_releasegroups "$1" | cache_put_artist_releasegroups "$1"
|
api_mb_browse_artist_releasegroups "$1" | cache_put_artist_releasegroups "$1"
|
||||||
@@ -39,6 +57,10 @@ __mb_artist_cache_or_fetch_releasegroups() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz data for releases of given release group from cache (if
|
||||||
|
# it exists), and otherwise download it using the MusicBrainz API.
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
__mb_releasegroup_cache_or_fetch_releases() {
|
__mb_releasegroup_cache_or_fetch_releases() {
|
||||||
if ! cache_get_releasegroup_releases "$1"; then
|
if ! cache_get_releasegroup_releases "$1"; then
|
||||||
api_mb_browse_releasegroup_releases "$1" | cache_put_releasegroup_releases "$1"
|
api_mb_browse_releasegroup_releases "$1" | cache_put_releasegroup_releases "$1"
|
||||||
@@ -55,14 +77,18 @@ __mb_releasegroup_cache_or_fetch_releases() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get MusicBrainz json for artist
|
# The following methods provide the external interface
|
||||||
# @argument $1: MusicBrainz Artist ID
|
|
||||||
|
# Retrieve MusicBrainz data for artist
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
mb_artist() {
|
mb_artist() {
|
||||||
__mb_artist_cache_or_fetch "$1"
|
__mb_artist_cache_or_fetch "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get Wikidata json for artist
|
# Retrieve Wikidata data for artist
|
||||||
# @argument $1: MusicBrainz Artist ID
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
mb_artist_wikidata() {
|
mb_artist_wikidata() {
|
||||||
if ! cache_get_artist_wikidata "$1"; then
|
if ! cache_get_artist_wikidata "$1"; then
|
||||||
wikidataid=$(mb_artist "$1" |
|
wikidataid=$(mb_artist "$1" |
|
||||||
@@ -76,8 +102,9 @@ mb_artist_wikidata() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get Wikipedia (English) summary json for artist
|
# Retrieve Wikipedia (English) summary json for artist
|
||||||
# @argument $1: MusicBrainz Artist ID
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
mb_artist_enwikipedia() {
|
mb_artist_enwikipedia() {
|
||||||
if ! cache_get_artist_enwikipedia "$1"; then
|
if ! cache_get_artist_enwikipedia "$1"; then
|
||||||
# To fetch the wikipedia data, we need the wikipedia URL
|
# To fetch the wikipedia data, we need the wikipedia URL
|
||||||
@@ -97,8 +124,9 @@ mb_artist_enwikipedia() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get Discogs json for artist
|
# Retrieve Discogs json for artist
|
||||||
# @argument $1: MusicBrainz Artist ID
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
mb_artist_discogs() {
|
mb_artist_discogs() {
|
||||||
if ! cache_get_artist_discogs "$1"; then
|
if ! cache_get_artist_discogs "$1"; then
|
||||||
discogsid=$(mb_artist "$1" |
|
discogsid=$(mb_artist "$1" |
|
||||||
@@ -112,22 +140,131 @@ mb_artist_discogs() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get release-groups json for artist
|
# Retrieve release groups for artist
|
||||||
# @argument $1: MusicBrainz Artist ID
|
#
|
||||||
|
# @argument $1: MusicBrainz artist ID
|
||||||
mb_artist_releasegroups() {
|
mb_artist_releasegroups() {
|
||||||
__mb_artist_cache_or_fetch_releasegroups "$1"
|
__mb_artist_cache_or_fetch_releasegroups "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Get MusicBrainz json for release group
|
# Retrieve MusicBrainz release group
|
||||||
# @argument $1: MusicBrainz Release-Group ID
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
mb_releasegroup() {
|
mb_releasegroup() {
|
||||||
__mb_releasegroup_cache_or_fetch "$1"
|
__mb_releasegroup_cache_or_fetch "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz releases of release group
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release-group ID
|
||||||
mb_releasegroup_releases() {
|
mb_releasegroup_releases() {
|
||||||
__mb_releasegroup_cache_or_fetch_releases "$1"
|
__mb_releasegroup_cache_or_fetch_releases "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Retrieve MusicBrainz release
|
||||||
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
mb_release() {
|
mb_release() {
|
||||||
__mb_release_cache_or_fetch "$1"
|
__mb_release_cache_or_fetch "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Reload hook that is used after a change in the query (when searching
|
||||||
|
# MusicBrainz).
|
||||||
|
#
|
||||||
|
# This method waits for the search to complete, then it parses the search
|
||||||
|
# results and prints them.
|
||||||
|
mb_results_async() {
|
||||||
|
# Wait for async. process to terminate
|
||||||
|
sleep 1
|
||||||
|
while [ -f "$LOCKFILE" ]; do
|
||||||
|
sleep 1
|
||||||
|
done
|
||||||
|
# Show results
|
||||||
|
column -t -s "$(printf '\t')" "$RESULTS" |
|
||||||
|
sed 's| \+\([0-9a-f-]\+\) \+\([0-9a-f-]\+\)$|\t\1\t\2|'
|
||||||
|
}
|
||||||
|
|
||||||
|
# Initiate search on MusicBrainz
|
||||||
|
#
|
||||||
|
# @argument $1: view
|
||||||
|
#
|
||||||
|
# This methods initiates an asynchronous search for both views
|
||||||
|
# (VIEW_SEARCH_ARTIST and VIEW_SEARCH_ALBUM). If a running query is detected,
|
||||||
|
# that one is killed first. The search results are then stored and become
|
||||||
|
# retrievable using `mb_results_async`.
|
||||||
|
mb_search_async() {
|
||||||
|
view="$1"
|
||||||
|
# Kill any running search
|
||||||
|
if [ -f "$PIDFILE" ]; then
|
||||||
|
pid=$(cat "$PIDFILE")
|
||||||
|
rm -f "$PIDFILE"
|
||||||
|
kill -9 "$pid" >/dev/null 2>&1 || true
|
||||||
|
fi
|
||||||
|
# Stop, if no search string is given
|
||||||
|
[ "$FZF_QUERY" ] || exit 0
|
||||||
|
# Store PID of current process
|
||||||
|
echo "$$" >"$PIDFILE"
|
||||||
|
touch "$LOCKFILE"
|
||||||
|
sleep 1
|
||||||
|
if [ "$view" = "$VIEW_SEARCH_ARTIST" ]; then
|
||||||
|
api_mb_search_artist "$FZF_QUERY" |
|
||||||
|
$JQ '.artists[] | [
|
||||||
|
.id,
|
||||||
|
.type,
|
||||||
|
.name,
|
||||||
|
.disambiguation,
|
||||||
|
.["life-span"].begin,
|
||||||
|
.["life-span"].end
|
||||||
|
] | join("\t")' |
|
||||||
|
awk \
|
||||||
|
-F "\t" \
|
||||||
|
-v file_local_artists="${LOCALDATA_ARTISTS:-}" \
|
||||||
|
-v format_person="$AV_PERSON" \
|
||||||
|
-v format_group="$AV_GROUP" \
|
||||||
|
-v format_disambiguation="$AV_DISAMBIGUATION" \
|
||||||
|
-v format_local="$FORMAT_LOCAL" \
|
||||||
|
"$AWK_ARTISTS" >"$RESULTS" ||
|
||||||
|
true
|
||||||
|
else
|
||||||
|
api_mb_search_releasegroup "$FZF_QUERY" |
|
||||||
|
$JQ '."release-groups"[] | [
|
||||||
|
.id,
|
||||||
|
."primary-type",
|
||||||
|
(."secondary-types" // []|join(";")),
|
||||||
|
."first-release-date",
|
||||||
|
.title,
|
||||||
|
(."artist-credit" | map(([.name, .joinphrase]|join(""))) | join(""))
|
||||||
|
] | join("\t")' |
|
||||||
|
awk \
|
||||||
|
-F "\t" \
|
||||||
|
-v file_local_releasegroups="${LOCALDATA_RELEASEGROUPS:-}" \
|
||||||
|
-v format_release="$RGV_RELEASE" \
|
||||||
|
-v format_release_w_artist="$RGV_RELEASE_W_ARTIST" \
|
||||||
|
-v format_year="$RGV_YEAR" \
|
||||||
|
-v format_album="$FORMAT_TYPE_ALBUM" \
|
||||||
|
-v format_single="$FORMAT_TYPE_SINGLE" \
|
||||||
|
-v format_ep="$FORMAT_TYPE_EP" \
|
||||||
|
-v format_broadcast="$FORMAT_TYPE_BROADCAST" \
|
||||||
|
-v format_other="$FORMAT_TYPE_OTHER" \
|
||||||
|
-v format_has_secondary="$FORMAT_TYPE_HAS_SECONDARY" \
|
||||||
|
-v format_secondary="$FORMAT_TYPE_SECONDARY" \
|
||||||
|
-v format_compilation="$FORMAT_TYPE_SECONDARY_COMPILATION" \
|
||||||
|
-v format_soundtrack="$FORMAT_TYPE_SECONDARY_SOUNDTRACK" \
|
||||||
|
-v format_spokenword="$FORMAT_TYPE_SECONDARY_SPOKENWORD" \
|
||||||
|
-v format_interview="$FORMAT_TYPE_SECONDARY_INTERVIEW" \
|
||||||
|
-v format_audiobook="$FORMAT_TYPE_SECONDARY_AUDIOBOOK" \
|
||||||
|
-v format_audiodrama="$FORMAT_TYPE_SECONDARY_AUDIODRAMA" \
|
||||||
|
-v format_live="$FORMAT_TYPE_SECONDARY_LIVE" \
|
||||||
|
-v format_remix="$FORMAT_TYPE_SECONDARY_REMIX" \
|
||||||
|
-v format_djmix="$FORMAT_TYPE_SECONDARY_DJMIX" \
|
||||||
|
-v format_mixtape="$FORMAT_TYPE_SECONDARY_MIXTAPE" \
|
||||||
|
-v format_demo="$FORMAT_TYPE_SECONDARY_DEMO" \
|
||||||
|
-v format_fieldrec="$FORMAT_TYPE_SECONDARY_FIELDREC" \
|
||||||
|
-v format_local="$FORMAT_LOCAL" \
|
||||||
|
"$AWK_RELEASEGROUPS" |
|
||||||
|
cut -d "$(printf '\t')" -f 2- >"$RESULTS" ||
|
||||||
|
true
|
||||||
|
fi
|
||||||
|
# Process ends now: Display and quit
|
||||||
|
rm -f "$LOCKFILE" "$PIDFILE"
|
||||||
|
}
|
||||||
|
@@ -1,61 +1,104 @@
|
|||||||
|
# Interface to the mpv music player. This interface communicates to an mpv
|
||||||
|
# instance through the socket `MPV_SOCKET`.
|
||||||
|
|
||||||
|
# Internal helper method to send a command without arguments to mpv
|
||||||
|
#
|
||||||
|
# @argument $1: command
|
||||||
__mpv_command() {
|
__mpv_command() {
|
||||||
printf "{ \"command\": [\"%s\"] }\n" "$1" | $SOCAT - "$MPV_SOCKET"
|
printf "{ \"command\": [\"%s\"] }\n" "$1" | $SOCAT - "$MPV_SOCKET"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Internal helper method to send a command with a single argument to mpv
|
||||||
|
#
|
||||||
|
# @argument $1: command
|
||||||
|
# @argument $2: argument
|
||||||
__mpv_command_with_arg() {
|
__mpv_command_with_arg() {
|
||||||
printf "{ \"command\": [\"%s\", \"%s\"] }\n" "$1" "$2" | $SOCAT - "$MPV_SOCKET"
|
printf "{ \"command\": [\"%s\", \"%s\"] }\n" "$1" "$2" | $SOCAT - "$MPV_SOCKET"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Internal helper method to send a command with two arguments to mpv
|
||||||
|
#
|
||||||
|
# @argument $1: command
|
||||||
|
# @argument $2: argument 1
|
||||||
|
# @argument $3: argument 2
|
||||||
__mpv_command_with_args2() {
|
__mpv_command_with_args2() {
|
||||||
printf "{ \"command\": [\"%s\", \"%s\", \"%s\"] }\n" "$1" "$2" "$3" | $SOCAT - "$MPV_SOCKET"
|
printf "{ \"command\": [\"%s\", \"%s\", \"%s\"] }\n" "$1" "$2" "$3" | $SOCAT - "$MPV_SOCKET"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Internal helper method to resolve mpv variables
|
||||||
|
#
|
||||||
|
# @argument $1: mpv expression
|
||||||
__mpv_get() {
|
__mpv_get() {
|
||||||
__mpv_command_with_arg "expand-text" "$1" | $JQ '.data'
|
__mpv_command_with_arg "expand-text" "$1" | $JQ '.data'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Get the total number of tracks in the playlist
|
||||||
mpv_playlist_count() {
|
mpv_playlist_count() {
|
||||||
__mpv_get '${playlist-count}'
|
__mpv_get '${playlist-count}'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Get the position of the current track in the playlist (0 based)
|
||||||
mpv_playlist_position() {
|
mpv_playlist_position() {
|
||||||
__mpv_get '${playlist-pos}'
|
__mpv_get '${playlist-pos}'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Move track on playlist
|
||||||
|
#
|
||||||
|
# @argument $1: track index 1
|
||||||
|
# @argument $2: track index 2
|
||||||
|
#
|
||||||
|
# Moves the track at the first index to the position of the track of the second
|
||||||
|
# index. Also here, indices are 0 based.
|
||||||
mpv_playlist_move() {
|
mpv_playlist_move() {
|
||||||
__mpv_command_with_args2 "playlist-move" "$1" "$2" >>/tmp/foo
|
__mpv_command_with_args2 "playlist-move" "$1" "$2" >>/tmp/foo
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Remove all tracks from the playlist
|
||||||
mpv_playlist_clear() {
|
mpv_playlist_clear() {
|
||||||
__mpv_command "playlist-clear"
|
__mpv_command "playlist-clear"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Randomly shuffle the order of the tracks in the playlist
|
||||||
mpv_playlist_shuffle() {
|
mpv_playlist_shuffle() {
|
||||||
__mpv_command "playlist-shuffle"
|
__mpv_command "playlist-shuffle"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Revert a previously shuffle command
|
||||||
|
#
|
||||||
|
# This method works only for a first shuffle.
|
||||||
mpv_playlist_unshuffle() {
|
mpv_playlist_unshuffle() {
|
||||||
__mpv_command "playlist-unshuffle"
|
__mpv_command "playlist-unshuffle"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Quit the mpv instance bound to the socket `MPV_SOCKET`
|
||||||
mpv_quit() {
|
mpv_quit() {
|
||||||
__mpv_command "quit"
|
__mpv_command "quit"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Start an mpv instance and bind it to the socket `MPV_SOCKET`
|
||||||
mpv_start() {
|
mpv_start() {
|
||||||
MPV_SOCKET="$(mktemp --suffix=.sock)"
|
MPV_SOCKET="$(mktemp --suffix=.sock)"
|
||||||
trap 'mpv_quit >/dev/null; rm -f "$MPV_SOCKET"' EXIT INT
|
trap 'mpv_quit >/dev/null; rm -f "$MPV_SOCKET"' EXIT INT
|
||||||
$MPV --no-config --no-terminal --input-ipc-server="$MPV_SOCKET" --idle --no-osc --no-input-default-bindings &
|
$MPV --no-config --no-terminal --input-ipc-server="$MPV_SOCKET" --idle --no-osc --no-input-default-bindings &
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Play the track at the specified index in the playlist
|
||||||
|
#
|
||||||
|
# @argument $1: index (0 based)
|
||||||
mpv_play_index() {
|
mpv_play_index() {
|
||||||
__mpv_command_with_arg "playlist-play-index" "$1"
|
__mpv_command_with_arg "playlist-play-index" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Remove the track at the specified index from the playlist
|
||||||
|
#
|
||||||
|
# @argument $1: index (0 based)
|
||||||
mpv_rm_index() {
|
mpv_rm_index() {
|
||||||
__mpv_command_with_arg "playlist-remove" "$1"
|
__mpv_command_with_arg "playlist-remove" "$1"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Load the playlist with the specified list, and start playing
|
||||||
|
#
|
||||||
|
# This method reads from stdin a playlist file, e.g., a .m3u file.
|
||||||
mpv_play_list() {
|
mpv_play_list() {
|
||||||
t=$(mktemp)
|
t=$(mktemp)
|
||||||
cat >"$t"
|
cat >"$t"
|
||||||
@@ -63,6 +106,9 @@ mpv_play_list() {
|
|||||||
rm -f "$t"
|
rm -f "$t"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Append the playlist with the specified list, and start playing
|
||||||
|
#
|
||||||
|
# This method reads from stdin a playlist file, e.g., a .m3u file.
|
||||||
mpv_queue_list() {
|
mpv_queue_list() {
|
||||||
t=$(mktemp)
|
t=$(mktemp)
|
||||||
cat >"$t"
|
cat >"$t"
|
||||||
@@ -70,6 +116,10 @@ mpv_queue_list() {
|
|||||||
rm -f "$t"
|
rm -f "$t"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Insert the playlist with the specified list as the next item, and start
|
||||||
|
# playing
|
||||||
|
#
|
||||||
|
# This method reads from stdin a playlist file, e.g., a .m3u file.
|
||||||
mpv_queue_next_list() {
|
mpv_queue_next_list() {
|
||||||
t=$(mktemp)
|
t=$(mktemp)
|
||||||
cat >"$t"
|
cat >"$t"
|
||||||
@@ -86,22 +136,27 @@ mpv_queue_next_list() {
|
|||||||
done
|
done
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Play next track on playlist
|
||||||
mpv_next() {
|
mpv_next() {
|
||||||
__mpv_command "playlist-next"
|
__mpv_command "playlist-next"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Play previous track on playlist
|
||||||
mpv_prev() {
|
mpv_prev() {
|
||||||
__mpv_command "playlist-prev"
|
__mpv_command "playlist-prev"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Seek forward by 10 seconds
|
||||||
mpv_seek_forward() {
|
mpv_seek_forward() {
|
||||||
__mpv_command_with_arg "seek" "10"
|
__mpv_command_with_arg "seek" "10"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Seek backward by 10 seconds
|
||||||
mpv_seek_backward() {
|
mpv_seek_backward() {
|
||||||
__mpv_command_with_arg "seek" "-10"
|
__mpv_command_with_arg "seek" "-10"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Pause if mpv plays, and play if it is paused
|
||||||
mpv_toggle_pause() {
|
mpv_toggle_pause() {
|
||||||
__mpv_command_with_arg "cycle" "pause"
|
__mpv_command_with_arg "cycle" "pause"
|
||||||
}
|
}
|
||||||
|
@@ -21,6 +21,7 @@ if [ ! "${PLAYBACK_LOADED:-}" ]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
# Obtain playback command from key press
|
# Obtain playback command from key press
|
||||||
|
#
|
||||||
# @argument $1: key
|
# @argument $1: key
|
||||||
__playback_cmd_from_key() {
|
__playback_cmd_from_key() {
|
||||||
key=$1
|
key=$1
|
||||||
@@ -43,9 +44,10 @@ __playback_cmd_from_key() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# Generate playlist from MB release ID and path to decoration
|
# Generate playlist from MB release ID and path to decoration
|
||||||
# @argument $1: MusicBrainz Release ID
|
#
|
||||||
|
# @argument $1: MusicBrainz release ID
|
||||||
# @argument $2: Path to decoration file
|
# @argument $2: Path to decoration file
|
||||||
# @argument $3: MusicBrainz Track ID to select (optional)
|
# @argument $3: MusicBrainz track ID to select (optional)
|
||||||
__generate_playlist() {
|
__generate_playlist() {
|
||||||
printf "#EXTM3U\n"
|
printf "#EXTM3U\n"
|
||||||
dir="$(dirname "$2")"
|
dir="$(dirname "$2")"
|
||||||
|
@@ -1,3 +1,7 @@
|
|||||||
|
# Playlist manipulation
|
||||||
|
#
|
||||||
|
# This files provides an interface to manipulate the playlist. The available
|
||||||
|
# commands are defined in the following variables.
|
||||||
if [ ! "${PLAYLIST_LOADED:-}" ]; then
|
if [ ! "${PLAYLIST_LOADED:-}" ]; then
|
||||||
PLAYLIST_CMD_REMOVE="rm"
|
PLAYLIST_CMD_REMOVE="rm"
|
||||||
PLAYLIST_CMD_UP="up"
|
PLAYLIST_CMD_UP="up"
|
||||||
@@ -26,14 +30,14 @@ playlist() {
|
|||||||
"$PLAYLIST_CMD_DOWN") mpv_playlist_move $((FZF_POS - 0)) $((FZF_POS - 1)) ;;
|
"$PLAYLIST_CMD_DOWN") mpv_playlist_move $((FZF_POS - 0)) $((FZF_POS - 1)) ;;
|
||||||
"$PLAYLIST_CMD_CLEAR") mpv_playlist_clear ;;
|
"$PLAYLIST_CMD_CLEAR") mpv_playlist_clear ;;
|
||||||
"$PLAYLIST_CMD_CLEAR_ABOVE")
|
"$PLAYLIST_CMD_CLEAR_ABOVE")
|
||||||
for i in $(seq "$FZF_POS"); do
|
for _ in $(seq "$FZF_POS"); do
|
||||||
mpv_rm_index 0
|
mpv_rm_index 0
|
||||||
done
|
done
|
||||||
;;
|
;;
|
||||||
"$PLAYLIST_CMD_CLEAR_BELOW")
|
"$PLAYLIST_CMD_CLEAR_BELOW")
|
||||||
cnt=$(mpv_playlist_count)
|
cnt=$(mpv_playlist_count)
|
||||||
rem=$((cnt - FZF_POS + 1))
|
rem=$((cnt - FZF_POS + 1))
|
||||||
for i in $(seq "$rem"); do
|
for _ in $(seq "$rem"); do
|
||||||
mpv_rm_index $((FZF_POS - 1))
|
mpv_rm_index $((FZF_POS - 1))
|
||||||
done
|
done
|
||||||
;;
|
;;
|
||||||
|
@@ -1,10 +1,19 @@
|
|||||||
|
# Preview methods
|
||||||
|
#
|
||||||
|
# For now, only artist previews are supported.
|
||||||
|
|
||||||
|
# This internal method reshapes the text to be shown in the preview. This
|
||||||
|
# creates a border on both horizontal ends.
|
||||||
|
#
|
||||||
|
# The text is read from stdin.
|
||||||
__shape() {
|
__shape() {
|
||||||
cat | tr -d '\r' | fold -s -w "$((FZF_PREVIEW_COLUMNS - 4))" | awk '{print " "$0" "}'
|
cat | tr -d '\r' | fold -s -w "$((FZF_PREVIEW_COLUMNS - 4))" | awk '{print " "$0" "}'
|
||||||
}
|
}
|
||||||
|
|
||||||
# Print preview of artist
|
# Print preview of artist
|
||||||
# @input $1: MusicBrainz Artist ID
|
#
|
||||||
__preview_artist() {
|
# @input $1: MusicBrainz artist ID
|
||||||
|
preview_artist() {
|
||||||
desc=$(mb_artist_enwikipedia "$1" | $JQ '.extract' | __shape)
|
desc=$(mb_artist_enwikipedia "$1" | $JQ '.extract' | __shape)
|
||||||
[ "$desc" ] || desc=$(mb_artist_discogs "$1" | $JQ '.profile' | sed 's/\[a=\([^]]*\)\]/\1/g' | __shape)
|
[ "$desc" ] || desc=$(mb_artist_discogs "$1" | $JQ '.profile' | sed 's/\[a=\([^]]*\)\]/\1/g' | __shape)
|
||||||
if [ "$(mb_artist "$1" | $JQ '.type')" = "Person" ]; then
|
if [ "$(mb_artist "$1" | $JQ '.type')" = "Person" ]; then
|
||||||
|
@@ -25,11 +25,16 @@
|
|||||||
# - q_single: Release group is of type single
|
# - q_single: Release group is of type single
|
||||||
# - q_official: Release is official
|
# - q_official: Release is official
|
||||||
|
|
||||||
|
# Clean a filter string
|
||||||
|
#
|
||||||
|
# This method reads from stdin a string and removes all colors and escapes
|
||||||
|
# white spaces.
|
||||||
__clean_filter() {
|
__clean_filter() {
|
||||||
cat | sed "s/${ESC}\[[0-9;]*[mK]//g" | sed "s/ /\\\ /g"
|
cat | sed "s/${ESC}\[[0-9;]*[mK]//g" | sed "s/ /\\\ /g"
|
||||||
}
|
}
|
||||||
|
|
||||||
# Determine preset query
|
# Determine preset query
|
||||||
|
#
|
||||||
# @argument $1: Current view
|
# @argument $1: Current view
|
||||||
# @argument $2: Key pressed (optional)
|
# @argument $2: Key pressed (optional)
|
||||||
#
|
#
|
||||||
|
@@ -1,3 +1,13 @@
|
|||||||
|
# Load the tools required for this application. The tools are preset with
|
||||||
|
# default command-line arguments.
|
||||||
|
#
|
||||||
|
# List of tools:
|
||||||
|
# - fzf: in order to display, search, and navigate lists
|
||||||
|
# - curl: for API access
|
||||||
|
# - jq: to parse json files
|
||||||
|
# - mpv: music player
|
||||||
|
# - socat: to communicate with the socket mpv is bound to
|
||||||
|
# - xsel: to copy content to the clipboard (not necessary)
|
||||||
if [ ! "${TOOLS_LOADED:-}" ]; then
|
if [ ! "${TOOLS_LOADED:-}" ]; then
|
||||||
if command -v "fzf" >/dev/null; then
|
if command -v "fzf" >/dev/null; then
|
||||||
FZF="fzf --black --ansi --cycle --tiebreak=chunk,index"
|
FZF="fzf --black --ansi --cycle --tiebreak=chunk,index"
|
||||||
|
Reference in New Issue
Block a user