|
| 1 | +# Runs only when production code under src/ changes. Version must be > latest v* tag (not vs base branch). |
| 2 | + |
| 3 | +name: Check Version Bump |
| 4 | + |
| 5 | +on: |
| 6 | + pull_request: |
| 7 | + |
| 8 | +jobs: |
| 9 | + check-version-bump: |
| 10 | + runs-on: ubuntu-latest |
| 11 | + steps: |
| 12 | + - uses: actions/checkout@v4 |
| 13 | + with: |
| 14 | + fetch-depth: 0 |
| 15 | + - name: Validate version and changelog updates |
| 16 | + shell: bash |
| 17 | + run: | |
| 18 | + set -euo pipefail |
| 19 | +
|
| 20 | + VERSION_FILE="package.json" |
| 21 | + CHANGELOG_FILE="CHANGELOG.md" |
| 22 | + BASE_SHA="${{ github.event.pull_request.base.sha }}" |
| 23 | + HEAD_SHA="${{ github.event.pull_request.head.sha }}" |
| 24 | +
|
| 25 | + mapfile -t CHANGED_FILES < <(git diff --name-only "$BASE_SHA" "$HEAD_SHA") |
| 26 | + if [ "${#CHANGED_FILES[@]}" -eq 0 ]; then |
| 27 | + echo "No changed files detected." |
| 28 | + exit 0 |
| 29 | + fi |
| 30 | +
|
| 31 | + is_production_source_change() { |
| 32 | + local f="$1" |
| 33 | + [[ "$f" == src/* ]] |
| 34 | + } |
| 35 | +
|
| 36 | + has_source_changes=false |
| 37 | + for file in "${CHANGED_FILES[@]}"; do |
| 38 | + if is_production_source_change "$file"; then |
| 39 | + has_source_changes=true |
| 40 | + break |
| 41 | + fi |
| 42 | + done |
| 43 | +
|
| 44 | + if [ "$has_source_changes" = false ]; then |
| 45 | + echo "Skipping: no src/ production code changes." |
| 46 | + exit 0 |
| 47 | + fi |
| 48 | +
|
| 49 | + changed_file() { |
| 50 | + local target="$1" |
| 51 | + for file in "${CHANGED_FILES[@]}"; do |
| 52 | + if [ "$file" = "$target" ]; then |
| 53 | + return 0 |
| 54 | + fi |
| 55 | + done |
| 56 | + return 1 |
| 57 | + } |
| 58 | +
|
| 59 | + changed_file "$VERSION_FILE" || { echo "Version bump required in $VERSION_FILE."; exit 1; } |
| 60 | + changed_file "$CHANGELOG_FILE" || { echo "Matching changelog update required in $CHANGELOG_FILE."; exit 1; } |
| 61 | +
|
| 62 | + head_version=$(node -e "console.log(require('./package.json').version)") |
| 63 | + CHANGELOG_HEAD=$(sed -nE 's/^## v?([^[:space:]]+).*/\1/p' "$CHANGELOG_FILE" | head -1) |
| 64 | +
|
| 65 | + [ -n "$CHANGELOG_HEAD" ] || { echo "::error::Could not find a top changelog heading like '## vX.Y.Z' in $CHANGELOG_FILE."; exit 1; } |
| 66 | + [ "$CHANGELOG_HEAD" = "$head_version" ] || { echo "::error::$CHANGELOG_FILE top version ($CHANGELOG_HEAD) does not match project version ($head_version)."; exit 1; } |
| 67 | +
|
| 68 | + latest_tag=$(git tag --list 'v*' --sort=-version:refname | sed -n '1p') |
| 69 | + latest_version="${latest_tag#v}" |
| 70 | + [ -n "$latest_version" ] || latest_version="0.0.0" |
| 71 | +
|
| 72 | + version_gt() { |
| 73 | + python3 -c 'import sys;v=lambda s:[int(x) if x.isdigit() else 0 for x in (s.strip().lstrip("v").split("-",1)[0].split("+",1)[0].split(".")+["0","0","0"])[:3]];print("true" if v(sys.argv[1])>v(sys.argv[2]) else "false")' "$1" "$2" |
| 74 | + } |
| 75 | +
|
| 76 | + [ "$(version_gt "$head_version" "$latest_version")" = "true" ] || { echo "Version must be greater than latest tag version ($latest_version). Found $head_version."; exit 1; } |
0 commit comments