Last active
October 21, 2023 19:04
-
-
Save onedr0p/8fd8455f08f4781cad9e01a1d65bc34f to your computer and use it in GitHub Desktop.
Transmission Garbage Collector
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/bin/sh | |
# | |
# INFO | |
# | |
# This works if sonarr and radarr are set up to have a Category of sonarr and radarr respectively | |
# If you are using other Categories to save your automated downloads, update the script where you see: | |
# "radarr"|"sonarr") | |
# This script will not touch anything outside those Categories | |
# Set this file on a cron for every 5 minutes | |
# Using Docker? Make your cron something like this: | |
# /usr/bin/docker exec $(/usr/bin/docker ps | grep "linuxserver/transmission:latest" | awk '{print $1}') bash "/path/to/transmission-gc.sh" | |
# Set =~ to be insensitive | |
shopt -s nocasematch | |
TRANS_REMOTE_BIN=/usr/bin/transmission-remote | |
TRANS_HOST="127.0.0.1:9091" | |
# Amount of time (in seconds) after a torrent completes to delete them | |
RETENTION=2592000 | |
# Delete torrents only when ratio is above | |
RATIO="1.5" | |
# Clean up torrents where trackers have torrent not registered | |
# filter list by * (which signifies a tracker error) | |
TORRENT_DEAD_LIST=($("${TRANS_REMOTE_BIN}" "${TRANS_HOST}" -l | sed -e '1d;$d;s/^ *//' | cut --only-delimited --delimiter=' ' --fields=1 | egrep '[0-9]+\*' | sed 's/\*$//')) | |
for torrent_id in "${TORRENT_DEAD_LIST[@]}" | |
do | |
# Get the torrents metadata | |
torrent_info=$("${TRANS_REMOTE_BIN}" "${TRANS_HOST}" --torrent "${torrent_id}" -i -it) | |
torrent_name=$(echo "${torrent_info}" | grep "Name: *" | sed 's/Name\:\s//i' | awk '{$1=$1};1') | |
torrent_path=$(echo "${torrent_info}" | grep "Location: *" | sed 's/Location\:\s//i' | awk '{$1=$1};1') | |
torrent_size=$(echo "${torrent_info}" | grep "Downloaded: *" | sed 's/Downloaded\:\s//i' | awk '{$1=$1};1') | |
torrent_label=$(basename "${torrent_path}") | |
case "${torrent_label}" in | |
"radarr"|"sonarr") | |
torrent_error=$(echo "${torrent_info}" | grep "Got an error" | cut -d \" -f2) | |
if [[ "${torrent_error}" =~ "unregistered" ]] || [[ "${torrent_error}" =~ "not registered" ]]; then | |
# Delete torrent | |
"${TRANS_REMOTE_BIN}" "${TRANS_HOST}" --torrent "${torrent_id}" --remove-and-delete > /dev/null | |
fi | |
esac | |
done | |
# Clean up torrent where ratio is > ${RATIO} or seeding time > ${RETENTION} seconds | |
# do not filter the list, get all the torrents | |
TORRENT_ALL_LIST=($("${TRANS_REMOTE_BIN}" "${TRANS_HOST}" -l | sed -e '1d;$d;s/^ *//' | cut --only-delimited --delimiter=' ' --fields=1)) | |
for torrent_id in "${TORRENT_ALL_LIST[@]}" | |
do | |
# Get the torrents metadata | |
torrent_info=$("${TRANS_REMOTE_BIN}" "${TRANS_HOST}" --torrent "${torrent_id}" -i -it) | |
torrent_name=$(echo "${torrent_info}" | grep "Name: *" | sed 's/Name\:\s//i' | awk '{$1=$1};1') | |
torrent_path=$(echo "${torrent_info}" | grep "Location: *" | sed 's/Location\:\s//i' | awk '{$1=$1};1') | |
torrent_size=$(echo "${torrent_info}" | grep "Downloaded: *" | sed 's/Downloaded\:\s//i' | awk '{$1=$1};1') | |
torrent_label=$(basename "${torrent_path}") | |
torrent_seeding_seconds=$(echo "${torrent_info}" | grep "Seeding Time: *" | awk -F"[()]" '{print $2}' | sed 's/\sseconds//i') | |
torrent_ratio=$(echo "${torrent_info}" | grep "Ratio: *" | sed 's/Ratio\:\s//i' | awk '{$1=$1};1') | |
# Debug | |
# echo "${torrent_id} - ${torrent_ratio} - ${torrent_seeding_seconds} - ${torrent_label} - ${torrent_name}" | |
case "${torrent_label}" in | |
"radarr"|"sonarr") | |
# Torrents without a ratio have "None" instead of "0.0" let's fix that | |
if [[ "${torrent_ratio}" =~ "None" ]]; then | |
torrent_ratio="0.0" | |
fi | |
# delete torrents greater than ${TTL_SECONDS} | |
if [[ "${torrent_seeding_seconds}" -gt "${RETENTION}" ]]; then | |
"${TRANS_REMOTE_BIN}" "${TRANS_HOST}" --torrent "${torrent_id}" --remove-and-delete > /dev/null | |
fi | |
# delete torrents greater than ${RATIO} | |
if (( $(echo "${torrent_ratio} ${RATIO}" | awk '{print ($1 > $2)}') )); then | |
"${TRANS_REMOTE_BIN}" "${TRANS_HOST}" --torrent "${torrent_id}" --remove-and-delete > /dev/null | |
fi | |
esac | |
done |
The transmission-daemon resets the seeding time . So you will never get to the "RETENTION" seeding time.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
FYI, for anyone that lands here from google, but is looking for a solution for dockerized transmission - such as Haugene's awesome transmission/openvpn docker - Here's a solution that worked for me...
First create the script itself that will live inside the container and name it something recognizable. I used
remove_finished_torrents
. Don't worry about the environment variables yet.Then copy that file over to the container. It's probably best to copy it somewhere that will get mounted every time the container restarts and won't get wiped by any container updates from the maintainers. I have a local storage volume that gets mounted to /shared at container startup. (You can find [CONTAINER_ID] by running
docker ps
and looking for the container you recognize as your transmission client)Next, jump into your container and make sure the
remove_finished_torrents
script has the correct ownership & permissions. Because of how mine is setup, I made the script owned & only executable by root.Finally, outside the container, on the host, make another short script that cron will call. For my purposes, I have a
cleanup
script that runs daily from my cron. It looks something like this:Don't forget to make it executable:
User@HostMachine:~$ sudo chmod +x cleanup
If you're wondering about tmpUser, tmpPass, and the RPC variables: I have my transmission rpc login/pass set through /etc/environment. What I'm doing in this last script is taking the variables already setup - $RPC_U and $RPC_P - and passing them to temporary variables inside the transmission docker container - tmpUser and tmpPass. You can just as easily alter this by doing
-e tmpUser="MyActualUsername" -e tmpPass="MyActualPassword"
or omit the 2 environment flags entirely from this script and hardcode your user/pass into the original remove_finished_torrents script by changing the 2 instances of$tmpUser:$tmpPass
toMyActualUsername:MyActualPassword
.I would definitely recommend leaving in the
HAUGENE_ID
line (change the variable name to whatever you like) just to make the next command easier to read. Of course, if you use a different docker image instead of haugene/transmission-openvpn, you'll need to change "haugene" to something maybe "transmission" or "linuxserver/transmission" if that's what you use so that grep & awk can correctly match your container id. Remember your container id will change ever time you restart it, that's why this line is so crucial.My
cleanup
script just sits in the home dir and user's crontab references it:It runs at 4am local time everyday and redirects STOUT from the script to a clean.log file.
This might seem a little convoluted, having one script on the docker container and one script called from the host machine's cron to trigger the docker script, but it works great for how I have my server setup with usernames & passwords living in environment variables outside the container. The magic of finding the finished torrents and removing them is really in that one line from the first script, so that could be easily adapted to someone else's situation.
I worked out this solution using OP's script, as well as this one, and this comment