429 lines
9.7 KiB
Bash
Executable File
429 lines
9.7 KiB
Bash
Executable File
#! /bin/bash
|
|
#
|
|
# ptgdp - Play the Goddamned Playlist
|
|
# Copyright (C) 2019 Vintage Salt <rehashedsalt@cock.li>
|
|
#
|
|
# Distributed under terms of the MIT license.
|
|
#
|
|
set -e
|
|
|
|
# Import user-dirs.dirs
|
|
_xdguserdirs="${XDG_CONFIG_HOME:-$HOME/.config}/user-dirs.dirs"
|
|
[ -z "$XDG_MUSIC_DIR" ] && [ -r "$_xdguserdirs" ] && source "$_xdguserdirs"
|
|
unset _xdguserdirs
|
|
# Read-only set-once variables
|
|
declare -r _name="$(basename -- "$0")"
|
|
declare -r _musicdir="${XDG_MUSIC_DIR:-$HOME/Music}"
|
|
declare -r _ptgdpmusicdir="$_musicdir/PTGDP Songs"
|
|
declare -ra _supportedbackends=("mpd" "audacious")
|
|
# May-need-amended variables
|
|
declare _mpdroot="$_musicdir"
|
|
# Options
|
|
declare -A _config=(
|
|
[backend]="mpd"
|
|
[ytdl_args]="--geo-bypass"
|
|
)
|
|
declare _optconfigfile="${XDG_CONFIG_HOME:-$HOME/.config}/${_name}.conf"
|
|
declare -i _optautoplay=0
|
|
declare -i _optdlonly
|
|
declare -i _opthelp
|
|
declare -i _optmachinemode
|
|
declare -i _optverbose
|
|
# Working variables
|
|
declare -a _queue
|
|
declare -a _args
|
|
declare _return
|
|
|
|
# Helper functions
|
|
log() {
|
|
# Print a line to the terminal if _optverbose is greater than $2
|
|
# $2 defaults to 0
|
|
# loglevel 0: Daily-use messages
|
|
# loglevel 1: Detailed but not quite debugging
|
|
# loglevel 2: Definitely debugging
|
|
[ -z "$1" ] && return 1
|
|
local -i outstream=1
|
|
[ -n "$_optmachinemode" ] && outstream=2
|
|
if (( _optverbose >= ${2:-0} )); then
|
|
printf "%s\\n" "$1" >&"$outstream"
|
|
fi
|
|
}
|
|
warn() {
|
|
# Print a yellow line to the terminal, respecting _optverbose
|
|
[ -z "$1" ] && return 1
|
|
local -i outstream=1
|
|
[ -n "$_optmachinemode" ] && outstream=2
|
|
if (( _optverbose >= ${2:-0} )); then
|
|
if [ -t "$outstream" ]; then
|
|
printf "\\e[33m%s\\e[0m\\n" "$1" >&"$outstream"
|
|
else
|
|
printf "WARN: %s\\n" "$1" >&"$outstream"
|
|
fi
|
|
fi
|
|
}
|
|
error() {
|
|
# Print a red line to the terminal, exit if $2 is specified
|
|
[ -z "$1" ] && return 1
|
|
if [ -t 2 ]; then
|
|
printf "\\e[31m%s\\e[0m\\n" "$1" 1>&2
|
|
else
|
|
printf "ERROR: %s\\n" "$1" 1>&2
|
|
fi
|
|
[ -z "$2" ] && return
|
|
exit "${2:-1}"
|
|
}
|
|
has() {
|
|
# Parse out all arguments and try to find them in path
|
|
# If an argument cannot be found, set _return and fail
|
|
for prog in "$@"; do
|
|
if ! command -v "$prog" > /dev/null 2>&1; then
|
|
_return="$prog"
|
|
return 1
|
|
fi
|
|
done
|
|
return 0
|
|
}
|
|
sanitize() {
|
|
# Strip a line of special charactersj
|
|
[ -z "$1" ] && return 1
|
|
_return="${1//[^ a-zA-Z0-9\[\]|()_-:]/}"
|
|
}
|
|
|
|
# More complex helper functions
|
|
checksong() {
|
|
# Check to see if a song exists in the cache
|
|
[ -z "$1" ] && return 1
|
|
# Very rudimentary implementation
|
|
# TODO: Make this a lot more thorough, maybe ensure MIME is good
|
|
sanitize "$1"
|
|
for file in "$_ptgdpmusicdir/$_return"*; do
|
|
if [ -f "$file" ]; then
|
|
_return="$file"
|
|
return 0
|
|
fi
|
|
done
|
|
unset _return
|
|
return 1
|
|
}
|
|
cachesong() {
|
|
# Check to see if a song exists in the cache, downloading if not
|
|
# Sets _returnstatus to either "dl", "cache", or "err"
|
|
# Sets _return to the song filename
|
|
[ -z "$1" ] && return 1
|
|
sanitize "$1"
|
|
filename="$_ptgdpmusicdir/$_return"
|
|
if ! checksong "$_return"; then
|
|
log "Downloading new song \"$1\"" 2
|
|
local ytdllog
|
|
if ytdllog=$(youtube-dl \
|
|
--add-metadata \
|
|
--audio-format "best" \
|
|
--default-search "ytsearch" \
|
|
--playlist-items 1 \
|
|
"${_config[ytdl_args]}" \
|
|
-x \
|
|
-o "$filename.%(ext)s" \
|
|
"$1" 2>&1)
|
|
then
|
|
# Sanitize removes all periods, so this is safe
|
|
for file in "$filename".*; do
|
|
if [ -f "$file" ]; then
|
|
_return="$file"
|
|
else
|
|
error "Downloaded song \"$_return\" but could not find output file"
|
|
log "$ytdllog" 2
|
|
_returnstatus="err"
|
|
return 1
|
|
fi
|
|
done
|
|
_returnstatus="dl"
|
|
log "Downloaded song to \"$_return\"" 2
|
|
log "$ytdllog" 2
|
|
return 0
|
|
else
|
|
log "$ytdllog" 2
|
|
_returnstatus="err"
|
|
return 1
|
|
fi
|
|
else
|
|
_returnstatus="cache"
|
|
return 0
|
|
fi
|
|
}
|
|
queuesong() {
|
|
# Add a song to _queue, downloading if necessary
|
|
# Sets _return to one of: dl, cache, err
|
|
[ -z "$1" ] && return 1
|
|
cachesong "$1" || return 1
|
|
case $_returnstatus in
|
|
cache)
|
|
log "Using cached song \"$1\"" 1
|
|
log " $_return" 2
|
|
;;
|
|
*)
|
|
log "Downloaded song \"$1\""
|
|
log " $_return" 2
|
|
;;
|
|
esac
|
|
_queue+=("$_return")
|
|
}
|
|
validateline() {
|
|
# Takes a line and errors if it's just whitespace or a comment
|
|
local linenows=${1//[[:space:]]}
|
|
if ! [ "${1#\#}" = "$1" ] || [ -z "$linenows" ]; then
|
|
return 1
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
# Backend-specific functions
|
|
backend-validate() {
|
|
# Ensure the backend is even proper
|
|
# Returns 1 for missing dependencies
|
|
# Everything after that is backend-specific. See below
|
|
(
|
|
for backend in ${_supportedbackends[@]}; do
|
|
if [ "$backend" = "${_config[backend]}" ]; then
|
|
return 0
|
|
fi
|
|
done
|
|
return 1
|
|
) || return $?
|
|
# Backend-specific checks
|
|
case ${_config[backend]} in
|
|
audacious)
|
|
has audacious audtool || return 1
|
|
;;
|
|
mpd)
|
|
has mpd mpc || return 1
|
|
pgrep mpd > /dev/null 2>&1 || return 2
|
|
;;
|
|
esac
|
|
return 0
|
|
}
|
|
backend-enqueue() { # Enqueues a song
|
|
# Note: mpd will assume you've updated the library since!
|
|
[ -z "$1" ] && return 1
|
|
case ${_config[backend]} in
|
|
audacious)
|
|
audtool --playlist-addurl "$1"
|
|
return 0
|
|
;;
|
|
mpd)
|
|
file=${1##$_mpdroot/}
|
|
if ! mpc add "$file" > /dev/null 2>&1; then
|
|
errorval="$?"
|
|
error "Could not add file: \"$file\" ($errorval)"
|
|
fi
|
|
return 0
|
|
;;
|
|
esac
|
|
return 1
|
|
}
|
|
backend-play() {
|
|
# Plays a set up queue
|
|
case ${_config[backend]} in
|
|
audacious)
|
|
audtool --playback-play
|
|
;;
|
|
mpd)
|
|
if ! mpc play > /dev/null 2>&1; then
|
|
error "Could not play queue ($?)"
|
|
return 1
|
|
fi
|
|
return 0
|
|
;;
|
|
esac
|
|
log "Started playback" 1
|
|
}
|
|
backend-execqueue() {
|
|
# Executes a queue, enqueueing files and autoplaying if configured
|
|
case ${_config[backend]} in
|
|
audacious)
|
|
if ! pgrep audacious > /dev/null 2>&1; then
|
|
warn "Audacious is not running; songs will be downloaded but not enqueued"
|
|
fi
|
|
;;
|
|
mpd)
|
|
if ! mpc update --wait > /dev/null 2>&1; then
|
|
error "Failed to update mpd library" 51
|
|
fi
|
|
;;
|
|
esac
|
|
for song in "${_queue[@]}"; do
|
|
backend-enqueue "$song"
|
|
done
|
|
if [ "$_optautoplay" != "0" ]; then
|
|
backend-play
|
|
fi
|
|
}
|
|
|
|
# Core program functions
|
|
printhelp() {
|
|
cat << EOF
|
|
Usage: $_name [OPTION]... [FILE]...
|
|
Use youtube-dl and a music player to queue up or download a number of songs
|
|
given plaintext files with only search queries
|
|
|
|
-c [FILE] Load the given file in place of the usual config file
|
|
-d Download songs but don't queue them up
|
|
-h Print this help text
|
|
-m Machine-processible mode. Output all log text to STDERR
|
|
and print only the final list of songs to STDOUT,
|
|
separated by newlines.
|
|
-p Play the queue after it's built
|
|
-v Print more status messages. Stacks
|
|
|
|
Copyright (c) 2019 rehashedsalt@cock.li
|
|
Licensed under the MIT license
|
|
EOF
|
|
}
|
|
playlist() {
|
|
[ -z "$1" ] && return 1
|
|
[ -e "$1" ] || error "Playlist \"$1\" does not exist" 60
|
|
[ -r "$1" ] || error "Playlist \"$1\" could not be read" 61
|
|
local -i dlcache=0
|
|
local -i dlsuccess=0
|
|
local -i dlerr=0
|
|
local -i goodlines=0
|
|
local -i totallines=0
|
|
while read line; do
|
|
totallines+=1
|
|
if validateline "$line"; then
|
|
goodlines+=1
|
|
queuesong "$line" || error "Failed to enqueue song: \"$line\""
|
|
case $_returnstatus in
|
|
dl)
|
|
dlsuccess+=1
|
|
;;
|
|
cache)
|
|
dlcache+=1
|
|
;;
|
|
err)
|
|
dlerr+=1
|
|
;;
|
|
esac
|
|
else
|
|
continue
|
|
fi
|
|
done < "$1"
|
|
if [ -n "$_optmachinemode" ]; then
|
|
for file in "${_queue[@]}"; do
|
|
printf "$file\\n"
|
|
done
|
|
elif [ -z "$_optdlonly" ]; then
|
|
backend-execqueue
|
|
fi
|
|
log "Finished: $dlcache cached, $dlsuccess downloaded, $dlerr failed"
|
|
log "Playlist has $totallines total lines, $goodlines of them songs" 1
|
|
}
|
|
|
|
# Main
|
|
main() {
|
|
# Parse out arguments
|
|
# Done in a nested loop so that flags are position-independent
|
|
while [ -n "$1" ]; do
|
|
# Parse out flags
|
|
while getopts ":c:dhmpv" opt; do
|
|
case $opt in
|
|
c)
|
|
_optconfigfile="$OPTARG"
|
|
;;
|
|
d)
|
|
_optdlonly=1
|
|
;;
|
|
h)
|
|
_opthelp=1
|
|
;;
|
|
m)
|
|
_optmachinemode=1
|
|
;;
|
|
p)
|
|
_optautoplay=1
|
|
;;
|
|
v)
|
|
_optverbose+=1
|
|
;;
|
|
:)
|
|
error "Option requires argument: -$OPTARG" 2
|
|
;;
|
|
*)
|
|
error "Invalid option: -$OPTARG" 2
|
|
;;
|
|
esac
|
|
done
|
|
# Store arguments
|
|
shift $((OPTIND - 1))
|
|
if [ -n "$1" ]; then
|
|
_args+=("$1")
|
|
shift
|
|
fi
|
|
unset OPTIND
|
|
done
|
|
# Early hook for help
|
|
[ -n "$_opthelp" ] && printhelp && exit 0
|
|
# Parse out a config file if it exists
|
|
if [ -f "$_optconfigfile" ]; then
|
|
log "Loading config file" 2
|
|
while read line; do
|
|
# If the line has an equals sign and isn't a comment
|
|
if [ "$line" != "${line#*=}" ] && validateline "$line"; then
|
|
local key="${line%=*}"
|
|
local value="${line#*=}"
|
|
_config[$key]="$value"
|
|
log "Setting $key to $value" 2
|
|
fi
|
|
done < "$_optconfigfile"
|
|
else
|
|
warn "Could not find configuration file" 2
|
|
fi
|
|
# Validate critical options
|
|
if [ -z "$_optmachinemode" ]; then
|
|
# Ensure we have a good backend, assuming we're not in a scripting mode
|
|
(
|
|
for backend in ${_supportedbackends[@]}; do
|
|
if [ "$backend" = "${_config[backend]}" ]; then
|
|
return 0
|
|
fi
|
|
done
|
|
return 1
|
|
) || error "Unsupported backend: ${_config[backend]}" 50
|
|
# Ensure we meet its requirements
|
|
if ! backend-validate; then
|
|
errorcode=$?
|
|
case $errorcode in
|
|
1)
|
|
error "Missing dependency for backend ${_config[backend]}: $_return" 50
|
|
;;
|
|
*)
|
|
error "Backend error: $errorcode: $_return" 50
|
|
;;
|
|
esac
|
|
fi
|
|
fi
|
|
# Validate core program dependencies
|
|
log "Validating dependencies" 2
|
|
if ! has youtube-dl pgrep; then
|
|
error "Failed to find program: $_return" 1
|
|
fi
|
|
|
|
# Do the do
|
|
if [ -n "${_args[*]}" ]; then
|
|
# Files specified on the command line have priority
|
|
for arg in "${_args[@]}"; do
|
|
playlist "$arg"
|
|
unset _queue
|
|
declare -a _queue
|
|
done
|
|
elif ! [ -t 0 ]; then
|
|
# If there are none of those, read from STDIN
|
|
playlist "/dev/stdin"
|
|
else
|
|
warn "Nothing to do"
|
|
exit 0
|
|
fi
|
|
}
|
|
|
|
main "$@"
|
|
|