#!/bin/sh clear ### SET GLOBAL VARIABLES ### TEMPDIR="/tmp/pijarr" APPLIST="jackett sonarr lidarr radarr readarr prowlarr bazarr qbittorrent-nox flaresolverr" # Set terminal global variable for colors if supported. if [ -t 1 ]; then RED=$(printf '\033[31m') GREEN=$(printf '\033[32m') YELLOW=$(printf '\033[33m') BLUE=$(printf '\033[34m') MAGENTA=$(printf '\033[35m') CYAN=$(printf '\033[36m') BOLD=$(printf '\033[1m') RESET=$(printf '\033[0m') else RED="" GREEN="" YELLOW="" BLUE="" MAGENTA="" CYAN="" BOLD="" RESET="" fi # Temporary directory for application sources make_temp_dir() { task_start "Creating temporary directory ${TEMPDIR}..." mkdir -p "${TEMPDIR}" 2>/dev/null check_result } remove_temp_dir() { task_start "Removing temporary directory and files ${TEMPDIR}..." rm -Rf "${TEMPDIR}" 2>/dev/null check_result } task_start() { printf "\r[TASK] ${1}$(tput el)" } task_fail() { printf "\r[${RED}FAIL${RESET}] ${1}\n" } task_pass() { printf "\r[${GREEN}PASS${RESET}] ${1}\n" } task_skip() { printf "\r[${BLUE}SKIP${RESET}] ${1}\n" } task_info() { printf "\r[${CYAN}INFO${RESET}] ${1}$(tput el)\n" } task_warn() { printf "\r[${CYAN}WARN${RESET}] ${1}$(tput el)\n" } task_dbug() { printf "\r[${YELLOW}DBUG${RESET}] ${1}$(tput el)\n" } title_case() { printf "%s" "${1}" | tr '[:upper:]' '[:lower:]' | awk '{for(i=1;i<=NF;i++){ $i=toupper(substr($i,1,1)) substr($i,2) }}1' } lower_case() { printf "%s" "${1}" | tr '[:upper:]' '[:lower:]' } upper_case() { printf "%s" "${1}" | tr '[:lower:]' '[:upper:]' } check_result() { if [ $? -eq 0 ]; then task_pass else task_fail fi } ### START CHECK SYSTEM REQUIREMENTS AND ARCHITECTURE ### if cat /proc/cpuinfo >/dev/null 2>&1; then check_cpu=$(cat /proc/cpuinfo | grep 'model name' | head -1 | cut -d':' -f2 | xargs) check_model=$(cat /proc/cpuinfo | grep Model | head -1 | cut -d':' -f2 | xargs) if [ -z "${check_cpu}" ]; then result="${check_model}" else result="${check_cpu}" fi if [ -n "${result}" ]; then cpuBoardInfo="${result}" fi elif sysctl -a >/dev/null 2>&1; then result=$(sysctl -a | egrep -i 'hw.model' | sed 's/[^ ]* //') if [ -n "${result}" ]; then cpuBoardInfo="${result}" fi fi if command -v lsb_release >/dev/null 2>&1; then dist=$(lsb_release -d --short) osInfo="${dist}" fi if $(command -v uname) >/dev/null 2>&1; then arch=$(uname -m) kernel=$(uname -r) kname=$(uname -s) if [ -z "${dist}" ]; then osInfo="${kname}" fi kernelInfo="${kernel}" archInfo="${arch}" fi task_info "Checking hardware requirements..." task_info "Detected: ${cpuBoardInfo}" task_info "Detected: ${osInfo} ${kernelInfo}" systemInfo="${cpuBoardInfo} ${kname} ${osInfo} ${kernelInfo} ${archInfo}" case "${systemInfo}" in *x86_64* | *X86_64* | *amd64* | *AMD64*) task_pass "Detected: x86 64-Bit Architecture$(tput el)" JACKETT_ARCH="AMDx64" SERVARR_ARCH="x64" ;; esac case "${systemInfo}" in *aarch64* | *AARCH64* | *arm64* | *ARM64* | *armv8* | *ARMV8*) task_pass "Detected: ARM 64-Bit Architecture$(tput el)" JACKETT_ARCH="ARM64" SERVARR_ARCH="arm64" ;; esac case "${systemInfo}" in *aarch32* | *AARCH32* | *arm32* | *ARM32* | *armv7l* | *ARMV7L*) task_pass "Detected: ARM 32-Bit Architecture$(tput el)" JACKETT_ARCH="ARM32" SERVARR_ARCH="arm" ;; esac case "${systemInfo}" in *linux* | *Linux* | *LINUX*) task_pass "Detected: Linux based OS$(tput el)" ;; *) task_fail "Linux required. Not detected. Exiting...$(tput el)" echo "" exit 1 ;; esac ### END CHECK SYSTEM REQUIREMENTS AND ARCHITECTURE ### # Fetch latest jackett release from https://github.com/Jackett/Jackett/releases if command -v curl >/dev/null 2>&1; then # Run the curl command jackett_latest=$(curl -s https://github.com/Jackett/Jackett/releases | sed -n 's/.*href="\([^"]*\).*/\1/p' | grep Linux${JACKETT_ARCH}.tar.gz -A 0 | head -n 1) jackett_src_url="https://github.com${jackett_latest}" else # Run the wget command jackett_latest=$(wget -qO- https://github.com/Jackett/Jackett/releases | sed -n 's/.*href="\([^"]*\).*/\1/p' | grep Linux${JACKETT_ARCH}.tar.gz -A 0 | head -n 1) jackett_src_url="https://github.com${jackett_latest}" fi # Fetch latest radarr, lidarr and sonarr builds. Links below select latest release. radarr_src_url="https://radarr.servarr.com/v1/update/master/updatefile?os=linux&runtime=netcore&arch=${SERVARR_ARCH}" lidarr_src_url="https://lidarr.servarr.com/v1/update/master/updatefile?os=linux&runtime=netcore&arch=${SERVARR_ARCH}" prowlarr_src_url="http://prowlarr.servarr.com/v1/update/master/updatefile?os=linux&runtime=netcore&arch=${SERVARR_ARCH}" readarr_src_url="http://readarr.servarr.com/v1/update/develop/updatefile?os=linux&runtime=netcore&arch=${SERVARR_ARCH}" sonarr_src_url="https://services.sonarr.tv/v1/download/main/latest?version=4&os=linux&arch=${SERVARR_ARCH}" bazarr_src_url="https://github.com/morpheus65535/bazarr/releases/latest/download/bazarr.zip" qbittorrent_nox_src_url="Not applicable. Installed via apt." flaresolverr_src_url="https://github.com/FlareSolverr/FlareSolverr/archive/refs/tags/v3.4.6.tar.gz" check_sources() { task_info "Application Installation Source URLs" for app in ${APPLIST}; do app_var_name=$(echo "${app}" | tr '-' '_') # Replace hyphens with underscores src_url_var="${app_var_name}_src_url" src_url=$(eval echo \$"$src_url_var") task_info "${app} src: ${src_url}" done } # Function to pause script and check if the user wishes to continue. check_continue() { local response while true; do read -r -p "[${GREEN}USER${RESET}] Do you wish to continue (y/N)? " response case "${response}" in [yY][eE][sS] | [yY]) echo break ;; *) echo exit ;; esac done } press_any_key() { echo "" printf "%s " "[${GREEN}USER${RESET}] Press enter to continue..." read ans } # Function to check if superuser or running using sudo check_superuser() { if [ "$(id -u)" -ne 0 ]; then task_fail "Script must be run by superuser or using sudo command\n" exit 1 fi } pkg_updates() { task_info "Updating and upgrading packages using apt..." apt update apt upgrade -y apt --fix-broken install -y apt autoclean -y 2>/dev/null apt autoremove -y 2>/dev/null } # Function to check if packages are installed and install them if they are not found. pkg_install() { for pkg in "${@}"; do task_start "Checking for required package > ${pkg}" pkgStatus=$(dpkg -s "${pkg}" 2>/dev/null) result=$? if [ "${result}" -ne 0 ]; then task_warn "Package ${pkg} not installed.$(tput el)" task_info "Installing ${pkg}..." apt install -y "${pkg}" else task_pass "Package ${pkg} already installed.$(tput el)" fi done } pkg_remove() { for pkg in "${@}"; do task_start "Checking for required package > ${pkg}" pkgStatus=$(dpkg -s "${pkg}" 2>/dev/null) result=$? if [ "${result}" -eq 0 ]; then task_warn "Package ${pkg} installed.$(tput el)" task_info "Removing ${pkg}..." apt remove -y "${pkg}" else task_pass "Package ${pkg} not installed.$(tput el)" fi done } # Function to install all the dependencies including packages and server keys. setup_dependencies() { task_info "Installing required dependencies..." pkg_install curl wget unzip apt-transport-https dirmngr gnupg ca-certificates task_info "Installing mono, sqlite3 and supporting libraries..." pkg_install mono-complete mediainfo sqlite3 libmono-cil-dev libchromaprint-tools } setup_bazarr_dependencies() { task_info "Installing required Bazarr dependencies..." pkg_install python3-dev python3-pip python3-libxml2 python3-lxml unrar-free unar ffmpeg libxml2-dev libxslt1-dev libatlas-base-dev python_version=$(python3 -V 2>&1 | grep -Po '(?<=Python )\d+\.\d+') # Attempt to install specific python3-venv version if ! pkg_install python${python_version}-venv; then # If installation fails, install the default python3-venv pkg_install python3-venv fi } setup_flaresolverr_dependencies() { task_info "Installing required FlareSolverr dependencies..." # Install Xvfb for headless browser display pkg_install xvfb # Install Python 3.13 from deadsnakes PPA task_info "Adding deadsnakes PPA for Python 3.13..." # Check if add-apt-repository command exists, if not try to install it if ! command -v add-apt-repository >/dev/null 2>&1; then task_info "Installing software-properties-common for add-apt-repository..." apt update apt install -y software-properties-common 2>/dev/null || { task_warn "software-properties-common not available, trying manual PPA add..." echo "deb http://ppa.launchpad.net/deadsnakes/ppa/ubuntu jammy main" > /etc/apt/sources.list.d/deadsnakes-ppa.list apt-key adv --keyserver keyserver.ubuntu.com --recv-keys F23C5A6CF475977595C89F51BA6932366A755776 } fi # Try to add PPA if add-apt-repository is available if command -v add-apt-repository >/dev/null 2>&1; then add-apt-repository -y ppa:deadsnakes/ppa fi apt update task_info "Installing Python 3.13 and venv..." pkg_install python3.13 python3.13-venv python3.13-dev # Chrome is already installed per user confirmation } # Function to output PiJARR ascii and details of script. banner_info() { printf '%s\n' " ${RED}▓▓▓▓▓▓▓▓▓▓▓▓ ${GREEN}▓▓▓▓${RESET} ▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ${RED}▓▓▓▓ ▓▓▓▓ ${RESET} ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ${RED}▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓${RESET} ▓▓▓▓ ▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓▓▓ ${RED}▓▓▓▓ ▓▓▓▓${RESET} ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ${RED}▓▓▓▓ ▓▓▓▓${RESET} ▓▓▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓ ▓▓▓▓▓ ▓▓▓▓ ▓▓▓▓▓ " } script_info() { banner_info printf '%s\n' " Description: Installer for Jackett, Sonarr, Radarr, Lidarr, Readarr, Prowler, *Bazarr, and *FlareSolverr Tested with Raspberry Pi 3 & 4 running offical Raspberry Pi OS 64-Bit. Compatibility with other Intel AMD x64 (64-bit only) Debian based Linux distros Author: github.com/pijarr Notes: Requiries sudo/root superuser permissions to run. *Bazarr requires Python3 and additional packages to be downloaded. It will be run in a Python virtual environment (venv) to avoid dependency issues. *FlareSolverr requires Python 3.13 and Chrome/Chromium browser. It will be run in a Python virtual environment (venv) with Xvfb for headless browser operation. Commands \"apt update upgrade autoclean autoremove\" will be run upon continue. This is to ensure all packages are up to date and latest versions are used. If they are not already installed the required dependencies including Mono (Cross platform, open source .NET framework) may take a while to download and install. After initial setup the web interface for the application may need a short time to start before being available on the nominated host port. " } # Primary function to install and configure the applications. # Function can take one or more apps ie. setup_app jackett sonarr lidarr radarr setup_app() { clear make_temp_dir setup_dependencies for app in "${@}"; do if [ $(systemctl is-active "${app}") = "active" ]; then task_skip "A service for ${app} is already exists." continue fi date_stamp="$(date '+%Y-%m-%d %H%M')" app_name="$(lower_case ${app})" app_opt_path="/opt/$(title_case ${app_name})" app_lib_path="/var/lib/${app}" app_config_path="/var/lib/${app_name}/.config/$(title_case ${app_name})" if [ "${app}" = "bazarr" ]; then setup_bazarr_dependencies file_extension="zip" else file_extension="tar.gz" fi app_user="${app_name}" app_group="media" src_url=${app_name}\_src_url src_url=$(eval echo \$"$src_url") new_file="${app_name}.${file_extension}" task_info "Commencing install for ${app_name}..." task_start "Adding ${app_user} service user..." if id "${app_user}" >/dev/null 2>&1; then task_pass "User account for ${app_user} already exists." else useradd -s /usr/sbin/nologin -d "/var/lib/${app_user}" -r -m -U "${app_user}" 2>/dev/null check_result fi task_start "Adding ${app_group} service group..." if getent group "${app_group}" >/dev/null 2>&1; then task_pass "Group ${app_group} already exists." else groupadd "${app_group}" 2>/dev/null check_result fi task_start "Adding ${app_user} to ${app_group} service group..." if id "$app_user" | grep -qE "\b${app_group}\b"; then task_pass "User $app_user is already a member of the ${app_group} group." else usermod -a -G "${app_group}" "$app_user" check_result fi # Determine the actual username (not sudo or root) actual_user="${SUDO_USER:-$USER}" task_start "Adding ${actual_user} to ${app_group} service group..." if id "$actual_user" | grep -qE "\b${app_group}\b"; then task_pass "User $actual_user is already a member of the ${app_group} group." else usermod -a -G "${app_group}" "$actual_user" check_result fi task_info "Download source URL: ${src_url}" wget -O "${temp_dir}"/"${new_file}" -q --show-progress --progress=bar:force "${src_url}" 2>&1 && task_pass "Source file downloaded. SHA256: $(sha256sum ${temp_dir}/${new_file} | cut -d ' ' -f 1)$(tput el)" task_info "Extracting ${temp_dir}/${new_file} to ${app_opt_path}..." if [ "${app_name}" = "bazarr" ]; then unzip "${temp_dir}/${new_file}" -d "${app_opt_path}" task_info "Creating Python virtual environment (venv) for bazarr..." python3 -m venv "${app_opt_path}/venv" task_start "Activate Python venv for bazarr to install requirements..." # /bin/sh alternative for source command . "${app_opt_path}/venv/bin/activate" check_result task_info "Installing Bazarr Python import module requirements..." pip install -r "${app_opt_path}/requirements.txt" task_start "Deactivate Python venv..." deactivate check_result elif [ "${app_name}" = "flaresolverr" ]; then tar -xzf "${temp_dir}/${new_file}" -C "/opt/" # GitHub release extracts to FlareSolverr-3.4.6 format, need to rename mv /opt/FlareSolverr-* "${app_opt_path}" task_info "Creating Python virtual environment (venv) for flaresolverr..." python3.13 -m venv "${app_opt_path}/venv" task_start "Activate Python venv for flaresolverr to install requirements..." # /bin/sh alternative for source command . "${app_opt_path}/venv/bin/activate" check_result task_info "Installing FlareSolverr Python import module requirements..." pip install --upgrade pip pip install -r "${app_opt_path}/requirements.txt" task_start "Deactivate Python venv..." deactivate check_result else tar -xf "${temp_dir}/${new_file}" -C "/opt/" fi task_start "Setting user permissions on ${app_opt_path}..." chown -R "${app_user}":"${app_group}" "${app_opt_path}" check_result # Just in case some apps have permission problems with their /var/lib config working directories. task_start "Creating /var/lib config files..." mkdir -p "${app_config_path}" check_result task_start "Setting user permissions on ${app_lib_path}..." chown -R "${app_user}":"${app_group}" "${app_lib_path}" && chmod 775 "${app_lib_path}" check_result # Begin writting out the service configuration file. Minor change needed for Jackett. if [ "${app_name}" = "jackett" ]; then app_exec="${app_opt_path}/${app_name}_launcher.sh" elif [ "${app_name}" = "sonarr" ]; then app_exec="${app_opt_path}/$(title_case ${app_name}) -nobrowser -data=${app_lib_path}" elif [ "${app_name}" = "bazarr" ]; then app_exec="${app_opt_path}/venv/bin/python ${app_opt_path}/bazarr.py" elif [ "${app_name}" = "flaresolverr" ]; then app_exec="${app_opt_path}/venv/bin/python ${app_opt_path}/src/flaresolverr.py" else app_exec="${app_opt_path}/$(title_case ${app_name})" fi task_start "Writing service configuration file /etc/systemd/system/${app_name}.service..." if [ "${app_name}" = "flaresolverr" ]; then tee /etc/systemd/system/"${app_name}".service 1>/dev/null < /dev/null 2>&1 &' ExecStartPre=/bin/sleep 10 ExecStart=${app_exec} KillSignal=SIGINT TimeoutStopSec=20 [Install] WantedBy=multi-user.target EOF else tee /etc/systemd/system/"${app_name}".service 1>/dev/null </dev/null check_result task_start "Enabling auto start for ${app_name} service..." systemctl enable "${app_name}" 2>/dev/null check_result task_start "Starting ${app_name} service..." systemctl start "${app_name}" 2>/dev/null check_result check_service ${app_name} task_info "Completed install for ${app_name}\n" done remove_temp_dir } # Function to assist in removing the applications and their configuration files # Function can take one or more apps ie. remove_app jackett sonarr lidarr radarr remove_app() { clear for app in "${@}"; do app=$(lower_case "${app}") app_opt_path="/opt/$(title_case "${app}")" app_lib_path="/var/lib/${app}" task_warn "You are about to delete all settings and files for ${app}..." check_continue task_info "Deleting existing settings or files for ${app}..." task_start "Stopping ${app} service..." systemctl stop "${app}" 2>/dev/null task_pass task_start "Removing ${app_opt_path}..." if [ "${app_opt_path}" != "/opt" ] && [ "${app_opt_path}" != "/opt/" ]; then rm -rf "${app_opt_path}" 2>/dev/null task_pass else task_skip "Skipping removal of default /opt/ directory." fi task_start "Removing ${app_lib_path}..." if [ "${app_lib_path}" != "/var/lib" ] && [ "${app_lib_path}" != "/var/lib/" ]; then rm -rf "${app_lib_path}" 2>/dev/null task_pass else task_skip "Skipping removal of default /var/lib/ directory." fi task_start "Removing service config /etc/systemd/system/${app}.service..." rm "/etc/systemd/system/${app}.service"* 2>/dev/null task_pass task_info "Removing ${app} service user account..." deluser "${app}" 2>/dev/null task_start "Reloading systemctl daemon..." systemctl daemon-reload 2>/dev/null task_pass task_info "${app} deleted.\n" done } setup_qbittorrent_nox() { date_stamp="$(date '+%Y-%m-%d %H%M')" app_name="qbittorrent-nox" app_user="${app_name}" app_lib_path="/var/lib/${app_name}" app_group="media" task_info "Commencing install for ${app_name}..." task_start "Adding ${app_user} service user..." if id "${app_user}" >/dev/null 2>&1; then task_pass "User account for ${app_user} already exists." else useradd -s /usr/sbin/nologin -d "/var/lib/${app_user}" -r -m -U "${app_user}" 2>/dev/null check_result fi task_start "Adding ${app_group} service group..." if getent group "${app_group}" >/dev/null 2>&1; then task_pass "Group ${app_group} already exists." else groupadd "${app_group}" 2>/dev/null check_result fi task_start "Adding ${app_user} to ${app_group} service group..." if id "$app_user" | grep -qE "\b${app_group}\b"; then task_pass "User $app_user is already a member of the ${app_group} group." else usermod -a -G "${app_group}" "$app_user" check_result fi # Determine the actual username (not sudo or root) actual_user="${SUDO_USER:-$USER}" task_start "Adding ${actual_user} to ${app_group} service group..." if id "$actual_user" | grep -qE "\b${app_group}\b"; then task_pass "User $actual_user is already a member of the ${app_group} group." else usermod -a -G "${app_group}" "$actual_user" check_result fi task_info "Installing ${app_name} package..." pkg_install qbittorrent-nox task_start "Setting user permissions on ${app_lib_path}..." chown -R "${app_user}":"${app_group}" "${app_lib_path}" && chmod 775 "${app_lib_path}" check_result # Setup qBittorrent configuration with password before starting service setup_qbittorrent_config "pijarr" task_start "Writing service configuration file /etc/systemd/system/${app_name}.service..." tee /etc/systemd/system/"${app_name}".service 1>/dev/null </dev/null check_result task_start "Enabling auto start for ${app_name} service..." systemctl enable "${app_name}" 2>/dev/null check_result task_start "Starting ${app_name} service..." systemctl start "${app_name}" 2>/dev/null check_result check_service "${app_name}" task_info "Completed install for ${app_name}\n" task_info "${GREEN}═══════════════════════════════════════════════════════════════${RESET}" task_info "${BOLD}qBittorrent WebUI Credentials:${RESET}" task_info " URL: ${CYAN}http://$(hostname -I | awk '{print $1}'):8080${RESET}" task_info " Username: ${YELLOW}admin${RESET}" task_info " Password: ${YELLOW}pijarr${RESET}" task_info "" task_info "${RED}IMPORTANT:${RESET} Change this password immediately after first login!" task_info "Downloads: /var/lib/qbittorrent-nox/Downloads" task_info "${GREEN}═══════════════════════════════════════════════════════════════${RESET}" } generate_qbittorrent_password_hash() { local password="${1:-pijarr}" # Use Python to generate PBKDF2 hash compatible with qBittorrent python3 -c " import hashlib import os import base64 password = '${password}' salt = os.urandom(16) iterations = 100000 dk = hashlib.pbkdf2_hmac('sha512', password.encode(), salt, iterations) encoded_salt = base64.b64encode(salt).decode() encoded_hash = base64.b64encode(dk).decode() print(f'{encoded_salt}:{encoded_hash}') " } setup_qbittorrent_config() { local app_user="qbittorrent-nox" local config_dir="/var/lib/${app_user}/.config/qBittorrent" local config_file="${config_dir}/qBittorrent.conf" local password="${1:-pijarr}" task_info "Configuring qBittorrent with default credentials..." # Create config directory if it doesn't exist mkdir -p "${config_dir}" # Generate password hash local password_hash=$(generate_qbittorrent_password_hash "${password}") # Create qBittorrent.conf with WebUI password cat > "${config_file}" </dev/null task_pass task_start "Removing service config /etc/systemd/system/${app_name}.service..." rm "/etc/systemd/system/${app_name}.service"* 2>/dev/null task_pass pkg_remove qbittorrent-nox task_info "Removing ${app} service user account..." deluser "${app_name}" 2>/dev/null task_start "Removing ${app_lib_path}..." if [ "${app_lib_path}" != "/var/lib" ] && [ "${app_lib_path}" != "/var/lib/" ]; then rm -rf "${app_lib_path}" 2>/dev/null task_pass else task_skip "Skipping removal of default /var/lib/ directory." fi task_info "${app_name} deleted.\n" } active_services() { task_info "Active Services" for app in ${APPLIST}; do if [ $(systemctl is-active "${app}") = "active" ]; then task_pass "${app} service is active and running." fi done } check_service() { task_start "Checking service status for ${1}..." if [ $(systemctl is-active "$1") = "active" ]; then task_pass else task_fail fi } default_ports() { task_info "Default Application Ports" task_info "Jackett: http://hostip:9117" task_info "Sonarr: http://hostip:8989" task_info "Lidarr: http://hostip:8686" task_info "Radarr: http://hostip:7878" task_info "Readarr: http://hostip:8787" task_info "Prowlarr: http://hostip:9696" task_info "Bazarr: http://hostip:6767" task_info "FlareSolverr: http://hostip:8191" task_info "qBittorrent-nox: http://hostip:8080 (admin/pijarr)" } # Display a list of menu items for selection display_menu() { clear banner_info echo "==============" echo " Menu Options " echo "==============" echo printf "1. Install ALL (jackett sonarr lidarr radarr readarr prowlarr bazarr flaresolverr)\n" printf "2. Install jackett only\n" printf "3. Install sonarr only\n" printf "4. Install lidarr only\n" printf "5. Install radarr only\n" printf "6. Install readarr only\n" printf "7. Install prowlarr only\n" printf "8. Install bazarr only\n" printf "9. Install flaresolverr only\n" printf "\n10. Install qbittorrent-nox (headless BitTorrent client)\n" printf "\n11. Remove ALL (jackett sonarr lidarr radarr readarr prowlarr bazarr flaresolverr)\n" printf "12. Remove jackett only\n" printf "13. Remove sonarr only\n" printf "14. Remove lidarr only\n" printf "15. Remove radarr only\n" printf "16. Remove readarr only\n" printf "17. Remove prowlarr only\n" printf "18. Remove bazarr only\n" printf "19. Remove flaresolverr only\n" printf "\n20. Remove qbittorrent-nox\n" printf "\n21. Show active services\n" printf "22. Show application default ports\n" printf "23. Show application source urls\n" printf "\n24. Exit\n" echo printf " Enter option [1-24]: " while :; do read choice case ${choice} in 1) setup_app jackett sonarr lidarr radarr readarr prowlarr bazarr flaresolverr ;; 2) setup_app jackett ;; 3) setup_app sonarr ;; 4) setup_app lidarr ;; 5) setup_app radarr ;; 6) setup_app readarr ;; 7) setup_app prowlarr ;; 8) setup_app bazarr ;; 9) setup_dependencies setup_flaresolverr_dependencies setup_app flaresolverr ;; 10) setup_qbittorrent_nox ;; 11) remove_app jackett sonarr lidarr radarr readarr prowlarr bazarr flaresolverr ;; 12) remove_app jackett ;; 13) remove_app sonarr ;; 14) remove_app lidarr ;; 15) remove_app radarr ;; 16) remove_app readarr ;; 17) remove_app prowlarr ;; 18) remove_app bazarr ;; 19) remove_app flaresolverr ;; 20) remove_qbittorrent_nox ;; 21) clear active_services ;; 22) clear default_ports ;; 23) clear check_sources ;; 24) printf "\nExiting...\n" exit ;; *) clear display_menu ;; esac printf "\nSelection [${choice}] completed.\n" press_any_key clear display_menu done } main() { check_sources script_info check_superuser check_continue pkg_updates check_status display_menu } main