Compare commits

9 Commits

2 changed files with 21 additions and 193 deletions

View File

@ -32,11 +32,7 @@ function update_script() {
fi fi
setup_uv setup_uv
# Crawling the new version and checking whether an update is required # Crawling the new version and checking whether an update is required
RELEASE=$(curl -fsSL https://api.github.com/repos/BiluliB/profilarr/releases/latest | grep "tag_name" | cut -d'"' -f4) RELEASE=$(curl -fsSL https://api.github.com/repos/Dictionarry-Hub/profilarr/releases/latest | grep "tag_name" | awk '{print substr($2, 2, length($2)-3)}')
if [[ -z "$RELEASE" ]]; then
msg_error "Failed to fetch latest release version"
exit 1
fi
if [[ -f /opt/${APP}_version.txt ]] && [[ "${RELEASE}" == "$(cat /opt/${APP}_version.txt)" ]]; then if [[ -f /opt/${APP}_version.txt ]] && [[ "${RELEASE}" == "$(cat /opt/${APP}_version.txt)" ]]; then
msg_ok "No update required. ${APP} is already at ${RELEASE}" msg_ok "No update required. ${APP} is already at ${RELEASE}"
exit exit
@ -57,49 +53,31 @@ function update_script() {
msg_ok "Backup Created" msg_ok "Backup Created"
# Execute Update # Execute Update
msg_info "Updating $APP to ${RELEASE}" msg_info "Updating $APP to v${RELEASE}"
temp_file=$(mktemp) temp_file=$(mktemp)
curl -fsSL -o "$temp_file" "https://github.com/BiluliB/profilarr/archive/refs/tags/${RELEASE}.zip" curl -fsSL -o "$temp_file" "https://github.com/Dictionarry-Hub/profilarr/archive/refs/tags/${RELEASE}.zip"
unzip -q -o "$temp_file" -d /tmp unzip -q -o "$temp_file" -d /tmp
# Find the actual extracted directory name
EXTRACTED_DIR=$(find /tmp -maxdepth 1 -name "profilarr-*" -type d | head -n1)
if [[ -z "$EXTRACTED_DIR" ]]; then
msg_error "Failed to find extracted directory"
exit 1
fi
rm -rf /opt/${APP}/backend /opt/${APP}/frontend rm -rf /opt/${APP}/backend /opt/${APP}/frontend
mv "${EXTRACTED_DIR}/backend" /opt/${APP}/ mv "/tmp/profilarr-${RELEASE}/backend" /opt/${APP}/
mv "${EXTRACTED_DIR}/frontend" /opt/${APP}/ mv "/tmp/profilarr-${RELEASE}/frontend" /opt/${APP}/
# Update Python dependencies # Update Python dependencies
msg_info "Updating Python dependencies" msg_info "Updating Python dependencies"
cd /opt/${APP}/backend || exit cd /opt/${APP}/backend || exit
# Create a temporary requirements file excluding the incompatible PyYAML version
temp_req_file=$(mktemp)
grep -v "^PyYAML" requirements.txt >"$temp_req_file"
if [[ -f "/opt/${APP}/.requirements_checksum" ]]; then if [[ -f "/opt/${APP}/.requirements_checksum" ]]; then
CURRENT_CHECKSUM=$(md5sum "$temp_req_file" | awk '{print $1}') CURRENT_CHECKSUM=$(md5sum requirements.txt | awk '{print $1}')
STORED_CHECKSUM=$(cat /opt/${APP}/.requirements_checksum) STORED_CHECKSUM=$(cat /opt/${APP}/.requirements_checksum)
if [[ "$CURRENT_CHECKSUM" != "$STORED_CHECKSUM" ]]; then if [[ "$CURRENT_CHECKSUM" != "$STORED_CHECKSUM" ]]; then
msg_info "Requirements have changed. Performing full upgrade." msg_info "Requirements have changed. Performing full upgrade."
# Ensure compatible PyYAML is installed $STD uv pip install -r requirements.txt --python /opt/${APP}/venv/bin/python
$STD uv pip install --python /opt/${APP}/venv/bin/python "PyYAML>=6.0"
$STD uv pip install -r "$temp_req_file" --python /opt/${APP}/venv/bin/python
else else
msg_info "Requirements unchanged. Verifying installation." msg_info "Requirements unchanged. Verifying installation."
$STD uv pip install -r "$temp_req_file" --python /opt/${APP}/venv/bin/python $STD uv pip install -r requirements.txt --python /opt/${APP}/venv/bin/python
fi fi
else else
# Ensure compatible PyYAML is installed $STD uv pip install -r requirements.txt --python /opt/${APP}/venv/bin/python
$STD uv pip install --python /opt/${APP}/venv/bin/python "PyYAML>=6.0"
$STD uv pip install -r "$temp_req_file" --python /opt/${APP}/venv/bin/python
fi fi
md5sum "$temp_req_file" | awk '{print $1}' >/opt/${APP}/.requirements_checksum md5sum requirements.txt | awk '{print $1}' >/opt/${APP}/.requirements_checksum
rm -f "$temp_req_file"
msg_ok "Updated Python dependencies" msg_ok "Updated Python dependencies"
# Build frontend # Build frontend
@ -107,8 +85,6 @@ function update_script() {
cd /opt/${APP}/frontend || exit cd /opt/${APP}/frontend || exit
npm install npm install
npm run build npm run build
# Ensure the static directory exists before copying
mkdir -p /opt/${APP}/backend/app/static/
cp -r dist/* /opt/${APP}/backend/app/static/ cp -r dist/* /opt/${APP}/backend/app/static/
msg_ok "Built Frontend" msg_ok "Built Frontend"
@ -120,7 +96,7 @@ function update_script() {
# Cleaning up # Cleaning up
msg_info "Cleaning Up" msg_info "Cleaning Up"
rm -f "$temp_file" rm -f "$temp_file"
rm -rf "$EXTRACTED_DIR" rm -rf "/tmp/profilarr-${RELEASE}"
msg_ok "Cleanup Completed" msg_ok "Cleanup Completed"
# Last Action # Last Action

View File

@ -18,10 +18,7 @@ APPLICATION="profilarr"
msg_info "Installing Dependencies" msg_info "Installing Dependencies"
$STD apt-get install -y \ $STD apt-get install -y \
unzip \ unzip \
build-essential \ build-essential
libyaml-dev \
python3-dev \
git
msg_ok "Installed Dependencies" msg_ok "Installed Dependencies"
msg_info "Installing Python" msg_info "Installing Python"
@ -39,44 +36,18 @@ NODE_VERSION="20" install_node_and_modules
msg_ok "Installed Node.js" msg_ok "Installed Node.js"
msg_info "Setup ${APPLICATION}" msg_info "Setup ${APPLICATION}"
RELEASE=$(curl -fsSL https://api.github.com/repos/BiluliB/profilarr/releases/latest | grep "tag_name" | cut -d'"' -f4) RELEASE=$(curl -fsSL https://api.github.com/repos/Dictionarry-Hub/profilarr/releases/latest | grep "tag_name" | awk '{print substr($2, 2, length($2)-3)}')
if [[ -z "$RELEASE" ]]; then
msg_error "Failed to fetch latest release version"
exit 1
fi
temp_file=$(mktemp) temp_file=$(mktemp)
$STD curl -fsSL -o "$temp_file" "https://github.com/BiluliB/profilarr/archive/refs/tags/${RELEASE}.zip" $STD curl -fsSL -o "$temp_file" "https://github.com/Dictionarry-Hub/profilarr/archive/refs/tags/${RELEASE}.zip"
$STD unzip -q "$temp_file" -d /tmp $STD unzip -q "$temp_file" -d /tmp
$STD mkdir -p /opt/${APPLICATION} $STD mkdir -p /opt/${APPLICATION}
$STD mkdir -p /opt/${APPLICATION}_config $STD mkdir -p /opt/${APPLICATION}_config
$STD mv "/tmp/profilarr-${RELEASE}/backend" /opt/${APPLICATION}/
# Find the actual extracted directory name $STD mv "/tmp/profilarr-${RELEASE}/frontend" /opt/${APPLICATION}/
EXTRACTED_DIR=$(find /tmp -maxdepth 1 -name "profilarr-*" -type d | head -n1)
if [[ -z "$EXTRACTED_DIR" ]]; then
msg_error "Failed to find extracted directory"
exit 1
fi
$STD mv "${EXTRACTED_DIR}/backend" /opt/${APPLICATION}/
$STD mv "${EXTRACTED_DIR}/frontend" /opt/${APPLICATION}/
echo "${RELEASE}" >/opt/${APPLICATION}_version.txt echo "${RELEASE}" >/opt/${APPLICATION}_version.txt
$STD uv venv /opt/${APPLICATION}/venv $STD uv venv /opt/${APPLICATION}/venv
$STD uv pip install --python /opt/${APPLICATION}/venv/bin/python -r /opt/${APPLICATION}/backend/requirements.txt
# Install compatible PyYAML first, then exclude it from requirements
$STD uv pip install --python /opt/${APPLICATION}/venv/bin/python "PyYAML>=6.0"
# Create modified requirements file
temp_req_file=$(mktemp)
grep -v "^PyYAML" /opt/${APPLICATION}/backend/requirements.txt >"$temp_req_file"
echo "PyYAML>=6.0" >>"$temp_req_file"
$STD uv pip install --python /opt/${APPLICATION}/venv/bin/python -r "$temp_req_file"
$STD uv pip install --python /opt/${APPLICATION}/venv/bin/python gunicorn $STD uv pip install --python /opt/${APPLICATION}/venv/bin/python gunicorn
# Store the modified requirements checksum for future updates
md5sum "$temp_req_file" | awk '{print $1}' >/opt/${APPLICATION}/.requirements_checksum
rm -f "$temp_req_file"
msg_ok "Setup ${APPLICATION}" msg_ok "Setup ${APPLICATION}"
msg_info "Building Frontend" msg_info "Building Frontend"
@ -92,140 +63,21 @@ cat <<EOF >/etc/systemd/system/${APPLICATION}.service
[Unit] [Unit]
Description=Profilarr Profile Manager Description=Profilarr Profile Manager
After=network.target After=network.target
[Service] [Service]
Type=simple Type=simple
User=root User=root
WorkingDirectory=/opt/${APPLICATION}/backend WorkingDirectory=/opt/${APPLICATION}/backend
Environment=PATH=/opt/${APPLICATION}/venv/bin:/usr/local/bin:/usr/bin:/bin Environment=PATH=/opt/${APPLICATION}/venv/bin
Environment=CONFIG_PATH=/opt/${APPLICATION}_config Environment=CONFIG_PATH=/opt/${APPLICATION}_config
Environment=PYTHONPATH=/opt/${APPLICATION}/backend ExecStart=/opt/${APPLICATION}/venv/bin/gunicorn --bind 0.0.0.0:6868 --workers 2 --timeout 120 app.main:create_app()
Environment=GIT_PYTHON_REFRESH=quiet
ExecStart=/opt/${APPLICATION}/venv/bin/python -m gunicorn --bind 0.0.0.0:6868 --workers 2 --timeout 120 --pythonpath /opt/${APPLICATION}/backend "app.main:create_app()"
Restart=always Restart=always
RestartSec=10 RestartSec=10
StandardOutput=journal
StandardError=journal
[Install] [Install]
WantedBy=multi-user.target WantedBy=multi-user.target
EOF EOF
$STD systemctl daemon-reload $STD systemctl daemon-reload
systemctl enable -q --now ${APPLICATION}
# Test the application manually first
msg_info "Testing Application"
cd /opt/${APPLICATION}/backend
# Check if the main module exists and is importable
if ! /opt/${APPLICATION}/venv/bin/python -c "
import sys
import os
sys.path.insert(0, '/opt/${APPLICATION}/backend')
os.environ['GIT_PYTHON_REFRESH'] = 'quiet'
try:
import app.main
print('✓ app.main imported successfully')
except ImportError as e:
print(f'✗ Import error: {e}')
sys.exit(1)
except Exception as e:
print(f'✗ Other error: {e}')
sys.exit(1)
"; then
msg_error "Application import test failed"
# Try alternative approaches
msg_info "Trying alternative startup methods"
# Check if there's a direct run.py or main.py
if [[ -f "/opt/${APPLICATION}/backend/run.py" ]]; then
cat <<EOF >/etc/systemd/system/${APPLICATION}.service
[Unit]
Description=Profilarr Profile Manager
After=network.target
[Service]
Type=simple
User=root
WorkingDirectory=/opt/${APPLICATION}/backend
Environment=PATH=/opt/${APPLICATION}/venv/bin:/usr/local/bin:/usr/bin:/bin
Environment=CONFIG_PATH=/opt/${APPLICATION}_config
Environment=GIT_PYTHON_REFRESH=quiet
ExecStart=/opt/${APPLICATION}/venv/bin/python run.py
Restart=always
RestartSec=10
StandardOutput=journal
StandardError=journal
[Install]
WantedBy=multi-user.target
EOF
elif [[ -f "/opt/${APPLICATION}/backend/main.py" ]]; then
cat <<EOF >/etc/systemd/system/${APPLICATION}.service
[Unit]
Description=Profilarr Profile Manager
After=network.target
[Service]
Type=simple
User=root
WorkingDirectory=/opt/${APPLICATION}/backend
Environment=PATH=/opt/${APPLICATION}/venv/bin:/usr/local/bin:/usr/bin:/bin
Environment=CONFIG_PATH=/opt/${APPLICATION}_config
Environment=GIT_PYTHON_REFRESH=quiet
ExecStart=/opt/${APPLICATION}/venv/bin/python main.py
Restart=always
RestartSec=10
StandardOutput=journal
StandardError=journal
[Install]
WantedBy=multi-user.target
EOF
else
# Fallback to Flask development server
cat <<EOF >/etc/systemd/system/${APPLICATION}.service
[Unit]
Description=Profilarr Profile Manager
After=network.target
[Service]
Type=simple
User=root
WorkingDirectory=/opt/${APPLICATION}/backend
Environment=PATH=/opt/${APPLICATION}/venv/bin:/usr/local/bin:/usr/bin:/bin
Environment=CONFIG_PATH=/opt/${APPLICATION}_config
Environment=GIT_PYTHON_REFRESH=quiet
Environment=FLASK_APP=app.main:create_app
Environment=FLASK_RUN_HOST=0.0.0.0
Environment=FLASK_RUN_PORT=6868
ExecStart=/opt/${APPLICATION}/venv/bin/python -m flask run
Restart=always
RestartSec=10
StandardOutput=journal
StandardError=journal
[Install]
WantedBy=multi-user.target
EOF
fi
$STD systemctl daemon-reload
fi
systemctl enable ${APPLICATION}
systemctl start ${APPLICATION}
# Wait and check status
sleep 5
if systemctl is-active --quiet ${APPLICATION}; then
msg_ok "Service started successfully"
else
msg_error "Service failed to start. Checking logs..."
journalctl -u ${APPLICATION} --no-pager -n 20
exit 1
fi
msg_ok "Created Service" msg_ok "Created Service"
motd_ssh motd_ssh
@ -233,7 +85,7 @@ customize
msg_info "Cleaning up" msg_info "Cleaning up"
rm -f "$temp_file" rm -f "$temp_file"
rm -rf "$EXTRACTED_DIR" rm -rf "/tmp/profilarr-${RELEASE}"
$STD apt-get -y autoremove $STD apt-get -y autoremove
$STD apt-get -y autoclean $STD apt-get -y autoclean
msg_ok "Cleaned" msg_ok "Cleaned"