#!/usr/bin/env bash # Hopefully one day this will be a full featured Pleroma client. # Let's see how far we can get. :) # Display usage information. help() { echo "${0##*/}" echo "Released under the terms of the WTFPL License" echo -e "Usage:\n" echo "With no arguments, open the interactive client." for i in "${!command[@]}" ; do echo "-${i/:/ }: ${command[${i}]}" done | sort echo echo "Configuration files can be found in ${configPath}" echo exit 0 } # Get an oauth token get_oauth_token() { echo "Welcome to ${softwareName}!" echo echo "Let's get you connected to your instance." while [[ -z "${instanceURL}" ]]; do echo read -er -p "Enter the URL of a Pleroma instance: " instanceURL if [[ ! "${instanceURL}" =~ ^https:// ]]; then instanceURL="https://${instanceURL}" fi done redirectURI="urn:ietf:wg:oauth:2.0:oob" website="https://git.stormux.org/storm/ratatoskr" # get client id and secret curl -s -X POST -d client_name="${softwareName}" -d "redirect_uris=${redirectURI}" -d "scopes=read write follow push" -d "website=${website}" "${instanceURL}/api/v1/apps" | jq --raw-output '"client_id=\"\(.client_id)\"\nclient_secret=\"\(.client_secret)\""' > "${configPath}/${configFile}" # Load the new variables from the configuration file source "${configPath}/${configFile}" # Create the url to get the oauth token local url="${instanceURL}/oauth/authorize?client_id=${client_id}&redirect_uri=${redirectURI}&response_type=code&scope=read%20write%20follow%20push" echo "Please open the following url in your browser." echo "Copy the generated token, and paste it here, then press enter to continue." echo if command -v xclip &> /dev/null ; then echo "${url}" | tee >(xclip -selection clipboard -d "${DISPLAY:-:0}" 2> /dev/null && echo "For convenience the url has been copied to your clipboard.") else echo "${url}" fi echo read -er oauth_token # This is actually a authorization token, so get the actual token. oauth_token="$(curl -sS -d "client_id=${client_id}" -d "client_secret=${client_secret}" -d "code=${oauth_token}" -d "grant_type=authorization_code" "${instanceURL}/oauth/token" | jq --raw-output '(.access_token)')" echo "oauth_token=\"${oauth_token}\"" >> "${configPath}/${configFile}" echo "instanceURL=\"${instanceURL}\"" >> "${configPath}/${configFile}" } play_sound() { local soundFile="${configPath}/soundpacks/${sound_pack:-default}/${1}.opus" if [[ -e "${soundFile}" ]]; then sox -qV0 "${soundFile}" -d &> /dev/null & fi } # Functions that deal with posting. # Scrobble music with -S flag scrobble_music() { local result result="$(curl -sS --oauth2-bearer "${oauth_token}" \ -d "$(playerctl metadata -f 'album={{album}}')" \ -d "$(playerctl metadata -f 'artist={{artist}}')" \ -d "$(playerctl metadata -f 'title={{title}}')" \ "${instanceURL}/api/v1/pleroma/scrobble")" # Check for errors if [[ $? -ne 0 ]]; then echo "there was a problem contacting the server" play_sound error exit 1 fi local error="$(echo "$result" | jq -r '.error')" if [[ "$error" != "null" ]]; then echo "Error: $error" play_sound error exit 1 fi echo "Track scrobbled!" play_sound scrobble exit 0 } # Post music with -M flag requires playerctl. post_music() { local text="[$(playerctl metadata -f 'Now playing "{{title}}" by "{{artist}}" from "{{album}}"')]" local link="(https://www.youtube.com/results?search_query=$(playerctl metadata -f '{{artist}} {{title}}' | urlencode -b))" local json=$(jq -n --arg status "$text$link" --arg spoiler_text "Music" --arg content_type "text/markdown" '{status: $status, spoiler_text: $spoiler_text, content_type: $content_type}') local result result="$(curl -sS --oauth2-bearer "${oauth_token}" -H "Content-Type: application/json" \ -d "$json" \ "${instanceURL}/api/v1/statuses")" # Check for errors if [[ $? -ne 0 ]]; then echo "there was a problem contacting the server" play_sound error exit 1 fi local error="$(echo "$result" | jq -r '.error')" if [[ "$error" != "null" ]]; then echo "Error: $error" play_sound error exit 1 fi echo "Music posted!" play_sound post exit 0 } # Post status with -p flag, command line. post_status() { local text="$@" visibility="${visibility:-public}" local content_type="${content_type:-text/markdown}" local json="$(jq -n --arg status "$text" \ --arg spoiler_text "$spoiler_text" \ --arg visibility "$visibility" \ --arg content_type "$content_type" \ '{ status: $status, spoiler_text: $spoiler_text, visibility: $visibility, content_type: $content_type }')" local result result="$(curl -sS --oauth2-bearer "${oauth_token}" -H "Content-Type: application/json" \ -d "$(echo "$json" | jq 'if .spoiler_text == "" then del(.spoiler_text) else . end | if .visibility == "" then del(.visibility) else . end')" \ "${instanceURL}/api/v1/statuses")" if [[ $? -ne 0 ]]; then echo "there was a problem contacting the server" play_sound error exit 1 fi local error="$(echo "$result" | jq -r '.error')" if [[ "$error" != "null" ]]; then echo "Error: $error" play_sound error exit 1 fi echo "Status posted!" play_sound post exit 0 } # Variable initialization configPath="${XDG_CONFIG_HOME:-$HOME/.config}/ratatoskr" # Path for settings, usually ~/.config/ratatoskr configFile="default.conf" # The default config file, eventually will support multiple accounts. softwareName="Ratatoskr" # The name of the client. # Main code starts here # Check for dependencies dependencies=( "jq" "sox" "urlencode" ) for i in "${dependencies[@]}" ; do if ! command -v "$i" &> /dev/null ; then echo "Missing dependency: $i" exit 2 fi done # make sure the configuration and soundpack paths exist: mkdir -p "${configPath}/soundpacks" # Associative array of command line parameters and short description of what they do. declare -A command=( [C]="Recreate default configuration file. Acquire new oauth token." [h]="Help, show usage information for ${0##*/}." [M]="Post the currently playing music track, requires playerctl." [p:]="Post from the command line, e.g. ${0##*/} -p \"hello world\"" [S]="Scrobble the currently playing music track, requires playerctl." ) # if the default file doesn't exist, create it if [[ ! -e "${configPath}/${configFile}" ]]; then get_oauth_token else # Read configuration file source "${configPath}/${configFile}" fi # Handle command line parameters # Convert the keys of the associative array to a format usable by getopts args="${!command[*]}" args="${args//[[:space:]]/}" while getopts "${args}" i ; do case "$i" in C) get_oauth_token;; h) help;; M) post_music;; p) post_status "${OPTARG}";; S) scrobble_music;; esac done exit 0