Files
nixos-config/.gitea/scripts/update-appimage-nix.sh
T

147 lines
4.1 KiB
Bash

#!/usr/bin/env bash
set -euo pipefail
: "${FILE:?FILE is required}"
: "${LATEST_RELEASE_URL:?LATEST_RELEASE_URL is required}"
: "${DOWNLOAD_URL_TEMPLATE:?DOWNLOAD_URL_TEMPLATE is required}"
: "${RELEASE_API_REPO:?RELEASE_API_REPO is required}"
if command -v python >/dev/null 2>&1; then
PYTHON_BIN=python
elif command -v python3 >/dev/null 2>&1; then
PYTHON_BIN=python3
else
echo "python is required but was not found"
exit 1
fi
version_strip_prefix="${LATEST_VERSION_STRIP_PREFIX:-v}"
release_tag_template="${RELEASE_TAG_TEMPLATE:-{version}}"
release_tag_template="${release_tag_template//$'\r'/}"
current_version=$($PYTHON_BIN - <<'PY'
import re
import os
p=os.environ['FILE']
s=open(p).read()
m=re.search(r'version\s*=\s*"([^"]+)";', s)
print(m.group(1) if m else "")
PY
)
latest_version=$(curl -fsSLI -o /dev/null -w '%{url_effective}' "$LATEST_RELEASE_URL" \
| sed -E 's#.*/##')
if [ -n "$version_strip_prefix" ]; then
latest_version="${latest_version#${version_strip_prefix}}"
fi
echo "current=$current_version"
echo "latest=$latest_version"
if [ -z "$latest_version" ] || [ "$latest_version" = "$current_version" ]; then
echo "updated=false" >> "$GITHUB_OUTPUT"
exit 0
fi
download_url="${DOWNLOAD_URL_TEMPLATE//\{version\}/$latest_version}"
new_hash=$(nix store prefetch-file --json "$download_url" | "$PYTHON_BIN" -c 'import json,sys; print(json.load(sys.stdin)["hash"])')
export LATEST_VERSION="$latest_version"
export NEW_HASH="$new_hash"
"$PYTHON_BIN" - <<'PY'
import os
import re
p=os.environ['FILE']
s=open(p).read()
s=re.sub(r'version\s*=\s*"[^"]+"', f'version = "{os.environ["LATEST_VERSION"]}"', s, count=1)
s=re.sub(r'hash\s*=\s*"[^"]+"', f'hash = "{os.environ["NEW_HASH"]}"', s, count=1)
open(p,'w').write(s)
PY
echo "updated=true" >> "$GITHUB_OUTPUT"
echo "version=$latest_version" >> "$GITHUB_OUTPUT"
echo "previous_version=$current_version" >> "$GITHUB_OUTPUT"
release_tag="${release_tag_template//\{version\}/$latest_version}"
release_tag="${release_tag#\{}"
release_tag="${release_tag%\}}"
release_tag="${release_tag#\'}"
release_tag="${release_tag%\'}"
release_url="${LATEST_RELEASE_URL%/latest}/tag/${release_tag}"
release_html=$(curl -fsSL "$release_url" || true)
release_notes=""
if [ -n "$release_html" ]; then
release_notes=$(printf '%s' "$release_html" | "$PYTHON_BIN" -c '
import re, sys
from html.parser import HTMLParser
html = sys.stdin.read()
m = re.search(r"<div[^>]*data-test-selector=\"body-content\"[^>]*class=\"[^\"]*markdown-body[^\"]*\"[^>]*>(.*?)</div>", html, re.S)
if not m:
print("")
raise SystemExit(0)
fragment = m.group(1)
class Extractor(HTMLParser):
def __init__(self):
super().__init__()
self.out = []
self.in_code = False
def handle_starttag(self, tag, attrs):
if tag in ("br",):
self.out.append("\n")
elif tag in ("p", "div", "h1", "h2", "h3", "h4", "h5", "h6", "li"):
if self.out and not self.out[-1].endswith("\n"):
self.out.append("\n")
elif tag in ("pre",):
if self.out and not self.out[-1].endswith("\n"):
self.out.append("\n")
self.out.append("```\n")
self.in_code = True
elif tag in ("code",) and not self.in_code:
self.out.append("`")
def handle_endtag(self, tag):
if tag in ("p", "div", "li", "h1", "h2", "h3", "h4", "h5", "h6"):
if not self.out or not self.out[-1].endswith("\n"):
self.out.append("\n")
elif tag in ("pre",):
if not self.out or not self.out[-1].endswith("\n"):
self.out.append("\n")
self.out.append("```\n")
self.in_code = False
elif tag in ("code",) and not self.in_code:
self.out.append("`")
def handle_data(self, data):
if data:
self.out.append(data)
parser = Extractor()
parser.feed(fragment)
text = "".join(parser.out)
text = re.sub(r"\n{3,}", "\n\n", text)
print(text.strip())
' || true)
else
echo "warning: failed to fetch release page ${release_url}"
fi
if [ -z "$release_notes" ]; then
release_notes="_No changelog found on upstream release page. Check ${release_url}._"
fi
delimiter="CHANGELOG_$(date +%s%N)"
{
echo "changelog<<${delimiter}"
printf '%s\n' "$release_notes"
echo "${delimiter}"
} >> "$GITHUB_OUTPUT"