From 81611c0dd1ff3da0982971b496d4f9e01fcef90d Mon Sep 17 00:00:00 2001 From: Simon Diesenreiter Date: Sat, 30 Nov 2024 10:23:58 -0800 Subject: [PATCH] ci: add conventional commit support ref: NOISSUE --- .gitea/conventional_commits/commit-msg | 46 +++++++++ .../conventional_commits/generate-version.sh | 95 +++++++++++++++++++ .../conventional_commits/hooks/text-styles.sh | 44 +++++++++ .gitignore | 3 +- Makefile | 26 +++-- 5 files changed, 204 insertions(+), 10 deletions(-) create mode 100755 .gitea/conventional_commits/commit-msg create mode 100755 .gitea/conventional_commits/generate-version.sh create mode 100755 .gitea/conventional_commits/hooks/text-styles.sh diff --git a/.gitea/conventional_commits/commit-msg b/.gitea/conventional_commits/commit-msg new file mode 100755 index 0000000..c262f41 --- /dev/null +++ b/.gitea/conventional_commits/commit-msg @@ -0,0 +1,46 @@ +#!/usr/bin/env sh +echo "Running commit message checks..." + +. "$(dirname -- "$0")/../../.gitea/conventional_commits/hooks/text-styles.sh" + + +# Get the commit message +commit="$(cat .git/COMMIT_EDITMSG)" +# Define the conventional commit regex +regex="^(build|chore|ci|docs|feat|fix|perf|refactor|revert|style|test)(\(.+\))?(!?):\s(.+)\s((ref(s?):?\s?)[A-Z0-9]+\-[0-9]+)$" + + +# Check if the commit message matches the conventional commit format +if ! echo "$commit" | grep -Pq "$regex" +then + echo + colorPrint red "❌ Failed to create commit. Your commit message does not follow the conventional commit format." + colorPrint red "Please use the following format: $(colorPrint brightRed 'type(scope)?: description')" + colorPrint red "Available types are listed below. Scope is optional. Use ! after type to indicate breaking change." + echo + colorPrint brightWhite "Quick examples:" + echo "feat: add email notifications on new direct messages refs ABC-1213" + echo "feat(shopping cart): add the amazing button ref: DEFG-23" + echo "feat!: remove ticket list endpoint ref DADA-109" + echo "fix(api): handle empty message in request body refs: MINE-82" + echo "chore(deps): bump some-package-name to version 2.0.0 refs ASDF-12" + echo + colorPrint brightWhite "Commit types:" + colorPrint brightCyan "build: $(colorPrint white "Changes that affect the build system or external dependencies (example scopes: gulp, broccoli, npm)" -n)" + colorPrint brightCyan "ci: $(colorPrint white "Changes to CI configuration files and scripts (example scopes: Travis, Circle, BrowserStack, SauceLabs)" -n)" + colorPrint brightCyan "chore: $(colorPrint white "Changes which doesn't change source code or tests e.g. changes to the build process, auxiliary tools, libraries" -n)" + colorPrint brightCyan "docs: $(colorPrint white "Documentation only changes" -n)" + colorPrint brightCyan "feat: $(colorPrint white "A new feature" -n)" + colorPrint brightCyan "fix: $(colorPrint white "A bug fix" -n)" + colorPrint brightCyan "perf: $(colorPrint white "A code change that improves performance" -n)" + colorPrint brightCyan "refactor: $(colorPrint white "A code change that neither fixes a bug nor adds a feature" -n)" + colorPrint brightCyan "revert: $(colorPrint white "Revert a change previously introduced" -n)" + colorPrint brightCyan "test: $(colorPrint white "Adding missing tests or correcting existing tests" -n)" + echo + + colorPrint brightWhite "Reminders" + echo "Put newline before extended commit body" + echo "More details at $(underline "http://www.conventionalcommits.org")" + echo + exit 1 +fi \ No newline at end of file diff --git a/.gitea/conventional_commits/generate-version.sh b/.gitea/conventional_commits/generate-version.sh new file mode 100755 index 0000000..c11c908 --- /dev/null +++ b/.gitea/conventional_commits/generate-version.sh @@ -0,0 +1,95 @@ +#!/bin/bash + +# Rules for generating semantic versioning +# major: breaking change +# minor: feat, style +# patch: build, fix, perf, refactor, revert + +GENERATE_VERSION=$1 +echo "Generate version: $GENERATE_VERSION" + +PREVENT_REMOVE_FILE=$2 +TEMP_FILE_PATH=.gitea/conventional_commits/tmp + +LAST_TAG=$(git describe --tags --abbrev=0 --always) +echo "Last tag: #$LAST_TAG#" +PATTERN="^[0-9]+\.[0-9]+\.[0-9]+$" + +increment_version() { + local version=$1 + local increment=$2 + local major=$(echo $version | cut -d. -f1) + local minor=$(echo $version | cut -d. -f2) + local patch=$(echo $version | cut -d. -f3) + + if [ "$increment" == "major" ]; then + major=$((major + 1)) + minor=0 + patch=0 + elif [ "$increment" == "minor" ]; then + minor=$((minor + 1)) + patch=0 + elif [ "$increment" == "patch" ]; then + patch=$((patch + 1)) + fi + + echo "${major}.${minor}.${patch}" +} + +create_file() { + local with_range=$1 + if [ -s $TEMP_FILE_PATH/messages.txt ]; then + return 1 + fi + if [ "$with_range" == "true" ]; then + git log $LAST_TAG..HEAD --no-decorate --pretty=format:"%s" > $TEMP_FILE_PATH/messages.txt + else + git log --no-decorate --pretty=format:"%s" > $TEMP_FILE_PATH/messages.txt + fi +} + +get_commit_range() { + if [[ $LAST_TAG =~ $PATTERN ]]; then + create_file true + else + create_file + LAST_TAG="0.0.0" + fi + echo " " >> $TEMP_FILE_PATH/messages.txt +} + +start() { + mkdir -p $TEMP_FILE_PATH + get_commit_range + new_version=$LAST_TAG + increment_type="" + + while read message; do + if [[ $message =~ (([a-z]+)(\(.+\))?\!:)|(BREAKING CHANGE:) ]]; then + increment_type="major" + break + elif [[ $message =~ (^(feat|style)(\(.+\))?:) ]]; then + if [ -z "$increment_type" ] || [ "$increment_type" == "patch" ]; then + increment_type="minor" + fi + elif [[ $message =~ ^((fix|build|perf|refactor|revert)(\(.+\))?:) ]]; then + if [ -z "$increment_type" ]; then + increment_type="patch" + fi + fi + done < $TEMP_FILE_PATH/messages.txt + + if [ -n "$increment_type" ]; then + new_version=$(increment_version $LAST_TAG $increment_type) + echo "New version: $new_version" + push_newversion $new_version + else + echo "No changes requiring a version increment." + fi +} + +start + +if [ -z "$PREVENT_REMOVE_FILE" ]; then + rm -f $TEMP_FILE_PATH/messages.txt +fi \ No newline at end of file diff --git a/.gitea/conventional_commits/hooks/text-styles.sh b/.gitea/conventional_commits/hooks/text-styles.sh new file mode 100755 index 0000000..452941f --- /dev/null +++ b/.gitea/conventional_commits/hooks/text-styles.sh @@ -0,0 +1,44 @@ +!/usr/bin/env sh + +colorPrint() { + local color=$1 + local text=$2 + shift 2 + local newline="\n" + local tab="" + + for arg in "$@" + do + if [ "$arg" = "-t" ]; then + tab="\t" + elif [ "$arg" = "-n" ]; then + newline="" + fi + done + + case $color in + black) color_code="30" ;; + red) color_code="31" ;; + green) color_code="32" ;; + yellow) color_code="33" ;; + blue) color_code="34" ;; + magenta) color_code="35" ;; + cyan) color_code="36" ;; + white) color_code="37" ;; + brightBlack) color_code="90" ;; + brightRed) color_code="91" ;; + brightGreen) color_code="92" ;; + brightYellow) color_code="93" ;; + brightBlue) color_code="94" ;; + brightMagenta) color_code="95" ;; + brightCyan) color_code="96" ;; + brightWhite) color_code="97" ;; + *) echo "Invalid color"; return ;; + esac + + printf "\e[${color_code}m${tab}%s\e[0m${newline}" "$text" +} + +underline () { + printf "\033[4m%s\033[24m" "$1" +} \ No newline at end of file diff --git a/.gitignore b/.gitignore index 31db540..185aa60 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,3 @@ **/bin -**/obj \ No newline at end of file +**/obj +.gitea/conventional_commits/tmp/* \ No newline at end of file diff --git a/Makefile b/Makefile index 9f0cf92..30a2456 100644 --- a/Makefile +++ b/Makefile @@ -1,5 +1,13 @@ .ONESHELL: +.PHONY: issetup +issetup: + @[ -f .git/hooks/commit-msg ] || (echo "You must run 'make setup' first to initialize the repo!" && exit 1) + +.PHONY: setup +setup: + @cp .gitea/conventional_commits/commit-msg .git/hooks/ + .PHONY: help help: ## Show the help. @echo "Usage: make " @@ -8,35 +16,35 @@ help: ## Show the help. @fgrep "##" Makefile | fgrep -v fgrep .PHONY: buildrel -buildrel: ## Format code. +buildrel: issetup ## Format code. dotnet build -c Release DotnetTestLib.sln .PHONY: build -build: ## Format code. +build: issetup ## Format code. dotnet build DotnetTestLib.sln .PHONY: publish -publish: ## Format code. +publish: issetup ## Format code. dotnet publish -c Release DotnetTestLib.sln .PHONY: fmt -fmt: ## Format code. +fmt: issetup ## Format code. dotnet format DotnetTestLib.sln .PHONY: lint -lint: ## Lint code. +lint: issetup ## Lint code. dotnet format --verify-no-changes --verbosity diagnostic DotnetTestLib.sln .PHONY: test -test: ## Run tests and generate coverage report. +test: issetup ## Run tests and generate coverage report. dotnet test DotnetTestLib.sln .PHONY: clean -clean: ## Clean unused files. +clean: issetup ## Clean unused files. dotnet clean DotnetTestLib.sln .PHONY: release -release: ## Create a new tag for release. +release: issetup ## Create a new tag for release. @echo "WARNING: This operation will create a version tag and push to gitea" @read -p "Version? (provide the next x.y.z semver) : " TAG; echo "$$TAG" > DotnetTestLib/VERSION @gitchangelog > HISTORY.md @@ -48,7 +56,7 @@ release: ## Create a new tag for release. @echo "Gitea Actions will detect the new tag and release the new version." .PHONY: docs -docs: ## Build the documentation. +docs: issetup ## Build the documentation. @echo "building documentation ..." @$(ENV_PREFIX)mkdocs build URL="site/index.html"; xdg-open $$URL || sensible-browser $$URL || x-www-browser $$URL || gnome-open $$URL || open $$URL