refactor: normalize user names in GetUsersList #2568
This file contains hidden or 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
| name: Build and Publish | |
| on: | |
| workflow_dispatch: | |
| inputs: | |
| logLevel: | |
| description: "Log level" | |
| required: true | |
| default: "warning" | |
| type: choice | |
| options: | |
| - info | |
| - warning | |
| - debug | |
| build: | |
| description: "Build" | |
| required: false | |
| default: false | |
| type: boolean | |
| update_readme: | |
| description: "Update Readme" | |
| required: false | |
| default: false | |
| type: boolean | |
| push: | |
| branches: | |
| - master | |
| - dev | |
| tags: | |
| - "v*" | |
| paths-ignore: | |
| - "**.md" | |
| - ".github/ISSUE_TEMPLATE/**" | |
| pull_request: | |
| branches: | |
| - master | |
| - dev | |
| paths-ignore: | |
| - "**.md" | |
| - ".github/ISSUE_TEMPLATE/**" | |
| env: | |
| REGISTRY: ${{ vars.REGISTRY != '' && vars.REGISTRY || '' }} | |
| BUILD_AMD64: ${{ vars.BUILD_AMD64 != 'false' && 'true' || 'false' }} | |
| BUILD_ARM64: ${{ vars.BUILD_ARM64 != 'false' && 'true' || 'false' }} | |
| DOCKERHUB_SLUG: arabcoders/watchstate | |
| GHCR_SLUG: ghcr.io/arabcoders/watchstate | |
| BUN_VERSION: latest | |
| NODE_VERSION: 20 | |
| jobs: | |
| validate-build-config: | |
| name: Validate Build Configuration | |
| runs-on: ubuntu-latest | |
| permissions: {} | |
| steps: | |
| - name: Ensure at least one architecture is enabled | |
| run: | | |
| if [ "${{ env.BUILD_AMD64 }}" = "false" ] && [ "${{ env.BUILD_ARM64 }}" = "false" ]; then | |
| echo "::error::Both BUILD_AMD64 and BUILD_ARM64 are disabled. Enable at least one architecture." | |
| exit 1 | |
| fi | |
| echo "Build configuration is valid" | |
| echo "BUILD_AMD64: ${{ env.BUILD_AMD64 }}" | |
| echo "BUILD_ARM64: ${{ env.BUILD_ARM64 }}" | |
| test: | |
| name: Tests & Frontend Build | |
| runs-on: ubuntu-latest | |
| needs: [validate-build-config] | |
| permissions: | |
| contents: read | |
| strategy: | |
| fail-fast: true | |
| matrix: | |
| php: [8.4] | |
| steps: | |
| - name: Checkout | |
| uses: actions/checkout@v4 | |
| with: | |
| fetch-depth: 0 | |
| - name: Update Version File | |
| uses: arabcoders/write-version-to-file@master | |
| with: | |
| filename: "/config/config.php" | |
| placeholder: "$(version_via_ci)" | |
| with_date: "true" | |
| with_branch: "true" | |
| - name: Setup PHP | |
| uses: shivammathur/setup-php@v2 | |
| with: | |
| php-version: ${{ matrix.php }} | |
| extensions: pdo, mbstring, ctype, curl, sqlite3 | |
| coverage: none | |
| tools: composer:v2 | |
| - name: Get composer cache directory | |
| id: composer-cache | |
| run: echo "COMPOSER_CACHE_DIR=$(composer config cache-files-dir)" >> "$GITHUB_OUTPUT" | |
| - name: Restore cached dependencies | |
| uses: actions/cache@v4 | |
| with: | |
| path: ${{ steps.composer-cache.outputs.COMPOSER_CACHE_DIR }} | |
| key: "${{ matrix.php }}-composer-${{ hashFiles('**/composer.lock') }}" | |
| restore-keys: ${{ matrix.php }}-composer- | |
| - name: Install PHP dependencies | |
| run: composer install --prefer-dist --no-interaction --no-progress | |
| - name: Run PHP formatter check | |
| run: composer run format:ci | |
| - name: Run PHP linter | |
| run: composer run lint | |
| - name: Run PHP tests | |
| run: composer run test | |
| - uses: actions/cache@v4 | |
| with: | |
| path: ~/.bun/install/cache | |
| key: ${{ runner.os }}-bun-${{ hashFiles('**/bun.lock') }} | |
| restore-keys: | | |
| ${{ runner.os }}-bun- | |
| - name: Install Node.js | |
| uses: actions/setup-node@v4 | |
| with: | |
| node-version: ${{ env.NODE_VERSION }} | |
| - name: Install bun | |
| uses: oven-sh/setup-bun@v2 | |
| with: | |
| bun-version: ${{ env.BUN_VERSION }} | |
| - name: Install frontend dependencies | |
| working-directory: frontend | |
| run: bun install --frozen-lockfile --prefer-offline | |
| - name: Run nuxt prepare | |
| working-directory: frontend | |
| env: | |
| NODE_ENV: development | |
| BUN_ENV: development | |
| run: bun run prepare:test | |
| - name: Run frontend linter | |
| working-directory: frontend | |
| run: bun run lint | |
| - name: Run frontend typecheck | |
| working-directory: frontend | |
| run: bun run lint:tsc | |
| - name: Build frontend | |
| working-directory: frontend | |
| env: | |
| NODE_ENV: production | |
| BUN_ENV: production | |
| run: bun run generate | |
| - name: Set up Python | |
| if: env.REGISTRY == '' && github.event_name != 'pull_request' | |
| uses: actions/setup-python@v5 | |
| with: | |
| python-version: "3.13" | |
| - name: Install GitPython | |
| if: env.REGISTRY == '' && github.event_name != 'pull_request' | |
| run: pip install gitpython | |
| - name: Generate CHANGELOG.json | |
| if: env.REGISTRY == '' && github.event_name != 'pull_request' | |
| run: | | |
| python3 .github/scripts/generate_changelog.py -p . -f ./frontend/exported/CHANGELOG.json | |
| - name: Upload frontend build | |
| if: env.REGISTRY == '' && github.event_name != 'pull_request' | |
| uses: actions/upload-artifact@v4 | |
| with: | |
| name: frontend-build | |
| path: frontend/exported/ | |
| retention-days: 1 | |
| docker-build-amd64: | |
| name: Build Container (amd64) | |
| runs-on: ubuntu-latest | |
| needs: [test, validate-build-config] | |
| if: (github.event_name == 'push' || (github.event_name == 'workflow_dispatch' && github.event.inputs.build)) && vars.BUILD_AMD64 != 'false' | |
| permissions: | |
| packages: write | |
| contents: write | |
| env: | |
| ARCH: amd64 | |
| steps: | |
| - name: Checkout | |
| uses: actions/checkout@v4 | |
| with: | |
| fetch-depth: 0 | |
| - name: Download frontend build | |
| if: env.REGISTRY == '' | |
| uses: actions/download-artifact@v4 | |
| with: | |
| name: frontend-build | |
| path: frontend/exported/ | |
| - name: Set up Docker Buildx | |
| uses: docker/setup-buildx-action@v3 | |
| - name: Update config/config.php | |
| run: | | |
| VERSION="${GITHUB_REF##*/}" | |
| SHA=$(git rev-parse HEAD) | |
| BRANCH=$(git rev-parse --abbrev-ref HEAD) | |
| DATE=$(date -u +"%Y%m%d") | |
| echo "Current version: ${VERSION}, SHA: ${SHA}, Date: ${DATE}" | |
| echo "APP_VERSION=${VERSION}" >> "$GITHUB_ENV" | |
| echo "APP_SHA=${SHA}" >> "$GITHUB_ENV" | |
| echo "APP_DATE=${DATE}" >> "$GITHUB_ENV" | |
| echo "APP_BRANCH=${BRANCH}" >> "$GITHUB_ENV" | |
| sed -i \ | |
| -e "s/'version' => 'dev-master'/'version' => '${VERSION}'/" \ | |
| -e "s/'version_sha' => 'unknown'/'version_sha' => '${SHA}'/" \ | |
| -e "s/'version_build' => 'unknown'/'version_build' => '${DATE}'/" \ | |
| -e "s/'version_branch' => 'unknown'/'version_branch' => '${BRANCH}'/" \ | |
| config/config.php | |
| - name: Docker meta | |
| id: meta | |
| uses: docker/metadata-action@v5 | |
| with: | |
| images: | | |
| name=${{ env.DOCKERHUB_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.GHCR_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.REGISTRY }}/${{ github.repository }},enable=${{ env.REGISTRY != '' }} | |
| flavor: | | |
| latest=false | |
| suffix=-${{ env.ARCH }} | |
| tags: | | |
| type=ref,event=branch | |
| type=ref,event=tag | |
| type=raw,value=latest,enable=${{ startsWith(github.ref, 'refs/tags/v') }} | |
| - name: Login to Private Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY != '' | |
| with: | |
| registry: ${{ env.REGISTRY }} | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GIT_TOKEN && secrets.GIT_TOKEN || secrets.GITHUB_TOKEN }} | |
| - name: Login to GitHub Container Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| registry: ghcr.io | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GITHUB_TOKEN }} | |
| - name: Login to DockerHub | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| username: ${{ secrets.DOCKERHUB_USERNAME }} | |
| password: ${{ secrets.DOCKERHUB_TOKEN }} | |
| - name: Build and push | |
| uses: docker/build-push-action@v5 | |
| with: | |
| context: . | |
| platforms: linux/${{ env.ARCH }} | |
| push: true | |
| tags: ${{ steps.meta.outputs.tags }} | |
| labels: ${{ steps.meta.outputs.labels }} | |
| cache-from: type=gha,scope=${{ github.workflow_ref }}-${{ env.ARCH }} | |
| cache-to: type=gha,mode=max,scope=${{ github.workflow_ref }}-${{ env.ARCH }} | |
| provenance: true | |
| docker-build-arm64: | |
| name: Build Container (arm64) | |
| runs-on: ubuntu-latest | |
| needs: [test, validate-build-config] | |
| if: (github.event_name == 'push' || (github.event_name == 'workflow_dispatch' && github.event.inputs.build)) && vars.BUILD_ARM64 != 'false' | |
| permissions: | |
| packages: write | |
| contents: write | |
| env: | |
| ARCH: arm64 | |
| steps: | |
| - name: Checkout | |
| uses: actions/checkout@v4 | |
| with: | |
| fetch-depth: 0 | |
| - name: Download frontend build | |
| if: env.REGISTRY == '' | |
| uses: actions/download-artifact@v4 | |
| with: | |
| name: frontend-build | |
| path: frontend/exported/ | |
| - name: Set up Docker Buildx | |
| uses: docker/setup-buildx-action@v3 | |
| - name: Update config/config.php | |
| run: | | |
| VERSION="${GITHUB_REF##*/}" | |
| SHA=$(git rev-parse HEAD) | |
| BRANCH=$(git rev-parse --abbrev-ref HEAD) | |
| DATE=$(date -u +"%Y%m%d") | |
| echo "Current version: ${VERSION}, SHA: ${SHA}, Date: ${DATE}" | |
| echo "APP_VERSION=${VERSION}" >> "$GITHUB_ENV" | |
| echo "APP_SHA=${SHA}" >> "$GITHUB_ENV" | |
| echo "APP_DATE=${DATE}" >> "$GITHUB_ENV" | |
| echo "APP_BRANCH=${BRANCH}" >> "$GITHUB_ENV" | |
| sed -i \ | |
| -e "s/'version' => 'dev-master'/'version' => '${VERSION}'/" \ | |
| -e "s/'version_sha' => 'unknown'/'version_sha' => '${SHA}'/" \ | |
| -e "s/'version_build' => 'unknown'/'version_build' => '${DATE}'/" \ | |
| -e "s/'version_branch' => 'unknown'/'version_branch' => '${BRANCH}'/" \ | |
| config/config.php | |
| - name: Docker meta | |
| id: meta | |
| uses: docker/metadata-action@v5 | |
| with: | |
| images: | | |
| name=${{ env.DOCKERHUB_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.GHCR_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.REGISTRY }}/${{ github.repository }},enable=${{ env.REGISTRY != '' }} | |
| flavor: | | |
| latest=false | |
| suffix=-${{ env.ARCH }} | |
| tags: | | |
| type=ref,event=branch | |
| type=ref,event=tag | |
| type=raw,value=latest,enable=${{ startsWith(github.ref, 'refs/tags/v') }} | |
| - name: Login to Private Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY != '' | |
| with: | |
| registry: ${{ env.REGISTRY }} | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GIT_TOKEN && secrets.GIT_TOKEN || secrets.GITHUB_TOKEN }} | |
| - name: Login to GitHub Container Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| registry: ghcr.io | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GITHUB_TOKEN }} | |
| - name: Login to DockerHub | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| username: ${{ secrets.DOCKERHUB_USERNAME }} | |
| password: ${{ secrets.DOCKERHUB_TOKEN }} | |
| - name: Build and push | |
| uses: docker/build-push-action@v5 | |
| with: | |
| context: . | |
| platforms: linux/${{ env.ARCH }} | |
| push: true | |
| tags: ${{ steps.meta.outputs.tags }} | |
| labels: ${{ steps.meta.outputs.labels }} | |
| cache-from: type=gha,scope=${{ github.workflow }}-${{ env.ARCH }} | |
| cache-to: type=gha,mode=max,scope=${{ github.workflow }}-${{ env.ARCH }} | |
| provenance: true | |
| docker-publish-manifest: | |
| name: Publish multi-arch manifest | |
| runs-on: ubuntu-latest | |
| needs: [docker-build-amd64, docker-build-arm64] | |
| if: | | |
| !cancelled() && github.event_name != 'pull_request' && | |
| (needs.docker-build-amd64.result == 'success' || needs.docker-build-arm64.result == 'success') | |
| permissions: | |
| packages: write | |
| contents: write | |
| env: | |
| AMD64_RESULT: ${{ needs.docker-build-amd64.result }} | |
| ARM64_RESULT: ${{ needs.docker-build-arm64.result }} | |
| steps: | |
| - name: Docker meta | |
| id: meta | |
| uses: docker/metadata-action@v5 | |
| with: | |
| images: | | |
| name=${{ env.DOCKERHUB_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.GHCR_SLUG }},enable=${{ env.REGISTRY == '' }} | |
| name=${{ env.REGISTRY }}/${{ github.repository }},enable=${{ env.REGISTRY != '' }} | |
| flavor: | | |
| latest=false | |
| tags: | | |
| type=ref,event=branch | |
| type=ref,event=tag | |
| type=raw,value=latest,enable=${{ startsWith(github.ref, 'refs/tags/v') }} | |
| - name: Login to Private Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY != '' | |
| with: | |
| registry: ${{ env.REGISTRY }} | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GIT_TOKEN && secrets.GIT_TOKEN || secrets.GITHUB_TOKEN }} | |
| - name: Login to GitHub Container Registry | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| registry: ghcr.io | |
| username: ${{ github.actor }} | |
| password: ${{ secrets.GITHUB_TOKEN }} | |
| - name: Login to DockerHub | |
| uses: docker/login-action@v3 | |
| if: env.REGISTRY == '' | |
| with: | |
| username: ${{ secrets.DOCKERHUB_USERNAME }} | |
| password: ${{ secrets.DOCKERHUB_TOKEN }} | |
| - name: Create and push manifest list | |
| shell: bash | |
| run: | | |
| IFS=$'\n' | |
| BUILD_AMD64="${{ env.BUILD_AMD64 }}" | |
| BUILD_ARM64="${{ env.BUILD_ARM64 }}" | |
| AMD64_RESULT="${AMD64_RESULT}" | |
| ARM64_RESULT="${ARM64_RESULT}" | |
| for tag in $(echo "${{ steps.meta.outputs.tags }}"); do | |
| echo "Creating manifest for ${tag}" | |
| IMAGES=() | |
| if [ "${BUILD_AMD64}" = "true" ] && [ "${AMD64_RESULT}" = "success" ]; then | |
| IMAGES+=("${tag}-amd64") | |
| fi | |
| if [ "${BUILD_ARM64}" = "true" ] && [ "${ARM64_RESULT}" = "success" ]; then | |
| IMAGES+=("${tag}-arm64") | |
| fi | |
| if [ ${#IMAGES[@]} -eq 0 ]; then | |
| echo "::error::No images available for manifest creation" | |
| exit 1 | |
| fi | |
| docker buildx imagetools create --tag "${tag}" "${IMAGES[@]}" | |
| done | |
| - name: Overwrite GitHub release notes | |
| if: startsWith(github.ref, 'refs/tags/v') && env.REGISTRY == '' | |
| uses: actions/github-script@v6 | |
| with: | |
| github-token: ${{ secrets.GITHUB_TOKEN }} | |
| script: | | |
| const tag = context.ref.replace('refs/tags/', ''); | |
| const { data: releases } = await github.rest.repos.listReleases({ | |
| owner: context.repo.owner, | |
| repo: context.repo.repo, | |
| }); | |
| const release = releases.find(r => r.tag_name === tag); | |
| if (!release) { | |
| core.setFailed(`Release with tag ${tag} not found`); | |
| return; | |
| } | |
| const baseTag = releases[1]?.tag_name || ''; | |
| if (!baseTag) { | |
| core.setFailed('Could not determine a base tag (no previous release found)'); | |
| return; | |
| } | |
| const { data: comparison } = await github.rest.repos.compareCommits({ | |
| owner: context.repo.owner, | |
| repo: context.repo.repo, | |
| base: baseTag, | |
| head: tag, | |
| }); | |
| const commits = comparison.commits.filter(c => c.parents.length === 1); | |
| const changelog = commits | |
| .map(c => `- ${c.sha.substring(0, 7)} ${c.commit.message.split('\n')[0]}`) | |
| .join('\n'); | |
| if (!changelog) { | |
| core.setFailed('No commits found for the changelog'); | |
| return; | |
| } | |
| const existingBody = (release.body || '').trim(); | |
| const separator = `\n\n---\n\n## Commits since ${baseTag}\n\n`; | |
| const newBody = existingBody | |
| ? `${existingBody}${separator}${changelog}` | |
| : `## Commits since ${baseTag}\n\n${changelog}`; | |
| console.log(`Updating release ${tag}. Appending ${commits.length} commits under existing body.`); | |
| await github.rest.repos.updateRelease({ | |
| owner: context.repo.owner, | |
| repo: context.repo.repo, | |
| release_id: release.id, | |
| body: newBody, | |
| }); | |
| dockerhub-sync-readme: | |
| name: DockerHub README sync | |
| runs-on: ubuntu-latest | |
| permissions: | |
| contents: read | |
| if: vars.REGISTRY == '' && ((github.event_name == 'push' && startsWith(github.ref, 'refs/tags/v')) || (github.event_name == 'workflow_dispatch' && github.event.inputs.update_readme == 'true')) | |
| steps: | |
| - name: Sync README | |
| uses: docker://lsiodev/readme-sync:latest | |
| env: | |
| DOCKERHUB_USERNAME: ${{ secrets.DOCKERHUB_USERNAME }} | |
| DOCKERHUB_PASSWORD: ${{ secrets.DOCKERHUB_TOKEN }} | |
| GIT_REPOSITORY: ${{ github.repository }} | |
| DOCKER_REPOSITORY: ${{ env.DOCKERHUB_SLUG }} | |
| GIT_BRANCH: ${{ github.event.repository.default_branch }} | |
| with: | |
| entrypoint: node | |
| args: /opt/docker-readme-sync/sync |