|
| 1 | +name: Update Releases Properties |
| 2 | + |
| 3 | +# This workflow runs in individual module repositories (e.g., module-php) |
| 4 | +# It automatically updates releases.properties file when a release is published |
| 5 | + |
| 6 | +on: |
| 7 | + release: |
| 8 | + types: [prereleased, released, edited] |
| 9 | + |
| 10 | + # Manual trigger for testing |
| 11 | + workflow_dispatch: |
| 12 | + inputs: |
| 13 | + release_tag: |
| 14 | + description: 'Release tag to process (e.g., 2025.10.31)' |
| 15 | + required: true |
| 16 | + |
| 17 | +jobs: |
| 18 | + update-properties: |
| 19 | + runs-on: ubuntu-latest |
| 20 | + |
| 21 | + steps: |
| 22 | + - name: Checkout repository |
| 23 | + uses: actions/checkout@v4 |
| 24 | + with: |
| 25 | + token: ${{ secrets.GH_PAT }} |
| 26 | + |
| 27 | + - name: Set up Python |
| 28 | + uses: actions/setup-python@v5 |
| 29 | + with: |
| 30 | + python-version: '3.11' |
| 31 | + |
| 32 | + - name: Install dependencies |
| 33 | + run: | |
| 34 | + pip install requests packaging |
| 35 | + |
| 36 | + - name: Extract module name from repository |
| 37 | + id: extract_module |
| 38 | + run: | |
| 39 | + # Extract module name from repo name (e.g., "module-php" -> "php") |
| 40 | + REPO_NAME="${{ github.event.repository.name }}" |
| 41 | + echo "Repository name: ${REPO_NAME}" |
| 42 | + |
| 43 | + # Remove "module-" prefix (case-insensitive) |
| 44 | + if [[ "${REPO_NAME}" == module-* ]]; then |
| 45 | + MODULE_NAME="${REPO_NAME#module-}" |
| 46 | + elif [[ "${REPO_NAME}" == Module-* ]]; then |
| 47 | + MODULE_NAME="${REPO_NAME#Module-}" |
| 48 | + else |
| 49 | + echo "ERROR: Repository name does not start with 'module-'" |
| 50 | + echo "Expected format: module-MODULENAME (e.g., module-php)" |
| 51 | + exit 1 |
| 52 | + fi |
| 53 | + |
| 54 | + # Convert to lowercase |
| 55 | + MODULE_NAME=$(echo "${MODULE_NAME}" | tr '[:upper:]' '[:lower:]') |
| 56 | + |
| 57 | + echo "module_name=${MODULE_NAME}" >> $GITHUB_OUTPUT |
| 58 | + echo "Module name: ${MODULE_NAME}" |
| 59 | + |
| 60 | + - name: Update releases properties |
| 61 | + env: |
| 62 | + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |
| 63 | + RELEASE_TAG: ${{ github.event.release.tag_name || github.event.inputs.release_tag }} |
| 64 | + REPO_OWNER: ${{ github.repository_owner }} |
| 65 | + REPO_NAME: ${{ github.event.repository.name }} |
| 66 | + MODULE_NAME: ${{ steps.extract_module.outputs.module_name }} |
| 67 | + run: | |
| 68 | + python << 'EOF' |
| 69 | + import os |
| 70 | + import re |
| 71 | + import requests |
| 72 | + from packaging import version |
| 73 | + from collections import OrderedDict |
| 74 | + |
| 75 | + # Get environment variables |
| 76 | + release_tag = os.environ['RELEASE_TAG'] |
| 77 | + repo_owner = os.environ['REPO_OWNER'] |
| 78 | + repo_name = os.environ['REPO_NAME'] |
| 79 | + module_name = os.environ['MODULE_NAME'] |
| 80 | + github_token = os.environ['GITHUB_TOKEN'] |
| 81 | + |
| 82 | + print(f"Processing release: {release_tag}") |
| 83 | + print(f"Module name: {module_name}") |
| 84 | + |
| 85 | + # Construct API URL |
| 86 | + api_url = f"https://api.github.com/repos/{repo_owner}/{repo_name}/releases/tags/{release_tag}" |
| 87 | + |
| 88 | + headers = { |
| 89 | + 'Authorization': f'token {github_token}', |
| 90 | + 'Accept': 'application/vnd.github.v3+json' |
| 91 | + } |
| 92 | + |
| 93 | + print(f"Fetching release information from: {api_url}") |
| 94 | + |
| 95 | + # Fetch release data |
| 96 | + response = requests.get(api_url, headers=headers) |
| 97 | + response.raise_for_status() |
| 98 | + release_data = response.json() |
| 99 | + |
| 100 | + # Extract assets that end with .7z only |
| 101 | + assets = [] |
| 102 | + for asset in release_data.get('assets', []): |
| 103 | + filename = asset['name'] |
| 104 | + |
| 105 | + # Check if file ends with .7z |
| 106 | + if filename.endswith('.7z'): |
| 107 | + download_url = asset['browser_download_url'] |
| 108 | + |
| 109 | + # Extract version number (flexible pattern) |
| 110 | + version_match = re.search(r'(\d+\.\d+\.\d+(?:\.\d+)?)', filename) |
| 111 | + if version_match: |
| 112 | + ver = version_match.group(1) |
| 113 | + assets.append({ |
| 114 | + 'version': ver, |
| 115 | + 'url': download_url, |
| 116 | + 'filename': filename |
| 117 | + }) |
| 118 | + print(f"Found: {filename} -> Version: {ver}") |
| 119 | + else: |
| 120 | + print(f"Warning: Could not extract version from: {filename}") |
| 121 | + |
| 122 | + if not assets: |
| 123 | + print(f"No .7z assets found in release") |
| 124 | + exit(0) |
| 125 | + |
| 126 | + # Read existing properties file |
| 127 | + properties_file = "releases.properties" |
| 128 | + |
| 129 | + if not os.path.exists(properties_file): |
| 130 | + print(f"Properties file not found: {properties_file}") |
| 131 | + print(f"Creating new properties file...") |
| 132 | + with open(properties_file, 'w', encoding='utf-8') as f: |
| 133 | + f.write(f"# {module_name.upper()} Releases Properties\n") |
| 134 | + f.write(f"# Auto-generated and maintained by automation\n\n") |
| 135 | + |
| 136 | + with open(properties_file, 'r', encoding='utf-8') as f: |
| 137 | + lines = f.readlines() |
| 138 | + |
| 139 | + # Parse existing properties |
| 140 | + properties = OrderedDict() |
| 141 | + header_lines = [] |
| 142 | + in_header = True |
| 143 | + |
| 144 | + for line in lines: |
| 145 | + stripped = line.strip() |
| 146 | + if in_header and (stripped.startswith('#') or stripped == ''): |
| 147 | + header_lines.append(line) |
| 148 | + else: |
| 149 | + in_header = False |
| 150 | + if '=' in line and not stripped.startswith('#'): |
| 151 | + key, value = line.split('=', 1) |
| 152 | + properties[key.strip()] = value.strip() |
| 153 | + |
| 154 | + # Add new versions |
| 155 | + for asset in assets: |
| 156 | + ver = asset['version'] |
| 157 | + url = asset['url'] |
| 158 | + |
| 159 | + # Create property key (just version number) |
| 160 | + key = ver |
| 161 | + properties[key] = url |
| 162 | + print(f"Added/Updated: {key} = {url}") |
| 163 | + |
| 164 | + # Sort properties by version (semver) |
| 165 | + def extract_version(key): |
| 166 | + # Parse version directly from key |
| 167 | + try: |
| 168 | + return version.parse(key) |
| 169 | + except: |
| 170 | + return version.parse("0.0.0") |
| 171 | + |
| 172 | + sorted_properties = OrderedDict( |
| 173 | + sorted(properties.items(), key=lambda x: extract_version(x[0]), reverse=True) |
| 174 | + ) |
| 175 | + |
| 176 | + # Write back to file |
| 177 | + with open(properties_file, 'w', encoding='utf-8') as f: |
| 178 | + # Write header |
| 179 | + for line in header_lines: |
| 180 | + f.write(line) |
| 181 | + |
| 182 | + # Write sorted properties with spaces around equals sign |
| 183 | + for key, value in sorted_properties.items(): |
| 184 | + f.write(f"{key} = {value}\n") |
| 185 | + |
| 186 | + print(f"\nSuccessfully updated {properties_file}") |
| 187 | + print(f"Total versions: {len(sorted_properties)}") |
| 188 | + |
| 189 | + EOF |
| 190 | + |
| 191 | + - name: Create Pull Request |
| 192 | + id: create_pr |
| 193 | + uses: peter-evans/create-pull-request@v6 |
| 194 | + with: |
| 195 | + token: ${{ secrets.GH_PAT }} |
| 196 | + base: main |
| 197 | + commit-message: | |
| 198 | + Update releases.properties |
| 199 | + |
| 200 | + Auto-generated from release ${{ github.event.release.tag_name || github.event.inputs.release_tag }} |
| 201 | + branch: update-releases-${{ github.event.release.tag_name || github.event.inputs.release_tag }} |
| 202 | + delete-branch: true |
| 203 | + title: 'Update releases.properties from release ${{ github.event.release.tag_name || github.event.inputs.release_tag }}' |
| 204 | + body: | |
| 205 | + ## 🤖 Automated Releases Properties Update |
| 206 | + |
| 207 | + This PR updates the `releases.properties` file with new versions from release `${{ github.event.release.tag_name || github.event.inputs.release_tag }}`. |
| 208 | + |
| 209 | + ### Changes: |
| 210 | + - Extracted .7z assets from the release |
| 211 | + - Added version entries with download URLs |
| 212 | + - Maintained semver ordering (newest first) |
| 213 | + |
| 214 | + **Release URL:** ${{ github.event.release.html_url || format('https://github.com/{0}/{1}/releases/tag/{2}', github.repository_owner, github.event.repository.name, github.event.inputs.release_tag) }} |
| 215 | + |
| 216 | + ### Next Steps: |
| 217 | + 1. ⏳ Link validation will run automatically |
| 218 | + 2. ✅ Once validation passes, this PR will auto-merge |
| 219 | + 3. ❌ If validation fails, please review and fix invalid URLs |
| 220 | + labels: | |
| 221 | + automated |
| 222 | + releases-update |
| 223 | + |
| 224 | + - name: Wait for validation checks |
| 225 | + if: steps.create_pr.outputs.pull-request-number != '' |
| 226 | + run: | |
| 227 | + echo "Waiting for link validation to complete..." |
| 228 | + sleep 10 |
| 229 | + |
| 230 | + - name: Enable auto-merge |
| 231 | + if: steps.create_pr.outputs.pull-request-number != '' |
| 232 | + run: | |
| 233 | + gh pr merge ${{ steps.create_pr.outputs.pull-request-number }} --auto --squash |
| 234 | + env: |
| 235 | + GH_TOKEN: ${{ secrets.GH_PAT }} |
0 commit comments