534 lines
18 KiB
Bash
Executable File
534 lines
18 KiB
Bash
Executable File
#!/bin/bash
|
|
set -e
|
|
|
|
# Colors for output
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
BLUE='\033[0;34m'
|
|
NC='\033[0m'
|
|
|
|
print_status() { echo -e "${BLUE}[INFO]${NC} $1"; }
|
|
print_success() { echo -e "${GREEN}[SUCCESS]${NC} $1"; }
|
|
print_warning() { echo -e "${YELLOW}[WARNING]${NC} $1"; }
|
|
print_error() { echo -e "${RED}[ERROR]${NC} $1"; }
|
|
|
|
# Global variables
|
|
COMMIT_MESSAGE=""
|
|
RELEASE_MODE=false
|
|
VERSION_INCREMENT_TYPE="patch" # "patch", "minor", or "major"
|
|
|
|
show_usage() {
|
|
echo "C-Relay Increment and Push Script"
|
|
echo ""
|
|
echo "USAGE:"
|
|
echo " $0 [OPTIONS] \"commit message\""
|
|
echo ""
|
|
echo "COMMANDS:"
|
|
echo " $0 \"commit message\" Default: increment patch, commit & push"
|
|
echo " $0 -p \"commit message\" Increment patch version"
|
|
echo " $0 -m \"commit message\" Increment minor version"
|
|
echo " $0 -M \"commit message\" Increment major version"
|
|
echo " $0 -r \"commit message\" Create release with assets (no version increment)"
|
|
echo " $0 -r -m \"commit message\" Create release with minor version increment"
|
|
echo " $0 -h Show this help message"
|
|
echo ""
|
|
echo "OPTIONS:"
|
|
echo " -p, --patch Increment patch version (default)"
|
|
echo " -m, --minor Increment minor version"
|
|
echo " -M, --major Increment major version"
|
|
echo " -r, --release Create release with assets"
|
|
echo " -h, --help Show this help message"
|
|
echo ""
|
|
echo "EXAMPLES:"
|
|
echo " $0 \"Fixed event validation bug\""
|
|
echo " $0 -m \"Added new features\""
|
|
echo " $0 -M \"Breaking API changes\""
|
|
echo " $0 -r \"Release current version\""
|
|
echo " $0 -r -m \"Release with minor increment\""
|
|
echo ""
|
|
echo "VERSION INCREMENT MODES:"
|
|
echo " -p, --patch (default): Increment patch version (v1.2.3 → v1.2.4)"
|
|
echo " -m, --minor: Increment minor version, zero patch (v1.2.3 → v1.3.0)"
|
|
echo " -M, --major: Increment major version, zero minor+patch (v1.2.3 → v2.0.0)"
|
|
echo ""
|
|
echo "RELEASE MODE (-r flag):"
|
|
echo " - Build static binary using build_static.sh"
|
|
echo " - Create source tarball"
|
|
echo " - Git add, commit, push, and create Gitea release with assets"
|
|
echo " - Can be combined with version increment flags"
|
|
echo ""
|
|
echo "REQUIREMENTS FOR RELEASE MODE:"
|
|
echo " - Gitea token in ~/.gitea_token for release uploads"
|
|
echo " - Docker installed for static binary builds"
|
|
}
|
|
|
|
# Parse command line arguments
|
|
while [[ $# -gt 0 ]]; do
|
|
case $1 in
|
|
-r|--release)
|
|
RELEASE_MODE=true
|
|
shift
|
|
;;
|
|
-p|--patch)
|
|
VERSION_INCREMENT_TYPE="patch"
|
|
shift
|
|
;;
|
|
-m|--minor)
|
|
VERSION_INCREMENT_TYPE="minor"
|
|
shift
|
|
;;
|
|
-M|--major)
|
|
VERSION_INCREMENT_TYPE="major"
|
|
shift
|
|
;;
|
|
-h|--help)
|
|
show_usage
|
|
exit 0
|
|
;;
|
|
*)
|
|
# First non-flag argument is the commit message
|
|
if [[ -z "$COMMIT_MESSAGE" ]]; then
|
|
COMMIT_MESSAGE="$1"
|
|
fi
|
|
shift
|
|
;;
|
|
esac
|
|
done
|
|
|
|
# Validate inputs
|
|
if [[ -z "$COMMIT_MESSAGE" ]]; then
|
|
print_error "Commit message is required"
|
|
echo ""
|
|
show_usage
|
|
exit 1
|
|
fi
|
|
|
|
# Check if we're in a git repository
|
|
check_git_repo() {
|
|
if ! git rev-parse --git-dir > /dev/null 2>&1; then
|
|
print_error "Not in a git repository"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# Function to get current version and increment appropriately
|
|
increment_version() {
|
|
local increment_type="$1" # "patch", "minor", or "major"
|
|
|
|
print_status "Getting current version..."
|
|
|
|
# Get the highest version tag (not chronologically latest)
|
|
LATEST_TAG=$(git tag -l 'v*.*.*' | sort -V | tail -n 1 || echo "")
|
|
if [[ -z "$LATEST_TAG" ]]; then
|
|
LATEST_TAG="v0.0.0"
|
|
print_warning "No version tags found, starting from $LATEST_TAG"
|
|
fi
|
|
|
|
# Extract version components (remove 'v' prefix)
|
|
VERSION=${LATEST_TAG#v}
|
|
|
|
# Parse major.minor.patch using regex
|
|
if [[ $VERSION =~ ^([0-9]+)\.([0-9]+)\.([0-9]+)$ ]]; then
|
|
MAJOR=${BASH_REMATCH[1]}
|
|
MINOR=${BASH_REMATCH[2]}
|
|
PATCH=${BASH_REMATCH[3]}
|
|
else
|
|
print_error "Invalid version format in tag: $LATEST_TAG"
|
|
print_error "Expected format: v0.1.0"
|
|
exit 1
|
|
fi
|
|
|
|
# Increment version based on type
|
|
if [[ "$increment_type" == "major" ]]; then
|
|
# Major release: increment major, zero minor and patch
|
|
NEW_MAJOR=$((MAJOR + 1))
|
|
NEW_MINOR=0
|
|
NEW_PATCH=0
|
|
NEW_VERSION="v${NEW_MAJOR}.${NEW_MINOR}.${NEW_PATCH}"
|
|
print_status "Major version increment: incrementing major version"
|
|
elif [[ "$increment_type" == "minor" ]]; then
|
|
# Minor release: increment minor, zero patch
|
|
NEW_MAJOR=$MAJOR
|
|
NEW_MINOR=$((MINOR + 1))
|
|
NEW_PATCH=0
|
|
NEW_VERSION="v${NEW_MAJOR}.${NEW_MINOR}.${NEW_PATCH}"
|
|
print_status "Minor version increment: incrementing minor version"
|
|
else
|
|
# Default: increment patch
|
|
NEW_MAJOR=$MAJOR
|
|
NEW_MINOR=$MINOR
|
|
NEW_PATCH=$((PATCH + 1))
|
|
NEW_VERSION="v${NEW_MAJOR}.${NEW_MINOR}.${NEW_PATCH}"
|
|
print_status "Patch version increment: incrementing patch version"
|
|
fi
|
|
|
|
print_status "Current version: $LATEST_TAG"
|
|
print_status "New version: $NEW_VERSION"
|
|
|
|
# Update version in src/main.h
|
|
update_version_in_header "$NEW_VERSION" "$NEW_MAJOR" "$NEW_MINOR" "$NEW_PATCH"
|
|
|
|
# Export for use in other functions
|
|
export NEW_VERSION
|
|
}
|
|
|
|
# Function to update version macros in src/main.h
|
|
update_version_in_header() {
|
|
local new_version="$1"
|
|
local major="$2"
|
|
local minor="$3"
|
|
local patch="$4"
|
|
|
|
print_status "Updating version in src/main.h..."
|
|
|
|
# Check if src/main.h exists
|
|
if [[ ! -f "src/main.h" ]]; then
|
|
print_error "src/main.h not found"
|
|
exit 1
|
|
fi
|
|
|
|
# Update VERSION macro
|
|
sed -i "s/#define VERSION \".*\"/#define VERSION \"$new_version\"/" src/main.h
|
|
|
|
# Update VERSION_MAJOR macro
|
|
sed -i "s/#define VERSION_MAJOR [0-9]\+/#define VERSION_MAJOR $major/" src/main.h
|
|
|
|
# Update VERSION_MINOR macro
|
|
sed -i "s/#define VERSION_MINOR .*/#define VERSION_MINOR $minor/" src/main.h
|
|
|
|
# Update VERSION_PATCH macro
|
|
sed -i "s/#define VERSION_PATCH [0-9]\+/#define VERSION_PATCH $patch/" src/main.h
|
|
|
|
print_success "Updated version in src/main.h to $new_version"
|
|
}
|
|
|
|
# Function to commit and push changes
|
|
git_commit_and_push() {
|
|
print_status "Preparing git commit..."
|
|
|
|
# Stage all changes
|
|
if git add . > /dev/null 2>&1; then
|
|
print_success "Staged all changes"
|
|
else
|
|
print_error "Failed to stage changes"
|
|
exit 1
|
|
fi
|
|
|
|
# Check if there are changes to commit
|
|
if git diff --staged --quiet; then
|
|
print_warning "No changes to commit"
|
|
else
|
|
# Commit changes
|
|
if git commit -m "$NEW_VERSION - $COMMIT_MESSAGE" > /dev/null 2>&1; then
|
|
print_success "Committed changes"
|
|
else
|
|
print_error "Failed to commit changes"
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
# Create new git tag
|
|
if git tag "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Created tag: $NEW_VERSION"
|
|
else
|
|
print_warning "Tag $NEW_VERSION already exists"
|
|
fi
|
|
|
|
# Push changes and tags
|
|
print_status "Pushing to remote repository..."
|
|
if git push > /dev/null 2>&1; then
|
|
print_success "Pushed changes"
|
|
else
|
|
print_error "Failed to push changes"
|
|
exit 1
|
|
fi
|
|
|
|
# Push only the new tag to avoid conflicts with existing tags
|
|
if git push origin "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Pushed tag: $NEW_VERSION"
|
|
else
|
|
print_warning "Tag push failed, trying force push..."
|
|
if git push --force origin "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Force-pushed updated tag: $NEW_VERSION"
|
|
else
|
|
print_error "Failed to push tag: $NEW_VERSION"
|
|
exit 1
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Function to commit and push changes without creating a tag (tag already created)
|
|
git_commit_and_push_no_tag() {
|
|
print_status "Preparing git commit..."
|
|
|
|
# Stage all changes
|
|
if git add . > /dev/null 2>&1; then
|
|
print_success "Staged all changes"
|
|
else
|
|
print_error "Failed to stage changes"
|
|
exit 1
|
|
fi
|
|
|
|
# Check if there are changes to commit
|
|
if git diff --staged --quiet; then
|
|
print_warning "No changes to commit"
|
|
else
|
|
# Commit changes
|
|
if git commit -m "$NEW_VERSION - $COMMIT_MESSAGE" > /dev/null 2>&1; then
|
|
print_success "Committed changes"
|
|
else
|
|
print_error "Failed to commit changes"
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
# Push changes and tags
|
|
print_status "Pushing to remote repository..."
|
|
if git push > /dev/null 2>&1; then
|
|
print_success "Pushed changes"
|
|
else
|
|
print_error "Failed to push changes"
|
|
exit 1
|
|
fi
|
|
|
|
# Push only the new tag to avoid conflicts with existing tags
|
|
if git push origin "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Pushed tag: $NEW_VERSION"
|
|
else
|
|
print_warning "Tag push failed, trying force push..."
|
|
if git push --force origin "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Force-pushed updated tag: $NEW_VERSION"
|
|
else
|
|
print_error "Failed to push tag: $NEW_VERSION"
|
|
exit 1
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Function to build release binary
|
|
build_release_binary() {
|
|
print_status "Building release binary..."
|
|
|
|
# Check if build_static.sh exists
|
|
if [[ ! -f "build_static.sh" ]]; then
|
|
print_error "build_static.sh not found"
|
|
return 1
|
|
fi
|
|
|
|
# Run the static build script
|
|
if ./build_static.sh > /dev/null 2>&1; then
|
|
print_success "Built static binary successfully"
|
|
return 0
|
|
else
|
|
print_error "Failed to build static binary"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# Function to create source tarball
|
|
create_source_tarball() {
|
|
print_status "Creating source tarball..."
|
|
|
|
local tarball_name="c-relay-${NEW_VERSION#v}.tar.gz"
|
|
|
|
# Create tarball excluding build artifacts and git files
|
|
if tar -czf "$tarball_name" \
|
|
--exclude='build/*' \
|
|
--exclude='.git*' \
|
|
--exclude='*.db' \
|
|
--exclude='*.db-*' \
|
|
--exclude='*.log' \
|
|
--exclude='*.tar.gz' \
|
|
. > /dev/null 2>&1; then
|
|
print_success "Created source tarball: $tarball_name"
|
|
echo "$tarball_name"
|
|
return 0
|
|
else
|
|
print_error "Failed to create source tarball"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# Function to upload release assets to Gitea
|
|
upload_release_assets() {
|
|
local release_id="$1"
|
|
local binary_path="$2"
|
|
local tarball_path="$3"
|
|
|
|
print_status "Uploading release assets..."
|
|
|
|
# Check for Gitea token
|
|
if [[ ! -f "$HOME/.gitea_token" ]]; then
|
|
print_warning "No ~/.gitea_token found. Skipping asset uploads."
|
|
return 0
|
|
fi
|
|
|
|
local token=$(cat "$HOME/.gitea_token" | tr -d '\n\r')
|
|
local api_url="https://git.laantungir.net/api/v1/repos/laantungir/c-relay"
|
|
|
|
# Upload binary
|
|
if [[ -f "$binary_path" ]]; then
|
|
print_status "Uploading binary: $(basename "$binary_path")"
|
|
local binary_response=$(curl -s -X POST "$api_url/releases/$release_id/assets" \
|
|
-H "Authorization: token $token" \
|
|
-H "Content-Type: application/octet-stream" \
|
|
-F "attachment=@$binary_path;filename=$(basename "$binary_path")")
|
|
|
|
if echo "$binary_response" | grep -q '"id"'; then
|
|
print_success "Uploaded binary successfully"
|
|
else
|
|
print_warning "Failed to upload binary: $binary_response"
|
|
fi
|
|
fi
|
|
|
|
# Upload source tarball
|
|
if [[ -f "$tarball_path" ]]; then
|
|
print_status "Uploading source tarball: $(basename "$tarball_path")"
|
|
local tarball_response=$(curl -s -X POST "$api_url/releases/$release_id/assets" \
|
|
-H "Authorization: token $token" \
|
|
-H "Content-Type: application/octet-stream" \
|
|
-F "attachment=@$tarball_path;filename=$(basename "$tarball_path")")
|
|
|
|
if echo "$tarball_response" | grep -q '"id"'; then
|
|
print_success "Uploaded source tarball successfully"
|
|
else
|
|
print_warning "Failed to upload source tarball: $tarball_response"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Function to create Gitea release
|
|
create_gitea_release() {
|
|
print_status "Creating Gitea release..."
|
|
|
|
# Check for Gitea token
|
|
if [[ ! -f "$HOME/.gitea_token" ]]; then
|
|
print_warning "No ~/.gitea_token found. Skipping release creation."
|
|
print_warning "Create ~/.gitea_token with your Gitea access token to enable releases."
|
|
return 0
|
|
fi
|
|
|
|
local token=$(cat "$HOME/.gitea_token" | tr -d '\n\r')
|
|
local api_url="https://git.laantungir.net/api/v1/repos/laantungir/c-relay"
|
|
|
|
# Create release
|
|
print_status "Creating release $NEW_VERSION..."
|
|
local response=$(curl -s -X POST "$api_url/releases" \
|
|
-H "Authorization: token $token" \
|
|
-H "Content-Type: application/json" \
|
|
-d "{\"tag_name\": \"$NEW_VERSION\", \"name\": \"$NEW_VERSION\", \"body\": \"$COMMIT_MESSAGE\"}")
|
|
|
|
if echo "$response" | grep -q '"id"'; then
|
|
print_success "Created release $NEW_VERSION"
|
|
# Extract release ID for asset uploads
|
|
local release_id=$(echo "$response" | grep -o '"id":[0-9]*' | cut -d':' -f2)
|
|
echo $release_id
|
|
elif echo "$response" | grep -q "already exists"; then
|
|
print_warning "Release $NEW_VERSION already exists"
|
|
# Try to get existing release ID
|
|
local check_response=$(curl -s -H "Authorization: token $token" "$api_url/releases/tags/$NEW_VERSION")
|
|
if echo "$check_response" | grep -q '"id"'; then
|
|
local release_id=$(echo "$check_response" | grep -o '"id":[0-9]*' | cut -d':' -f2)
|
|
print_status "Using existing release ID: $release_id"
|
|
echo $release_id
|
|
fi
|
|
echo 0
|
|
else
|
|
print_error "Failed to create release $NEW_VERSION"
|
|
print_error "Response: $response"
|
|
|
|
# Try to check if the release exists anyway
|
|
print_status "Checking if release exists..."
|
|
local check_response=$(curl -s -H "Authorization: token $token" "$api_url/releases/tags/$NEW_VERSION")
|
|
if echo "$check_response" | grep -q '"id"'; then
|
|
print_warning "Release exists but creation response was unexpected"
|
|
local release_id=$(echo "$check_response" | grep -o '"id":[0-9]*' | cut -d':' -f2)
|
|
echo $release_id
|
|
else
|
|
print_error "Release does not exist and creation failed"
|
|
echo 1
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Main execution
|
|
main() {
|
|
print_status "C-Relay Increment and Push Script"
|
|
|
|
# Check prerequisites
|
|
check_git_repo
|
|
|
|
if [[ "$RELEASE_MODE" == true ]]; then
|
|
print_status "=== RELEASE MODE ==="
|
|
|
|
# Only increment version if explicitly requested (not just because of -r flag)
|
|
if [[ "$VERSION_INCREMENT_TYPE" != "patch" ]]; then
|
|
increment_version "$VERSION_INCREMENT_TYPE"
|
|
fi
|
|
|
|
# Create new git tag BEFORE compilation so version.h picks it up
|
|
if git tag "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Created tag: $NEW_VERSION"
|
|
else
|
|
print_warning "Tag $NEW_VERSION already exists, removing and recreating..."
|
|
git tag -d "$NEW_VERSION" > /dev/null 2>&1
|
|
git tag "$NEW_VERSION" > /dev/null 2>&1
|
|
fi
|
|
|
|
# Commit and push (but skip tag creation since we already did it)
|
|
git_commit_and_push_no_tag
|
|
|
|
# Build release binary
|
|
if build_release_binary; then
|
|
local binary_path="build/c_relay_static_x86_64"
|
|
else
|
|
print_warning "Binary build failed, continuing with release creation"
|
|
binary_path=""
|
|
fi
|
|
|
|
# Create source tarball
|
|
local tarball_path=""
|
|
if tarball_path=$(create_source_tarball); then
|
|
: # tarball_path is set by the function
|
|
else
|
|
print_warning "Source tarball creation failed, continuing with release creation"
|
|
fi
|
|
|
|
# Create Gitea release
|
|
local release_id=""
|
|
if release_id=$(create_gitea_release); then
|
|
# Upload assets if we have a release ID and assets
|
|
if [[ -n "$release_id" && (-n "$binary_path" || -n "$tarball_path") ]]; then
|
|
upload_release_assets "$release_id" "$binary_path" "$tarball_path"
|
|
fi
|
|
print_success "Release $NEW_VERSION completed successfully!"
|
|
else
|
|
print_error "Release creation failed"
|
|
fi
|
|
|
|
else
|
|
print_status "=== DEFAULT MODE ==="
|
|
|
|
# Increment version based on type (default to patch)
|
|
increment_version "$VERSION_INCREMENT_TYPE"
|
|
|
|
# Create new git tag BEFORE compilation so version.h picks it up
|
|
if git tag "$NEW_VERSION" > /dev/null 2>&1; then
|
|
print_success "Created tag: $NEW_VERSION"
|
|
else
|
|
print_warning "Tag $NEW_VERSION already exists, removing and recreating..."
|
|
git tag -d "$NEW_VERSION" > /dev/null 2>&1
|
|
git tag "$NEW_VERSION" > /dev/null 2>&1
|
|
fi
|
|
|
|
# Commit and push (but skip tag creation since we already did it)
|
|
git_commit_and_push_no_tag
|
|
|
|
print_success "Increment and push completed successfully!"
|
|
print_status "Version $NEW_VERSION pushed to repository"
|
|
fi
|
|
}
|
|
|
|
# Execute main function
|
|
main |