Compare commits
80 Commits
Author | SHA1 | Date | |
---|---|---|---|
466c1c4918 | |||
3d50c5cc8d
|
|||
a753d6aa3b
|
|||
7bba14fe49 | |||
c8628e7bd7
|
|||
df27af76ad
|
|||
6b4016177f | |||
33c5c71923
|
|||
8df31fb97e
|
|||
5abdff1d1c | |||
ec44bde050
|
|||
30f83b08ee
|
|||
5771982931 | |||
2df09c467e
|
|||
09e41744fb
|
|||
e12595d0a8 | |||
4254a7f534
|
|||
543b311d1e
|
|||
313ace34fe | |||
df2ba3b060
|
|||
27ee1746db
|
|||
03add717be | |||
01986944f0
|
|||
c25b2e9cd5
|
|||
f44f457e6b | |||
169d9ec4db
|
|||
cbcd5e2ab7
|
|||
ef3ac462ba | |||
d27a0cbd80
|
|||
1d9659b6df
|
|||
40ca5c987d | |||
d8700af97d
|
|||
856565d87f
|
|||
6b937d6408 | |||
d11131d9b2
|
|||
e5f5084931
|
|||
5eafa06477 | |||
039a14a473
|
|||
36e60c24ac | |||
525d8bfc44
|
|||
a4ff409fd8
|
|||
b6151b793d
|
|||
374f30a502
|
|||
70ae9ebc85 | |||
94d69f193a
|
|||
b75e412192
|
|||
06d36e5aba | |||
734002a8f1
|
|||
8a3960ad87
|
|||
518d1a2055 | |||
eadc2ebc63
|
|||
522689977f
|
|||
5c8f734acc | |||
380b9ad81a
|
|||
79d7c5a7ee
|
|||
75df670db7 | |||
4122b99eba
|
|||
5d42ea9ddb
|
|||
73badf986b | |||
2ab0da2e14
|
|||
35eff4400f | |||
041d7e9a8a
|
|||
bb27eccc86 | |||
212e8c6a49
|
|||
bbdd116645 | |||
bad396a5ff
|
|||
69a28efe62 | |||
3839fd8de0
|
|||
29243bd673
|
|||
ad1f85e1f7 | |||
03674e57c3
|
|||
c20d53cf61 | |||
79921dd636
|
|||
b20b8688c1
|
|||
5744c8bdc0 | |||
bb97a9b57a
|
|||
9a0586653a
|
|||
f585fd6bf6 | |||
043a01c2ba
|
|||
9a0d65b8eb
|
5
.gitea/default_merge_message/MERGE_TEMPLATE.md
Normal file
5
.gitea/default_merge_message/MERGE_TEMPLATE.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
chore(pr): ${PullRequestTitle} ${PullRequestReference}
|
||||||
|
|
||||||
|
${PullRequestDescription}
|
||||||
|
|
||||||
|
Merged from ${HeadBranch} into ${BaseBranch}
|
61
.gitea/workflows/create-major-tag.yml
Normal file
61
.gitea/workflows/create-major-tag.yml
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
name: Create Major Version Tag
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update-major-tag:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: write # nötig zum Löschen/Erstellen von Tags
|
||||||
|
|
||||||
|
steps:
|
||||||
|
# ▸ 1. Repository beim Release-Tag auschecken
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0 # alle Tags holen
|
||||||
|
ref: ${{ github.event.release.tag_name }}
|
||||||
|
|
||||||
|
# ▸ 2. Hauptversion aus dem Release-Tag extrahieren
|
||||||
|
- name: Derive major version
|
||||||
|
id: derive
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
TAG="${{ github.event.release.tag_name }}" # z. B. v2.3.4
|
||||||
|
echo "Release-Tag: $TAG"
|
||||||
|
|
||||||
|
TAG_NOPREFIX="${TAG#v}" # entfernt führendes v/V
|
||||||
|
MAJOR="${TAG_NOPREFIX%%.*}" # nimmt Teil vor erstem Punkt
|
||||||
|
MAJOR_TAG="v${MAJOR}" # z. B. v2
|
||||||
|
|
||||||
|
echo "Major-Tag: $MAJOR_TAG"
|
||||||
|
echo "major_tag=$MAJOR_TAG" >>"$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
# ▸ 3. Altes Major-Tag (falls vorhanden) löschen und neu anlegen
|
||||||
|
- name: Re-create major tag
|
||||||
|
env:
|
||||||
|
MAJOR_TAG: ${{ steps.derive.outputs.major_tag }}
|
||||||
|
RELEASE_TAG: ${{ github.event.release.tag_name }}
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
git config user.name "$CI_COMMIT_AUTHOR_NAME"
|
||||||
|
git config user.email "$CI_COMMIT_AUTHOR_EMAIL"
|
||||||
|
|
||||||
|
# Bestimmten Commit des Release-Tags ermitteln
|
||||||
|
COMMIT_SHA="$(git rev-list -n 1 "$RELEASE_TAG")"
|
||||||
|
|
||||||
|
echo "Commit des Release-Tags: $COMMIT_SHA"
|
||||||
|
|
||||||
|
# (a) Lokal + remote altes Major-Tag entfernen (falls es existiert)
|
||||||
|
git tag -d "$MAJOR_TAG" 2>/dev/null || true
|
||||||
|
git push --delete origin "$MAJOR_TAG" 2>/dev/null || true
|
||||||
|
|
||||||
|
# (b) Neues annotiertes Tag erstellen und pushen
|
||||||
|
git tag -a "$MAJOR_TAG" "$COMMIT_SHA" \
|
||||||
|
-m "Update $MAJOR_TAG → $RELEASE_TAG"
|
||||||
|
git push origin "$MAJOR_TAG"
|
||||||
|
|
||||||
|
echo "✅ $MAJOR_TAG zeigt nun auf $RELEASE_TAG ($COMMIT_SHA)"
|
@@ -14,6 +14,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Release
|
- name: Release
|
||||||
uses: https://git.0xmax42.io/actions/auto-changelog-release-action@main
|
uses: https://git.0xmax42.io/actions/auto-changelog-release-action@v0
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.RELEASE_PUBLISH_TOKEN }}
|
token: ${{ secrets.RELEASE_PUBLISH_TOKEN }}
|
||||||
|
allow_non_main_release: "true"
|
||||||
|
139
CHANGELOG.md
139
CHANGELOG.md
@@ -2,6 +2,145 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file.
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
|
## [2.0.5](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v2.0.4..v2.0.5) - 2025-07-04
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(script)* Correct path formatting in version detection script - ([a753d6a](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/a753d6aa3b45d4dab36e4ef56bbc71046eb032c9))
|
||||||
|
|
||||||
|
## [2.0.4](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v2.0.3..v2.0.4) - 2025-07-04
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(main)* Update script path to ensure correct resolution - ([df27af7](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/df27af76add1dc4b50ead8015976b5e96b177661))
|
||||||
|
|
||||||
|
## [2.0.3](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v2.0.2..v2.0.3) - 2025-07-04
|
||||||
|
|
||||||
|
### 🚜 Refactor
|
||||||
|
|
||||||
|
- *(action)* Simplify composite action structure - ([8df31fb](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/8df31fb97ebd73e71b2f8a9dc925fe8a81192be3))
|
||||||
|
|
||||||
|
## [2.0.2](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v2.0.1..v2.0.2) - 2025-07-04
|
||||||
|
|
||||||
|
### 🚜 Refactor
|
||||||
|
|
||||||
|
- *(actions)* Simplify action path structure - ([30f83b0](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/30f83b08eecabe17b7d1f53aaf0bb4d0cdd82256))
|
||||||
|
|
||||||
|
## [2.0.1](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.4.3..v2.0.1) - 2025-07-04
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(workflows)* Allow non-main branch releases - ([09e4174](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/09e41744fbf920c0253e13644cb38f1dd430c0f3))
|
||||||
|
- *(actions)* Add post-step support for composite actions - ([543b311](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/543b311d1eb1fb83e8e0ef4014c549e61a863038))
|
||||||
|
|
||||||
|
## [0.4.3](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.4.2..v0.4.3) - 2025-06-29
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(script)* Use dynamic branch name for git push - ([27ee174](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/27ee1746dbc7b2c6c1564a04c024337ff158a9c5))
|
||||||
|
|
||||||
|
## [0.4.2](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.4.1..v0.4.2) - 2025-06-29
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(action)* Update conditions for version detection - ([c25b2e9](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/c25b2e9cd551e923b84cd144b818940fd5e1ccff))
|
||||||
|
|
||||||
|
## [0.4.1](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.4.0..v0.4.1) - 2025-06-29
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(action)* Standardize input key naming - ([cbcd5e2](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/cbcd5e2ab7e7f2498021d5043da9999f62dcf44d))
|
||||||
|
|
||||||
|
## [0.4.0](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.6..v0.4.0) - 2025-06-29
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(action)* Add support for non-main branch releases - ([1d9659b](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/1d9659b6df0d63ed17432e677ee77a4c17d2f2f3))
|
||||||
|
|
||||||
|
## [0.3.6](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.5..v0.3.6) - 2025-06-29
|
||||||
|
|
||||||
|
### ◀️ Revert
|
||||||
|
|
||||||
|
- Integrate gha-timer for step timing - ([856565d](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/856565d87f2575d6883705ddde3d177555c1bd86))
|
||||||
|
|
||||||
|
## [0.3.5](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.4..v0.3.5) - 2025-06-29
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(action)* Integrate gha-timer for step timing - ([e5f5084](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/e5f50849316f5dc0b045ac696c3628e610bae695))
|
||||||
|
|
||||||
|
### ◀️ Revert
|
||||||
|
|
||||||
|
- Improve CI readability with grouped git setup logs - ([039a14a](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/039a14a473493fd74fb61b60910652db3cfd5976))
|
||||||
|
|
||||||
|
## [0.3.4](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.2..v0.3.4) - 2025-06-29
|
||||||
|
|
||||||
|
### ⚙️ Miscellaneous Tasks
|
||||||
|
|
||||||
|
- *(workflows)* Update action to specific version - ([a4ff409](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/a4ff409fd812a5ccd2291251de5cd2af9bb0e713))
|
||||||
|
- *(scripts)* Update file permissions to executable - ([374f30a](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/374f30a50206a426b8de7a0b78cde9ad4194469f))
|
||||||
|
|
||||||
|
## [0.3.2](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.1..v0.3.2) - 2025-06-29
|
||||||
|
|
||||||
|
### 🚜 Refactor
|
||||||
|
|
||||||
|
- *(action)* Simplify script invocation syntax - ([b75e412](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/b75e4121928131c1a1c0cbdea954a4fd602edd1a))
|
||||||
|
|
||||||
|
## [0.3.1](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.3.0..v0.3.1) - 2025-06-29
|
||||||
|
|
||||||
|
### ⚙️ Miscellaneous Tasks
|
||||||
|
|
||||||
|
- *(scripts)* Improve CI readability with grouped git setup logs - ([8a3960a](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/8a3960ad8712bc29823dc601a120a61582fa2a3b))
|
||||||
|
|
||||||
|
## [0.3.0](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.2.3..v0.3.0) - 2025-06-20
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(script)* Enhance git-cliff installer with jq support - ([5226899](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/522689977fe8883975e68add1b0cba2e685978ee))
|
||||||
|
|
||||||
|
### ⚙️ Miscellaneous Tasks
|
||||||
|
|
||||||
|
- *(workflows)* Update release action version - ([79d7c5a](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/79d7c5a7eefda51f03a9bde0e933689c29d57567))
|
||||||
|
|
||||||
|
## [0.2.3](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.2.2..v0.2.3) - 2025-06-14
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(script)* Adjust GIT_AUTHOR_DATE format for compatibility - ([5d42ea9](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/5d42ea9ddbfa4f151c83fc16c652d55618f2ee04))
|
||||||
|
|
||||||
|
## [0.2.2](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.2.1..v0.2.2) - 2025-06-14
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(scripts)* Validate git configuration during setup - ([212e8c6](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/212e8c6a499365ac77c98138a09b6736dec8fe7e))
|
||||||
|
|
||||||
|
### 🚜 Refactor
|
||||||
|
|
||||||
|
- *(scripts)* Standardize scripts and improve readability - ([041d7e9](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/041d7e9a8a7baa5b6f41ad6dabd7d4837f7d254a))
|
||||||
|
|
||||||
|
## [0.2.1](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.1.3..v0.2.1) - 2025-06-14
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(release)* Add retry logic for release creation - ([29243bd](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/29243bd67344a3020fc3023d238dacbdd96c53ad))
|
||||||
|
|
||||||
|
## [0.1.3](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.1.2..v0.1.3) - 2025-06-14
|
||||||
|
|
||||||
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
|
- *(workflows)* Configure Git user in major tag creation - ([79921dd](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/79921dd63676138f2cfbcc3894f294b292f0a653))
|
||||||
|
|
||||||
|
## [0.1.2](https://git.0xmax42.io/actions/auto-changelog-release-action/compare/v0.1.0..v0.1.2) - 2025-06-14
|
||||||
|
|
||||||
|
### 🚀 Features
|
||||||
|
|
||||||
|
- *(workflows)* Add workflow to create major version tags - ([9a05866](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/9a0586653a5e7aa6a3acb12d119f4a43c09e96c1))
|
||||||
|
|
||||||
|
### ⚙️ Miscellaneous Tasks
|
||||||
|
|
||||||
|
- *(workflows)* Update action to main branch - ([043a01c](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/043a01c2bacb78f3c872056f7d18aab4e845a401))
|
||||||
|
- *(workflows)* Update action version in release workflow - ([9a0d65b](https://git.0xmax42.io/actions/auto-changelog-release-action/commit/9a0d65b8eba3d1e06c88bc02f19e2dd5a7d0a731))
|
||||||
|
|
||||||
## [0.1.0] - 2025-06-14
|
## [0.1.0] - 2025-06-14
|
||||||
|
|
||||||
### 🚀 Features
|
### 🚀 Features
|
||||||
|
53
action.yml
53
action.yml
@@ -1,5 +1,5 @@
|
|||||||
name: Auto Changelog & Release
|
name: Auto Changelog & Release
|
||||||
description: "One-stop composite action for version-bump detection, changelog generation and Gitea release."
|
description: One-stop composite action
|
||||||
|
|
||||||
inputs:
|
inputs:
|
||||||
token:
|
token:
|
||||||
@@ -14,45 +14,16 @@ inputs:
|
|||||||
description: "Commit-Autore-Mail"
|
description: "Commit-Autore-Mail"
|
||||||
required: false
|
required: false
|
||||||
default: ""
|
default: ""
|
||||||
|
allow_non_main_release:
|
||||||
|
description: "Allow publishing releases from branches other than 'main'."
|
||||||
|
required: false
|
||||||
|
default: "false"
|
||||||
|
|
||||||
|
outputs:
|
||||||
|
release:
|
||||||
|
description: "Whether a release should be made (true/false)"
|
||||||
|
|
||||||
runs:
|
runs:
|
||||||
using: "composite"
|
using: "node20"
|
||||||
steps:
|
main: "dist/main.js"
|
||||||
- name: Detect version change
|
post: "dist/post.js"
|
||||||
shell: bash
|
|
||||||
id: detect
|
|
||||||
env:
|
|
||||||
GITHUB_EVENT_BEFORE: ${{ github.event.before || '' }}
|
|
||||||
GITHUB_SHA: ${{ github.sha || '' }}
|
|
||||||
GITHUB_REF: ${{ github.ref || '' }}
|
|
||||||
run: |
|
|
||||||
bash ${{ github.action_path }}/scripts/detect-version-change.sh
|
|
||||||
|
|
||||||
- name: Read CLIFF_VERSION
|
|
||||||
shell: bash
|
|
||||||
id: cliff_version
|
|
||||||
run: bash ${{ github.action_path }}/scripts/read-cliff-version.sh
|
|
||||||
|
|
||||||
- name: Install git-cliff
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
bash ${{ github.action_path }}/scripts/install-git-cliff.sh "${{ steps.cliff_version.outputs.version }}"
|
|
||||||
|
|
||||||
- name: Set up git
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
bash ${{ github.action_path }}/scripts/setup-git.sh \
|
|
||||||
"${{ inputs.author_name }}" \
|
|
||||||
"${{ inputs.author_email }}"
|
|
||||||
|
|
||||||
- name: Generate and commit changelog (unreleased)
|
|
||||||
if: steps.detect.outputs.version_changed == 'false' || github.ref != 'refs/heads/main'
|
|
||||||
shell: bash
|
|
||||||
run: bash ${{ github.action_path }}/scripts/generate-unreleased-changelog.sh
|
|
||||||
|
|
||||||
- name: Release from VERSION
|
|
||||||
if: steps.detect.outputs.version_changed == 'true' && github.ref == 'refs/heads/main'
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
RELEASE_PUBLISH_TOKEN: ${{ inputs.token || '' }}
|
|
||||||
run: bash ${{ github.action_path }}/scripts/release-from-version.sh
|
|
||||||
|
34
dist/main.js
vendored
Normal file
34
dist/main.js
vendored
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
const cp = require("child_process");
|
||||||
|
const path = require("path");
|
||||||
|
const fs = require("fs");
|
||||||
|
|
||||||
|
function run(script) {
|
||||||
|
cp.execFileSync(script, { stdio: "inherit", shell: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
function exportOutput(key, value) {
|
||||||
|
fs.appendFileSync(process.env.GITHUB_OUTPUT, `${key}=${value}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setEnv(key, value) {
|
||||||
|
fs.appendFileSync(process.env.GITHUB_ENV, `${key}=${value}\n`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function main() {
|
||||||
|
const actionPath = process.env.GITHUB_ACTION_PATH || ".";
|
||||||
|
const script = path.join("./", "scripts/detect-version-change.sh");
|
||||||
|
|
||||||
|
run(script);
|
||||||
|
|
||||||
|
// Hier nehmen wir an, dass das Script selbst bereits `version_changed=true/false`
|
||||||
|
// in $GITHUB_OUTPUT geschrieben hat.
|
||||||
|
// Zur Sicherheit lesen wir das nochmal aus und speichern es auch als release-Flag.
|
||||||
|
const output = fs.readFileSync(process.env.GITHUB_OUTPUT, "utf-8");
|
||||||
|
const match = output.match(/^version_changed=(.*)$/m);
|
||||||
|
const value = match?.[1]?.trim() || "false";
|
||||||
|
|
||||||
|
exportOutput("release", value);
|
||||||
|
setEnv("VERSION_CHANGED", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
31
dist/post.js
vendored
Normal file
31
dist/post.js
vendored
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
const cp = require("child_process");
|
||||||
|
const path = require("path");
|
||||||
|
|
||||||
|
function run(script, args = []) {
|
||||||
|
cp.execFileSync(script, {
|
||||||
|
stdio: "inherit",
|
||||||
|
shell: true,
|
||||||
|
env: process.env,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function main() {
|
||||||
|
const actionPath = process.env.GITHUB_ACTION_PATH || ".";
|
||||||
|
const versionChanged = (process.env.VERSION_CHANGED || "false").toLowerCase() === "true";
|
||||||
|
|
||||||
|
const readCliff = path.join(actionPath, "scripts/read-cliff-version.sh");
|
||||||
|
const cliffVersion = cp.execSync(readCliff, { encoding: "utf-8", shell: true }).trim().split("\n").pop();
|
||||||
|
|
||||||
|
run(path.join(actionPath, "scripts/install-git-cliff.sh"), [cliffVersion]);
|
||||||
|
|
||||||
|
run(`${path.join(actionPath, "scripts/setup-git.sh")} "${process.env.INPUT_AUTHOR_NAME}" "${process.env.INPUT_AUTHOR_EMAIL}"`);
|
||||||
|
|
||||||
|
if (versionChanged) {
|
||||||
|
process.env.RELEASE_PUBLISH_TOKEN = process.env.INPUT_TOKEN || "";
|
||||||
|
run(path.join(actionPath, "scripts/release-from-version.sh"));
|
||||||
|
} else {
|
||||||
|
run(path.join(actionPath, "scripts/generate-unreleased-changelog.sh"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
@@ -1,19 +1,22 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
# === Inputs from GitHub/Gitea Action environment ===
|
# Inputs from GitHub/Gitea Action environment
|
||||||
GIT_REF="${GITHUB_REF:-}"
|
GIT_REF="${GITHUB_REF:-}"
|
||||||
COMMIT_BEFORE="${GITHUB_EVENT_BEFORE:-}"
|
COMMIT_BEFORE="${GITHUB_EVENT_BEFORE:-}"
|
||||||
COMMIT_AFTER="${GITHUB_SHA:-}"
|
COMMIT_AFTER="${GITHUB_SHA:-}"
|
||||||
VERSION_FILE="VERSION"
|
VERSION_FILE="VERSION"
|
||||||
|
ALLOW_NON_MAIN_RELEASE="${ALLOW_NON_MAIN_RELEASE:-false}"
|
||||||
|
|
||||||
echo "🔍 Comparing commits:"
|
echo "🔍 Comparing commits:"
|
||||||
echo "Before: $COMMIT_BEFORE"
|
echo "Before: $COMMIT_BEFORE"
|
||||||
echo "After: $COMMIT_AFTER"
|
echo "After: $COMMIT_AFTER"
|
||||||
|
echo "Ref: $GIT_REF"
|
||||||
|
echo "Allow release from non-main branches: $ALLOW_NON_MAIN_RELEASE"
|
||||||
|
|
||||||
# === Check branch condition ===
|
# Check branch condition
|
||||||
if [[ "$GIT_REF" != "refs/heads/main" ]]; then
|
if [[ "$ALLOW_NON_MAIN_RELEASE" != "true" && "$GIT_REF" != "refs/heads/main" ]]; then
|
||||||
echo "Not on 'main' branch – skipping version check."
|
echo "🚫 Not on 'main' branch and non-main releases are disabled – skipping version check."
|
||||||
echo "version_changed=false" >> "$GITHUB_OUTPUT"
|
echo "version_changed=false" >> "$GITHUB_OUTPUT"
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
0
scripts/generate-unreleased-changelog.sh
Normal file → Executable file
0
scripts/generate-unreleased-changelog.sh
Normal file → Executable file
@@ -1,7 +1,7 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
# install-git-cliff.sh – holt neueste oder gewünschte git-cliff-Binary (x86_64)
|
# install-git-cliff.sh – fetches the latest or specified git-cliff binary (x86_64)
|
||||||
# Usage: sudo ./install-git-cliff.sh # neueste Version
|
# Usage: sudo ./install-git-cliff.sh # latest version
|
||||||
# sudo ./install-git-cliff.sh 2.9.0 # bestimmte Version
|
# sudo ./install-git-cliff.sh 2.9.0 # specific version
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
REPO="orhun/git-cliff"
|
REPO="orhun/git-cliff"
|
||||||
@@ -9,42 +9,50 @@ ARCH_OS="x86_64-unknown-linux-gnu"
|
|||||||
INSTALL_DIR="/usr/local/bin"
|
INSTALL_DIR="/usr/local/bin"
|
||||||
VERSION="${1:-latest}"
|
VERSION="${1:-latest}"
|
||||||
|
|
||||||
need() { command -v "$1" >/dev/null || { echo "$1 fehlt"; exit 1; }; }
|
need() { command -v "$1" >/dev/null || { echo "$1 is missing"; exit 1; }; }
|
||||||
need curl; need tar; need grep; need sed; need awk
|
need curl; need tar; need grep; need sed; need awk; need jq
|
||||||
|
|
||||||
# 1 Version ermitteln → Release-JSON abrufen
|
# 1 Determine version → Fetch release JSON
|
||||||
if [[ "$VERSION" == "latest" ]]; then
|
if [[ "$VERSION" == "latest" ]]; then
|
||||||
API_URL="https://api.github.com/repos/${REPO}/releases/latest"
|
API_URL="https://api.github.com/repos/${REPO}/releases/latest"
|
||||||
else
|
else
|
||||||
API_URL="https://api.github.com/repos/${REPO}/releases/tags/v${VERSION}"
|
API_URL="https://api.github.com/repos/${REPO}/releases/tags/v${VERSION}"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo "🔍 Hole Release-Info ($API_URL)…"
|
echo "🔍 Fetching release info ($API_URL)…"
|
||||||
JSON=$(curl -sL "$API_URL")
|
JSON=$(curl -fsSL "$API_URL") || {
|
||||||
|
echo "❌ Failed to fetch release info"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
VERSION=$(echo "$JSON" | grep -m1 '"tag_name":' | sed -E 's/.*"v?([^"]+)".*/\1/')
|
VERSION=$(jq -r '.tag_name' <<< "$JSON") || {
|
||||||
ASSET_URL=$(echo "$JSON" |
|
echo "❌ Could not extract version"
|
||||||
grep -Eo '"browser_download_url": *"[^"]+' |
|
exit 1
|
||||||
cut -d'"' -f4 |
|
}
|
||||||
grep "${ARCH_OS}\\.tar" | head -n1)
|
|
||||||
|
|
||||||
[[ -z "$ASSET_URL" ]] && { echo "❌ passender Asset nicht gefunden"; exit 1; }
|
ASSET_URL=$(jq -r '.assets[]?.browser_download_url' <<< "$JSON" |
|
||||||
|
grep "${ARCH_OS}\.tar" | head -n1)
|
||||||
|
|
||||||
|
if [[ -z "$ASSET_URL" ]]; then
|
||||||
|
echo "❌ Matching asset not found for architecture ${ARCH_OS}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
ASSET_FILE=$(basename "$ASSET_URL")
|
ASSET_FILE=$(basename "$ASSET_URL")
|
||||||
echo "📦 Lade git-cliff v${VERSION} (${ASSET_FILE}) …"
|
echo "📦 Downloading git-cliff v${VERSION} (${ASSET_FILE}) …"
|
||||||
TMP=$(mktemp -d)
|
TMP=$(mktemp -d)
|
||||||
curl -#L -o "${TMP}/${ASSET_FILE}" "$ASSET_URL"
|
curl -#L -o "${TMP}/${ASSET_FILE}" "$ASSET_URL"
|
||||||
|
|
||||||
# 2 Entpacken je nach Endung
|
# 2 Extract based on file extension
|
||||||
case "$ASSET_FILE" in
|
case "$ASSET_FILE" in
|
||||||
*.tar.gz|*.tgz) tar -C "$TMP" -xzf "${TMP}/${ASSET_FILE}" ;;
|
*.tar.gz|*.tgz) tar -C "$TMP" -xzf "${TMP}/${ASSET_FILE}" ;;
|
||||||
*.tar.xz) tar -C "$TMP" -xJf "${TMP}/${ASSET_FILE}" ;;
|
*.tar.xz) tar -C "$TMP" -xJf "${TMP}/${ASSET_FILE}" ;;
|
||||||
*.zip) need unzip; unzip -q "${TMP}/${ASSET_FILE}" -d "$TMP" ;;
|
*.zip) need unzip; unzip -q "${TMP}/${ASSET_FILE}" -d "$TMP" ;;
|
||||||
*) echo "❌ Unbekanntes Archivformat: $ASSET_FILE"; exit 1 ;;
|
*) echo "❌ Unknown archive format: $ASSET_FILE"; exit 1 ;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
BIN_PATH=$(find "$TMP" -type f -name git-cliff -perm -u+x | head -n1)
|
BIN_PATH=$(find "$TMP" -type f -name git-cliff -perm -u+x | head -n1)
|
||||||
[[ -z "$BIN_PATH" ]] && { echo "❌ Binary nicht gefunden"; exit 1; }
|
[[ -z "$BIN_PATH" ]] && { echo "❌ Binary not found"; exit 1; }
|
||||||
|
|
||||||
sudo install -m755 "$BIN_PATH" "${INSTALL_DIR}/git-cliff"
|
sudo install -m755 "$BIN_PATH" "${INSTALL_DIR}/git-cliff"
|
||||||
echo "✅ git-cliff $(git-cliff --version) installiert unter ${INSTALL_DIR}"
|
echo "✅ git-cliff $(git-cliff --version) installed in ${INSTALL_DIR}"
|
||||||
|
7
scripts/read-cliff-version.sh
Normal file → Executable file
7
scripts/read-cliff-version.sh
Normal file → Executable file
@@ -12,10 +12,11 @@ fi
|
|||||||
VERSION_LINE=$(awk -F '=' '/^# CLIFF_VERSION=/ { gsub(/[" ]/, "", $2); print $2 }' "$CLIFF_TOML" || true)
|
VERSION_LINE=$(awk -F '=' '/^# CLIFF_VERSION=/ { gsub(/[" ]/, "", $2); print $2 }' "$CLIFF_TOML" || true)
|
||||||
|
|
||||||
if [[ -n "$VERSION_LINE" ]]; then
|
if [[ -n "$VERSION_LINE" ]]; then
|
||||||
echo "✅ Extracted CLIFF_VERSION: $VERSION_LINE"
|
echo "✅ Extracted CLIFF_VERSION: $VERSION_LINE" >&2
|
||||||
else
|
else
|
||||||
echo "⚠️ No CLIFF_VERSION found in $CLIFF_TOML"
|
echo "⚠️ No CLIFF_VERSION found in $CLIFF_TOML" >&2
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Output für GitHub Actions / Composite Action
|
|
||||||
echo "version=${VERSION_LINE:-}" >> "$GITHUB_OUTPUT"
|
echo "version=${VERSION_LINE:-}" >> "$GITHUB_OUTPUT"
|
||||||
|
echo "$VERSION_LINE"
|
||||||
|
n
|
104
scripts/release-from-version.sh
Normal file → Executable file
104
scripts/release-from-version.sh
Normal file → Executable file
@@ -1,13 +1,60 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
|
create_release() {
|
||||||
|
local -r owner="$1" repo="$2" token="$3" version="$4" body_file="$5"
|
||||||
|
local -i max_attempts=3 delay=5 attempt
|
||||||
|
local http status
|
||||||
|
|
||||||
|
# Read release body for JSON payload
|
||||||
|
local body_json
|
||||||
|
body_json=$(tail -n +2 "$body_file" | jq -Rs .)
|
||||||
|
|
||||||
|
for (( attempt=1; attempt<=max_attempts; attempt++ )); do
|
||||||
|
echo "🚀 Try $attempt/$max_attempts: creating release v$version …"
|
||||||
|
|
||||||
|
http=$(curl -sS \
|
||||||
|
-H "Authorization: token $token" \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
-o resp.json -w '%{http_code}' \
|
||||||
|
-X POST "$GITHUB_API_URL/repos/$owner/$repo/releases" \
|
||||||
|
-d @- <<EOF
|
||||||
|
{
|
||||||
|
"tag_name": "v$version",
|
||||||
|
"target_commitish": "main",
|
||||||
|
"name": "Release v$version",
|
||||||
|
"body": $body_json,
|
||||||
|
"draft": false,
|
||||||
|
"prerelease": false
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
status=$http
|
||||||
|
if [[ $status == 201 ]]; then
|
||||||
|
echo "✅ Release created (201)."
|
||||||
|
rm -f resp.json
|
||||||
|
return 0
|
||||||
|
elif [[ $status == 409 ]]; then
|
||||||
|
echo "🔁 Release already exists (409) – skipping."
|
||||||
|
rm -f resp.json
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "⚠️ Release attempt failed (HTTP $status)"
|
||||||
|
cat resp.json
|
||||||
|
[[ $attempt -lt $max_attempts ]] || { echo "❌ Giving up."; return 1; }
|
||||||
|
sleep $((delay*attempt)) # Back-off: 5 s, 10 s, 15 s …
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
VERSION_FILE="VERSION"
|
VERSION_FILE="VERSION"
|
||||||
CHANGELOG_FILE="CHANGELOG.md"
|
CHANGELOG_FILE="CHANGELOG.md"
|
||||||
CLIFF_CONFIG="cliff.toml"
|
CLIFF_CONFIG="cliff.toml"
|
||||||
RELEASE_BODY_TMP="$(mktemp)"
|
RELEASE_BODY_TMP="$(mktemp)"
|
||||||
BODY_OUTPUT_VAR="changelog_body_path"
|
GIT_BRANCH="${GITHUB_REF##refs/heads/}"
|
||||||
|
|
||||||
# === Schritt 1: Read version from VERSION ===
|
# === Step 1: Read version ===
|
||||||
if [[ ! -f "$VERSION_FILE" ]]; then
|
if [[ ! -f "$VERSION_FILE" ]]; then
|
||||||
echo "❌ VERSION file not found"
|
echo "❌ VERSION file not found"
|
||||||
exit 1
|
exit 1
|
||||||
@@ -15,7 +62,7 @@ fi
|
|||||||
VERSION="$(<"$VERSION_FILE")"
|
VERSION="$(<"$VERSION_FILE")"
|
||||||
echo "📦 Version: $VERSION"
|
echo "📦 Version: $VERSION"
|
||||||
|
|
||||||
# === Schritt 2: Generate changelog for release ===
|
# === Step 2: Generate changelog for release ===
|
||||||
echo "📄 Generating changelog for tag v$VERSION"
|
echo "📄 Generating changelog for tag v$VERSION"
|
||||||
git-cliff -c "$CLIFF_CONFIG" -t "v$VERSION" -o "$CHANGELOG_FILE"
|
git-cliff -c "$CLIFF_CONFIG" -t "v$VERSION" -o "$CHANGELOG_FILE"
|
||||||
|
|
||||||
@@ -24,42 +71,40 @@ ESCAPED_VERSION="$(echo "$VERSION" | sed 's/\./\\./g')"
|
|||||||
awk -v ver="$ESCAPED_VERSION" '
|
awk -v ver="$ESCAPED_VERSION" '
|
||||||
$0 ~ "^## \\[" ver "\\]" {
|
$0 ~ "^## \\[" ver "\\]" {
|
||||||
print_flag=1
|
print_flag=1
|
||||||
line = $0
|
line=$0
|
||||||
sub(/^## /, "", line)
|
sub(/^## /,"",line)
|
||||||
sub(/\\s*\\(.*\\)/, "", line) # entfernt z. B. "(...)" oder "(*)"
|
sub(/\\s*\\(.*\\)/,"",line)
|
||||||
print line
|
print line
|
||||||
next
|
next
|
||||||
}
|
}
|
||||||
$0 ~ "^## \\[" && $0 !~ "^## \\[" ver "\\]" {
|
$0 ~ "^## \\[" && $0 !~ "^## \\[" ver "\\]" { print_flag=0 }
|
||||||
print_flag=0
|
|
||||||
}
|
|
||||||
print_flag
|
print_flag
|
||||||
' "$CHANGELOG_FILE" > "$RELEASE_BODY_TMP"
|
' "$CHANGELOG_FILE" > "$RELEASE_BODY_TMP"
|
||||||
|
|
||||||
# === Schritt 3: Commit updated changelog ===
|
# === Step 3: Commit changelog ===
|
||||||
git add "$CHANGELOG_FILE"
|
git add "$CHANGELOG_FILE"
|
||||||
if git diff --cached --quiet; then
|
if git diff --cached --quiet; then
|
||||||
echo "✅ No changes to commit"
|
echo "✅ No changes to commit"
|
||||||
else
|
else
|
||||||
echo "📝 Committing updated changelog"
|
echo "📝 Committing updated changelog"
|
||||||
git commit -m "chore(changelog): update changelog for v$VERSION"
|
git commit -m "chore(changelog): update changelog for v$VERSION"
|
||||||
git push origin main
|
git push origin "$GIT_BRANCH"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# === Schritt 4: Create tag if not exists ===
|
# === Step 4: Create tag if necessary ===
|
||||||
if git rev-parse "v$VERSION" >/dev/null 2>&1; then
|
if git rev-parse "v$VERSION" >/dev/null 2>&1; then
|
||||||
echo "🔁 Tag v$VERSION already exists, skipping."
|
echo "🔁 Tag v$VERSION already exists, skipping."
|
||||||
else
|
else
|
||||||
echo "🏷️ Creating annotated tag v$VERSION"
|
echo "🏷️ Creating annotated tag v$VERSION"
|
||||||
export GIT_AUTHOR_DATE="$(date --iso-8601=seconds)"
|
export GIT_AUTHOR_DATE="@$(date +%s)"
|
||||||
export GIT_COMMITTER_DATE="$GIT_AUTHOR_DATE"
|
export GIT_COMMITTER_DATE="$GIT_AUTHOR_DATE"
|
||||||
git tag -a "v$VERSION" -F "$RELEASE_BODY_TMP" --cleanup=verbatim
|
git tag -a "v$VERSION" -F "$RELEASE_BODY_TMP" --cleanup=verbatim
|
||||||
git push origin "v$VERSION"
|
git push origin "v$VERSION"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# === Schritt 5: Create Gitea release ===
|
# === Step 5: Create release (with retry) ===
|
||||||
OWNER="$(echo "$GITHUB_REPOSITORY" | cut -d/ -f1)"
|
OWNER="${GITHUB_REPOSITORY%/*}"
|
||||||
REPO="$(echo "$GITHUB_REPOSITORY" | cut -d/ -f2)"
|
REPO="${GITHUB_REPOSITORY#*/}"
|
||||||
TOKEN="${RELEASE_PUBLISH_TOKEN:-$ACTIONS_RUNTIME_TOKEN}"
|
TOKEN="${RELEASE_PUBLISH_TOKEN:-$ACTIONS_RUNTIME_TOKEN}"
|
||||||
|
|
||||||
if [[ -z "${RELEASE_PUBLISH_TOKEN:-}" ]]; then
|
if [[ -z "${RELEASE_PUBLISH_TOKEN:-}" ]]; then
|
||||||
@@ -69,29 +114,6 @@ if [[ -z "${RELEASE_PUBLISH_TOKEN:-}" ]]; then
|
|||||||
echo
|
echo
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Prüfen, ob Release bereits existiert
|
create_release "$OWNER" "$REPO" "$TOKEN" "$VERSION" "$RELEASE_BODY_TMP"
|
||||||
if curl -sf "$GITHUB_API_URL/repos/$OWNER/$REPO/releases/tags/v$VERSION" \
|
|
||||||
-H "Authorization: token $TOKEN" > /dev/null; then
|
|
||||||
echo "🔁 Release for tag v$VERSION already exists, skipping."
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "🚀 Creating Gitea release for v$VERSION"
|
echo "🎉 Workflow finished successfully."
|
||||||
|
|
||||||
RELEASE_BODY=$(tail -n +2 "$RELEASE_BODY_TMP" | jq -Rs .)
|
|
||||||
|
|
||||||
curl -X POST "$GITHUB_API_URL/repos/$OWNER/$REPO/releases" \
|
|
||||||
-H "Authorization: token $TOKEN" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-d @- <<EOF
|
|
||||||
{
|
|
||||||
"tag_name": "v$VERSION",
|
|
||||||
"target_commitish": "main",
|
|
||||||
"name": "Release v$VERSION",
|
|
||||||
"body": $RELEASE_BODY,
|
|
||||||
"draft": false,
|
|
||||||
"prerelease": false
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "✅ Release for tag $VERSION created successfully."
|
|
||||||
|
20
scripts/setup-git.sh
Normal file → Executable file
20
scripts/setup-git.sh
Normal file → Executable file
@@ -1,9 +1,9 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
# Optional inputs (positionals) oder Fallback auf Umgebungsvariablen
|
# Optional inputs (positionals) or fallback to environment variables
|
||||||
AUTHOR_NAME="${1:-${CI_COMMIT_AUTHOR_NAME:-CI Bot}}"
|
AUTHOR_NAME="${1:-${CI_COMMIT_AUTHOR_NAME:-CI Bot}}"
|
||||||
AUTHOR_EMAIL="${2:-${CI_COMMIT_AUTHOR_EMAIL:-ci@example.com}}"
|
AUTHOR_EMAIL="${2:-${CI_COMMIT_AUTHOR_EMAIL:-ci@bot.none}}"
|
||||||
|
|
||||||
echo "🔧 Setting up git author:"
|
echo "🔧 Setting up git author:"
|
||||||
echo " Name : $AUTHOR_NAME"
|
echo " Name : $AUTHOR_NAME"
|
||||||
@@ -11,3 +11,19 @@ echo " Email: $AUTHOR_EMAIL"
|
|||||||
|
|
||||||
git config --global user.name "$AUTHOR_NAME"
|
git config --global user.name "$AUTHOR_NAME"
|
||||||
git config --global user.email "$AUTHOR_EMAIL"
|
git config --global user.email "$AUTHOR_EMAIL"
|
||||||
|
|
||||||
|
# Check if the values were set correctly
|
||||||
|
CONFIGURED_NAME=$(git config --global user.name)
|
||||||
|
CONFIGURED_EMAIL=$(git config --global user.email)
|
||||||
|
|
||||||
|
if [[ "$CONFIGURED_NAME" != "$AUTHOR_NAME" ]]; then
|
||||||
|
echo "❌ Error: Git username was not set correctly!" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$CONFIGURED_EMAIL" != "$AUTHOR_EMAIL" ]]; then
|
||||||
|
echo "❌ Error: Git email was not set correctly!" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "✅ Git configuration completed successfully."
|
||||||
|
Reference in New Issue
Block a user