From 0e827a1f4c4f1b0a7b709d496c2a6c711c7a2c1e Mon Sep 17 00:00:00 2001 From: Ivan Carlos de Almeida Date: Sat, 13 Dec 2025 02:10:50 -0300 Subject: [PATCH] main beautify --- .gitea/workflows/release_build.yml | 383 +++++++++++++++++++++++++++++ Dockerfile | 3 + LICENSE | 21 ++ manifest.json | 4 + public/favicon.ico | Bin 0 -> 15086 bytes public/index.html | 354 ++++++++++++++++++++++++++ public/robots.txt | 2 + 7 files changed, 767 insertions(+) create mode 100644 .gitea/workflows/release_build.yml create mode 100644 Dockerfile create mode 100644 LICENSE create mode 100644 manifest.json create mode 100644 public/favicon.ico create mode 100644 public/index.html create mode 100644 public/robots.txt diff --git a/.gitea/workflows/release_build.yml b/.gitea/workflows/release_build.yml new file mode 100644 index 0000000..6a6ca31 --- /dev/null +++ b/.gitea/workflows/release_build.yml @@ -0,0 +1,383 @@ +name: Build, Push, Publish + +on: + push: + branches: + - main + workflow_dispatch: + schedule: + - cron: '28 5 * * *' + # workflow_run support in Gitea can be tricky, keeping it but might need adjustment + workflow_run: + workflows: ["Sync Repo"] + types: + - completed + +jobs: + release: + name: Build & Release + runs-on: ubuntu-latest + container: + image: catthehacker/ubuntu:act-latest + + permissions: + contents: write + packages: write + + steps: + - name: 📥 Checkout code with full history and tags + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: Check for Dockerargs and Upstream Updates + id: check_upstream + run: | + if [ -f Dockerargs ]; then + echo "Dockerargs found. Checking upstream..." + echo "Dockerargs found. Checking upstream..." + # Parse repo info using awk to avoid git config restrictions on underscores in keys elsewhere in the file + REPO_URL=$(awk -F '=' '/\[repo\]/{flag=1; next} /\[/{flag=0} flag && /^url=/{print $2}' Dockerargs | tr -d ' \r\n') + REPO_BRANCH=$(awk -F '=' '/\[repo\]/{flag=1; next} /\[/{flag=0} flag && /^branch=/{print $2}' Dockerargs | tr -d ' \r\n') + if [ -z "$REPO_BRANCH" ]; then REPO_BRANCH="main"; fi + + # Fetch upstream SHA + if [ -n "$REPO_URL" ]; then + UPSTREAM_SHA=$(git ls-remote "$REPO_URL" "$REPO_BRANCH" | awk '{ print $1 }' | head -c 7) + echo "Upstream SHA: $UPSTREAM_SHA" + + if [ -f manifest.json ]; then + LOCAL_SHA=$(jq -r '.upstream_sha // empty' manifest.json) + else + LOCAL_SHA="" + fi + + if [ "$LOCAL_SHA" != "$UPSTREAM_SHA" ]; then + echo "Upstream changed ($LOCAL_SHA -> $UPSTREAM_SHA)." + echo "upstream_needs_update=true" >> "$GITHUB_OUTPUT" + echo "upstream_sha=$UPSTREAM_SHA" >> "$GITHUB_OUTPUT" + echo "repo_url=$REPO_URL" >> "$GITHUB_OUTPUT" + echo "repo_branch=$REPO_BRANCH" >> "$GITHUB_OUTPUT" + else + echo "Upstream up to date." + echo "upstream_needs_update=false" >> "$GITHUB_OUTPUT" + fi + + # Parse Build Args + echo "Parsing [args] from Dockerargs..." + ARGS_CONTENT=$(sed -n '/^\[args\]/,/^\[/p' Dockerargs | grep -v '^\[' | grep '=' || true) + if [ -n "$ARGS_CONTENT" ]; then + echo "Found args:" + echo "$ARGS_CONTENT" + echo "build_args<> "$GITHUB_OUTPUT" + echo "$ARGS_CONTENT" >> "$GITHUB_OUTPUT" + echo "EOF" >> "$GITHUB_OUTPUT" + else + echo "No args found." + echo "build_args=" >> "$GITHUB_OUTPUT" + fi + else + echo "Repo URL not found in Dockerargs." + echo "upstream_needs_update=false" >> "$GITHUB_OUTPUT" + echo "build_args=" >> "$GITHUB_OUTPUT" + fi + else + echo "No Dockerargs found." + echo "upstream_needs_update=false" >> "$GITHUB_OUTPUT" + echo "build_args=" >> "$GITHUB_OUTPUT" + fi + + - name: Check if any tags exist + id: check_tags_exist + run: | + git fetch --tags + TAG_COUNT=$(git tag | wc -l) + if [ "$TAG_COUNT" -eq 0 ]; then + echo "has_tags=false" >> "$GITHUB_OUTPUT" + echo "latest_tag=v0.0.0" >> "$GITHUB_OUTPUT" + else + echo "has_tags=true" >> "$GITHUB_OUTPUT" + LATEST_TAG=$(git describe --tags --abbrev=0) + echo "latest_tag=$LATEST_TAG" >> "$GITHUB_OUTPUT" + fi + + - name: Check if meaningful commits exist since latest tag + id: check_commits + run: | + UPSTREAM_UPDATE="${{ steps.check_upstream.outputs.upstream_needs_update }}" + if [ "$UPSTREAM_UPDATE" == "true" ]; then + echo "commit_count=1" >> "$GITHUB_OUTPUT" + echo "changed_files=Upstream Update to ${{ steps.check_upstream.outputs.upstream_sha }}" >> "$GITHUB_OUTPUT" + elif [ "${{ steps.check_tags_exist.outputs.has_tags }}" = "false" ]; then + # No tags exist, so we should create first release + echo "commit_count=1" >> "$GITHUB_OUTPUT" + CHANGED_FILES=$(git ls-files | grep -v '^manifest.json$' || true) + if [ -n "$CHANGED_FILES" ]; then + echo "changed_files<> "$GITHUB_OUTPUT" + printf '%s\n' "$CHANGED_FILES" >> "$GITHUB_OUTPUT" + echo "EOF" >> "$GITHUB_OUTPUT" + else + echo "changed_files=Initial release" >> "$GITHUB_OUTPUT" + fi + else + LATEST_TAG="${{ steps.check_tags_exist.outputs.latest_tag }}" + CHANGED_FILES="$(git diff --name-only "${LATEST_TAG}..HEAD" | grep -v '^manifest.json$' || true)" + if [ -n "$CHANGED_FILES" ]; then + echo "commit_count=1" >> "$GITHUB_OUTPUT" + echo "changed_files<> "$GITHUB_OUTPUT" + printf '%s\n' "$CHANGED_FILES" >> "$GITHUB_OUTPUT" + echo "EOF" >> "$GITHUB_OUTPUT" + else + echo "commit_count=0" >> "$GITHUB_OUTPUT" + fi + fi + + - name: Get latest release tag (from Gitea API) + id: get_latest_release + run: | + # Using Gitea API + LATEST_RELEASE_TAG=$(curl -sL -H "Accept: application/json" \ + -H "Authorization: token ${{ secrets.GITHUB_TOKEN }}" \ + "${{ gitea.api_url }}/repos/${{ gitea.repository }}/releases/latest" | jq -r .tag_name) + + if [ -z "$LATEST_RELEASE_TAG" ] || [ "$LATEST_RELEASE_TAG" = "null" ]; then + LATEST_RELEASE_TAG="v1.0.0" + fi + echo "latest_release_tag=$LATEST_RELEASE_TAG" >> "$GITHUB_OUTPUT" + echo "latest_release_version=${LATEST_RELEASE_TAG#v}" >> "$GITHUB_OUTPUT" + + # ------------------------------- + # Sync manifest.json to last release version if behind (only when no meaningful commits) + # ------------------------------- + - name: 🛠 Ensure manifest.json matches latest release version + if: steps.check_commits.outputs.commit_count == '0' + run: | + if [ -f manifest.json ]; then + MANIFEST_VERSION=$(jq -r '.version // empty' manifest.json) + else + MANIFEST_VERSION="" + fi + LATEST_RELEASE_VERSION="${{ steps.get_latest_release.outputs.latest_release_version }}" + PYTHON_CODE="from packaging import version; \ + print(version.parse('$LATEST_RELEASE_VERSION') > version.parse('$MANIFEST_VERSION') if '$MANIFEST_VERSION' else True)" + # Python3 is available in catthehacker/ubuntu:act-latest + NEED_UPDATE=$(python3 -c "$PYTHON_CODE") + if [ "$NEED_UPDATE" = "True" ]; then + echo "Updating manifest.json to version $LATEST_RELEASE_VERSION (sync with release)" + jq --arg v "$LATEST_RELEASE_VERSION" '.version = $v' manifest.json > tmp.json && mv tmp.json manifest.json + git config user.name "Gitea Actions" + git config user.email "actions@git.icc.gg" + git add manifest.json + git commit -m "Sync manifest.json to release $LATEST_RELEASE_VERSION [🔄]" || echo "Nothing to commit" + git push origin main || true + else + echo "Manifest.json is already up-to-date with the latest release." + fi + + # ------------------------------- + # Continue normal workflow if commits exist + # ------------------------------- + - name: 📃 Get list of changed files (Markdown bullet list) + if: steps.check_commits.outputs.commit_count != '0' + id: changed_files + run: | + BULLET_LIST="$(printf '%s\n' "${{ steps.check_commits.outputs.changed_files }}" | sed 's/^/- /')" + echo "CHANGED<> "$GITHUB_OUTPUT" + printf '%s\n' "$BULLET_LIST" >> "$GITHUB_OUTPUT" + echo "EOF" >> "$GITHUB_OUTPUT" + COUNT="$(printf '%s\n' "${{ steps.check_commits.outputs.changed_files }}" | wc -l)" + echo "COUNT=$COUNT" >> "$GITHUB_OUTPUT" + + - name: Get manifest version + if: steps.check_commits.outputs.commit_count != '0' + id: get_manifest_version + run: | + if [ -f manifest.json ]; then + MANIFEST_VERSION=$(jq -r '.version // empty' manifest.json) + if [ -z "$MANIFEST_VERSION" ] || [ "$MANIFEST_VERSION" = "null" ]; then + MANIFEST_VERSION="1.0.0" + fi + else + MANIFEST_VERSION="1.0.0" + fi + echo "manifest_version=$MANIFEST_VERSION" >> "$GITHUB_OUTPUT" + + - name: Pick base version + if: steps.check_commits.outputs.commit_count != '0' + id: pick_base_version + run: | + LATEST_RELEASE="${{ steps.get_latest_release.outputs.latest_release_version }}" + MANIFEST="${{ steps.get_manifest_version.outputs.manifest_version }}" + BASE_VERSION=$(python3 -c "from packaging import version; \ + print(str(max(version.parse('$LATEST_RELEASE'), version.parse('$MANIFEST'))))") + echo "base_version=$BASE_VERSION" >> "$GITHUB_OUTPUT" + + - name: 🔢 Determine version + if: steps.check_commits.outputs.commit_count != '0' + id: version + run: | + BASE_VERSION="${{ steps.pick_base_version.outputs.base_version }}" + MAJOR=$(echo "$BASE_VERSION" | cut -d. -f1) + MINOR=$(echo "$BASE_VERSION" | cut -d. -f2) + PATCH=$(echo "$BASE_VERSION" | cut -d. -f3) + COUNT="${{ steps.changed_files.outputs.COUNT }}" + if [ "$COUNT" -ge 5 ]; then + MAJOR=$((MAJOR + 1)) + MINOR=0 + PATCH=0 + elif [ "$COUNT" -ge 3 ]; then + MINOR=$((MINOR + 1)) + PATCH=0 + else + PATCH=$((PATCH + 1)) + fi + NEW_VERSION="${MAJOR}.${MINOR}.${PATCH}" + REPO_NAME="$(basename "$GITHUB_REPOSITORY")" + ZIP_NAME="${REPO_NAME}-${NEW_VERSION}.zip" + echo "VERSION=$NEW_VERSION" >> "$GITHUB_OUTPUT" + echo "ZIP_NAME=$ZIP_NAME" >> "$GITHUB_OUTPUT" + echo "REPO_NAME=$REPO_NAME" >> "$GITHUB_OUTPUT" + + - name: 🛠 Update or create manifest.json + if: steps.check_commits.outputs.commit_count != '0' + run: | + VERSION="${{ steps.version.outputs.VERSION }}" + AUTHOR="Ivan Carlos" + VERSION_FILE="manifest.json" + UPSTREAM_SHA="${{ steps.check_upstream.outputs.upstream_sha }}" + + if [ -f "$VERSION_FILE" ]; then + jq --arg v "$VERSION" \ + --arg a "$AUTHOR" \ + --arg u "$UPSTREAM_SHA" \ + '.version = $v | .author = $a | if $u != "" and $u != null then .upstream_sha = $u else . end' \ + "$VERSION_FILE" > tmp.json && mv tmp.json "$VERSION_FILE" + else + echo "{ \"version\": \"$VERSION\", \"author\": \"$AUTHOR\", \"upstream_sha\": \"$UPSTREAM_SHA\" }" > "$VERSION_FILE" + fi + + - name: 💾 Commit and push updated manifest.json + if: steps.check_commits.outputs.commit_count != '0' + run: | + git config user.name "Gitea Actions" + git config user.email "actions@git.icc.gg" + git add manifest.json + git commit -m "Update manifest version to ${{ steps.version.outputs.VERSION }} [▶️]" || echo "Nothing to commit" + git push origin main + + - name: 📦 Create ZIP package (excluding certain files) + if: steps.check_commits.outputs.commit_count != '0' + run: | + ZIP_NAME="${{ steps.version.outputs.ZIP_NAME }}" + zip -r "$ZIP_NAME" . -x ".git/*" ".github/*" "docker/*" ".dockerignore" "CNAME" "Dockerfile" "README.md" "LICENSE" ".gitea/*" + + - name: 🚀 Create Gitea Release + if: steps.check_commits.outputs.commit_count != '0' + id: create_release + env: + CHANGELOG_LIST: ${{ steps.changed_files.outputs.CHANGED }} + run: | + TAG_NAME="v${{ steps.version.outputs.VERSION }}" + RELEASE_NAME="${{ steps.version.outputs.REPO_NAME }} v${{ steps.version.outputs.VERSION }}" + + # Construct Markdown body safely using env var + # We use printf to avoid interpreting backslashes in the file list + BODY=$(printf "### Changelog\nFiles changed in this release:\n%s" "$CHANGELOG_LIST") + + # Create JSON payload using jq + jq -n \ + --arg tag_name "$TAG_NAME" \ + --arg name "$RELEASE_NAME" \ + --arg body "$BODY" \ + '{tag_name: $tag_name, name: $name, body: $body, draft: false, prerelease: false}' > release_payload.json + + echo "DEBUG: Generated Payload:" + cat release_payload.json + + # Create Release + curl -s -X POST "${{ gitea.api_url }}/repos/${{ gitea.repository }}/releases" \ + -H "Authorization: token ${{ secrets.GITHUB_TOKEN }}" \ + -H "Content-Type: application/json" \ + -d @release_payload.json > api_response.json + + echo "DEBUG: API Response:" + cat api_response.json || true + + RELEASE_ID=$(jq -r .id api_response.json) + echo "RELEASE_ID=$RELEASE_ID" >> "$GITHUB_OUTPUT" + + if [ "$RELEASE_ID" == "null" ] || [ -z "$RELEASE_ID" ]; then + echo "Failed to create release. Response content:" + cat api_response.json + exit 1 + fi + + - name: 📤 Upload Release Asset + if: steps.check_commits.outputs.commit_count != '0' + run: | + RELEASE_ID="${{ steps.create_release.outputs.RELEASE_ID }}" + ZIP_NAME="${{ steps.version.outputs.ZIP_NAME }}" + FILE_PATH="./$ZIP_NAME" + + curl -s -X POST "${{ gitea.api_url }}/repos/${{ gitea.repository }}/releases/$RELEASE_ID/assets" \ + -H "Authorization: token ${{ secrets.GITHUB_TOKEN }}" \ + -H "Content-Type: application/zip" \ + --data-binary @"$FILE_PATH" \ + -o /dev/null + + # ----- Docker steps ----- + - name: Clone Upstream Code (if needed) + if: steps.check_commits.outputs.commit_count != '0' && (steps.check_upstream.outputs.upstream_needs_update == 'true' || steps.check_upstream.outputs.repo_url != '') + run: | + rm -rf upstream_src + git clone --depth 1 --branch ${{ steps.check_upstream.outputs.repo_branch }} ${{ steps.check_upstream.outputs.repo_url }} upstream_src + + - name: 🔍 Check if Dockerfile exists + if: steps.check_commits.outputs.commit_count != '0' || steps.check_upstream.outputs.upstream_needs_update == 'true' + id: dockerfile_check + run: | + if [ -n "${{ steps.check_upstream.outputs.repo_url }}" ]; then + if [ -f upstream_src/Dockerfile ]; then + echo "exists=true" >> "$GITHUB_OUTPUT" + else + # Fallback or error? User said "ignore", but we need a dockerfile to build. + # Assuming if upstream_src is present, we trust it, or fail at build time. + # Let's say exists=true and let build fail if missing, per user hint. + echo "exists=true" >> "$GITHUB_OUTPUT" + fi + elif [ -f Dockerfile ]; then + echo "exists=true" >> "$GITHUB_OUTPUT" + else + echo "exists=false" >> "$GITHUB_OUTPUT" + fi + + - name: 🔐 Login to Gitea Container Registry + if: steps.check_commits.outputs.commit_count != '0' && steps.dockerfile_check.outputs.exists == 'true' + uses: docker/login-action@v3 + with: + registry: git.icc.gg + username: ${{ gitea.actor }} + password: ${{ secrets.CR_PAT }} + + - name: 🛠 Set up QEMU + if: steps.check_commits.outputs.commit_count != '0' && steps.dockerfile_check.outputs.exists == 'true' + uses: docker/setup-qemu-action@v3 + + - name: 🛠 Set up Docker Buildx + if: steps.check_commits.outputs.commit_count != '0' && steps.dockerfile_check.outputs.exists == 'true' + uses: docker/setup-buildx-action@v3 + + - name: 🐳 Build and Push Docker image + if: steps.check_commits.outputs.commit_count != '0' && steps.dockerfile_check.outputs.exists == 'true' + uses: docker/build-push-action@v5 + id: docker_build + with: + context: ${{ steps.check_upstream.outputs.repo_url != '' && './upstream_src' || '.' }} + platforms: linux/amd64,linux/arm64 + file: ${{ steps.check_upstream.outputs.repo_url != '' && './upstream_src/Dockerfile' || './Dockerfile' }} + push: true + build-args: | + ${{ steps.check_upstream.outputs.build_args }} + tags: | + git.icc.gg/${{ gitea.repository }}:latest + git.icc.gg/${{ gitea.repository }}:${{ steps.version.outputs.VERSION }} diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..2e6a965 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,3 @@ +FROM nginx:alpine + +COPY public/ /usr/share/nginx/html diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..6603463 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2025 Ivan Carlos + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/manifest.json b/manifest.json new file mode 100644 index 0000000..eca6cda --- /dev/null +++ b/manifest.json @@ -0,0 +1,4 @@ +{ + "version": "1.0.0", + "author": "Ivan Carlos" +} diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..ae3794588b467ed2e7bddef13a6cc62b5ace7fa5 GIT binary patch literal 15086 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 literal 0 HcmV?d00001 diff --git a/public/index.html b/public/index.html new file mode 100644 index 0000000..8ea4b7e --- /dev/null +++ b/public/index.html @@ -0,0 +1,354 @@ + + + + + + + AWS SES SMTP Password Generator + + + + + + + +
+

AWS SES SMTP Generator

+
+ Securely convert your AWS Secret Access Key into an SES SMTP password locally in your browser. +
+ +
+
+ +
+ +
+
+ +
+ + +
+
+ + + + + +
+ + + + + + \ No newline at end of file diff --git a/public/robots.txt b/public/robots.txt new file mode 100644 index 0000000..c2a49f4 --- /dev/null +++ b/public/robots.txt @@ -0,0 +1,2 @@ +User-agent: * +Allow: /