diff --git a/.editorconfig b/.editorconfig index 4836cb688..7f92ca809 100644 --- a/.editorconfig +++ b/.editorconfig @@ -1,14 +1,5 @@ root = true -[.editorconfig] -charset = utf-8 -end_of_line = lf -indent_size = 4 -indent_style = tab -insert_final_newline = true -tab_width = 4 -trim_trailing_whitespace = true - [*] charset = utf-8 end_of_line = lf @@ -18,20 +9,7 @@ insert_final_newline = true tab_width = 4 trim_trailing_whitespace = true -[*.sh] -charset = utf-8 -end_of_line = lf -indent_size = 2 -indent_style = space -insert_final_newline = true -tab_width = 2 -trim_trailing_whitespace = true - -[CMakeLists.txt] -charset = utf-8 -end_of_line = lf +[{*.sh,*.yml,CMakeLists.txt}] indent_size = 2 indent_style = space -insert_final_newline = true tab_width = 2 -trim_trailing_whitespace = true diff --git a/.github/generate-email.sh b/.github/generate-email.sh new file mode 100755 index 000000000..4b53696d9 --- /dev/null +++ b/.github/generate-email.sh @@ -0,0 +1,97 @@ +#!/bin/bash +# +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to you under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +# Enable strict mode +set -euo pipefail +IFS=$'\n\t' + +stderr() { + echo "$*" 1>&2 +} + +fail_for_invalid_args() { + stderr "Invalid arguments!" + stderr "Expected arguments: " + exit 1 +} + +# Check arguments +[ $# -ne 3 ] && fail_for_invalid_args + +# Constants +PROJECT_NAME="Apache Log4cxx" +PROJECT_ID="log4cxx" +PROJECT_VERSION="$2" +PROJECT_SITE="https://logging.apache.org/$PROJECT_ID" +PROJECT_STAGING_SITE="${PROJECT_SITE/apache.org/staged.apache.org}" +PROJECT_REPO="https://github.com/apache/logging-log4cxx" +COMMIT_ID="$3" +PROJECT_DIST_URL="https://dist.apache.org/repos/dist/dev/logging/$PROJECT_ID/$PROJECT_VERSION" + +case $1 in + +vote) + cat < "apache-log4cxx-$VERSION.tar.gz.sha512" - sha256sum "apache-log4cxx-$VERSION.tar.gz" > "apache-log4cxx-$VERSION.tar.gz.sha256" - tar xf "apache-log4cxx-$VERSION.tar.gz" - zip -rm "apache-log4cxx-$VERSION.zip" apache-log4cxx-$VERSION - sha512sum "apache-log4cxx-$VERSION.zip" > "apache-log4cxx-$VERSION.zip.sha512" - sha256sum "apache-log4cxx-$VERSION.zip" > "apache-log4cxx-$VERSION.zip.sha256" - - - uses: actions/upload-artifact@v4 - if: always() - with: - name: 'Upload release files' - path: | - apache-log4cxx-* + - name: Checkout repository + uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # 4.2.1 + with: + persist-credentials: false # do not persist auth token in the local git config + + - name: Download artifacts + uses: actions/download-artifact@fa0a91b85d4f404e444e00e005971372dc801d16 # 4.1.8 + with: + name: apache-log4cxx + path: CMakeFiles/reference + + - name: Check release files + id: check + shell: bash + run: | + ./package.sh + VERSION=$(sed -n -e "s/^set(log4cxx_VER \(.*\)\.[[:digit:]]\+)/\1/p" < src/cmake/projectVersionDetails.cmake) + current=CMakeFiles/dist/apache-log4cxx-$VERSION + reference=CMakeFiles/reference/apache-log4cxx-$VERSION + for format in tar.gz zip; do + if ! cmp --silent "$reference.$format" "$current.$format"; then + echo Files apache-log4cxx-$VERSION.$format differ\! >& 2 + exit 1 + fi + done + + - name: Upload reproducibility results + uses: actions/upload-artifact@b4b15b8c7c6ac21ea08fcf65892d2ee8f75cf882 # 4.4.3 + if: ${{ failure() && steps.check.conclusion == 'failure' }} + with: + name: apache-log4cxx-reproducibility-${{ matrix.os }} + path: CMakeFiles/dist/* + + sign-and-upload: + name: Sign and upload + if: ${{ github.event_name == 'workflow_dispatch' }} + needs: + - package + - verify-reproducibility + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # 4.2.1 + with: + persist-credentials: false # do not persist auth token in the local git config + + - name: Download artifacts + uses: actions/download-artifact@fa0a91b85d4f404e444e00e005971372dc801d16 # 4.1.8 + with: + name: apache-log4cxx + + - name: Setup GPG +# uses: crazy-max/ghaction-import-gpg@01dd5d3ca463c7f10f7f4f7b4f177225ac661ee4 # 6.1.0 +# with: +# gpg_private_key: ${{ secrets.LOGGING_GPG_SECRET_KEY }} + # Using `setup-java` as temporary workaround, since `crazy-max` is not authorized + uses: actions/setup-java@b36c23c0d998641eff861008f374ee103c25ac73 # 3.7.0 + with: + distribution: temurin + java-version: 17 + gpg-private-key: ${{ secrets.LOGGING_GPG_SECRET_KEY }} + + - name: Check files and sign + env: + PROJECT_VERSION: ${{ inputs.project-version }} + shell: bash + run: | + for format in tar.gz zip; do + tmp=(apache-log4cxx-*.$format) + name="${tmp[@]}" + version=$(echo "$name" | sed -e "s/apache-log4cxx-\(.*\)\.$format/\1/") + if [ "$PROJECT_VERSION" != "$version" ]; then + echo Unexpected version number for file "$name" >& 2 + exit 1 + fi + sha256sum -c "$name.sha256" + sha512sum -c "$name.sha512" + # Sign + gpg --armor --detach-sign --yes --pinentry-mode error "$name" + done + + - name: Upload to Subversion + shell: bash + env: + DIST_FILENAME_PREFIX: apache-log4cxx + DIST_FILENAME_VERSIONED_PREFIX: apache-log4cxx-${{ inputs.project-version }} + PROJECT_ID: log4cxx + PROJECT_VERSION: ${{ inputs.project-version }} + SVN_USERNAME: ${{ secrets.LOGGING_SVN_DEV_USERNAME }} + SVN_PASSWORD: ${{ secrets.LOGGING_SVN_DEV_PASSWORD }} + run: | + # Find the effective Git commit ID + export COMMIT_ID="$GITHUB_SHA" + + # Checkout the SVN repository + export SVN_DIR="$(mktemp -d)/svn-repo" + svn co \ + "https://dist.apache.org/repos/dist/dev/logging/$PROJECT_ID" \ + "$SVN_DIR" + cd "$SVN_DIR" + + # Switch to the distribution folder + [ -d "$PROJECT_VERSION" ] || { + mkdir "$PROJECT_VERSION" + svn add "$PROJECT_VERSION" + } + cd "$PROJECT_VERSION" + + # Clean up old files + find . -name "${DIST_FILENAME_PREFIX}*" -type f -print0 | xargs -0 -r svn delete + + # Generate emails + for EMAIL_TYPE in vote announce; do + "$GITHUB_WORKSPACE/.github/generate-email.sh" \ + "$EMAIL_TYPE" "$PROJECT_VERSION" "$COMMIT_ID" \ + > "${DIST_FILENAME_VERSIONED_PREFIX}-email-${EMAIL_TYPE}.txt" + done + + # Copy the distribution + cp "$GITHUB_WORKSPACE/$DIST_FILENAME_VERSIONED_PREFIX"* . + + # Add & commit changes + svn add "$DIST_FILENAME_PREFIX"* + svn commit \ + --username "$SVN_USERNAME" \ + --password "$SVN_PASSWORD" \ + -m "Added \`${DIST_FILENAME_PREFIX}\` artifacts for release \`${PROJECT_VERSION}\`" diff --git a/package.sh b/package.sh new file mode 100755 index 000000000..bb4afa489 --- /dev/null +++ b/package.sh @@ -0,0 +1,108 @@ +#!/bin/bash +# +set -e + +# Shorthand for digit class: +d="[[:digit:]]" +# Determine the version and build timestamp +VERSION=$(sed -n -e "s/^set(log4cxx_VER \(.*\)\.$d\+)/\1/p" < src/cmake/projectVersionDetails.cmake) +if ! echo "$VERSION" | grep -Eq "^$d+\.$d+\.$d+$"; then + echo Invalid version number: "$VERSION" >& 2 + exit 1 +fi + +OUTPUT_TIMESTAMP=$(sed -n -e "s/^set(log4cxx_OUTPUT_TIMESTAMP \"\(.*\)\")/\1/p" < src/cmake/projectVersionDetails.cmake) +if ! echo "$OUTPUT_TIMESTAMP" | grep -Eq "^$d{4}-$d{2}-$d{2}T$d{2}:$d{2}:$d{2}Z$"; then + echo Invalid build timestamp: "$OUTPUT_TIMESTAMP" >& 2 + echo Run '`'date -u +%Y-%m-%dT%H:%M:%SZ'`' to generate it + exit 1 +fi + +# Build directory containing temporary files, all the paths are relative to it. +# +build=CMakeFiles + +# Create source directory +mkdir -p "$build/dist" +OUTPUT_DIR="apache-log4cxx-$VERSION" +if [ -f "$OUTPUT_DIR" ]; then + if [ ! -d "$OUTPUT_DIR" ]; then + echo File "$OUTPUT_DIR" is not a directory >& 2 + exit 1 + fi + if [ ! -z "$(ls -A "$OUTPUT_DIR")" ]; then + echo Directory "$OUTPUT_DIR" is not empty >& 2 + exit 1 + fi +fi +mkdir -p "$build/$OUTPUT_DIR" + +# Copy files to directory +cp -r \ + CMakeLists.txt \ + KEYS \ + INSTALL \ + LICENSE \ + NOTICE \ + README.md \ + src \ + liblog4cxx.pc.in \ + liblog4cxx-qt.pc.in \ + "$build/$OUTPUT_DIR" +rm -r "$build/$OUTPUT_DIR"/src/main/abi-symbols + +# Fix last-modified time +find "$build/$OUTPUT_DIR" -exec touch --date="$OUTPUT_TIMESTAMP" -m {} + + +# Create TAR file +# +# See https://reproducible-builds.org/docs/archives/ for reproducibility tips +TAR_ARCHIVE="dist/apache-log4cxx-$VERSION.tar.gz" +echo 'Tar version:' +tar --version | sed -e 's/^/\t/' +echo 'Gzip version:' +gzip --version | sed -e 's/^/\t/' +if [ -f "$TAR_ARCHIVE" ]; then + echo Archive "$TAR_ARCHIVE" already exists >& 2 + exit 1 +fi + +( + cd "$build" + tar --owner=0 --group=0 --numeric-owner \ + --sort=name \ + --pax-option=exthdr.name=%d/PaxHeaders/%f,delete=atime,delete=ctime \ + --create --gzip --file "$TAR_ARCHIVE" "$OUTPUT_DIR" +) +echo -e Tar archive: "$build/$TAR_ARCHIVE" + +# Create ZIP file +# +# See https://reproducible-builds.org/docs/archives/ for reproducibility tips +# Change the mtime of all files +ZIP_ARCHIVE="dist/apache-log4cxx-$VERSION.zip" +echo 'Zip version:' +zip --version | sed 's/^/\t/' +if [ -f "$ZIP_ARCHIVE" ]; then + echo Archive "$ZIP_ARCHIVE" already exists >& 2 + exit 1 +fi + +# Sort files and zip. +( + cd "$build" + find "$OUTPUT_DIR" -print0 | + LC_ALL=C sort -z | + xargs -0 zip -q -X "$ZIP_ARCHIVE" +) + +echo -e ZIP archive: "$build/$ZIP_ARCHIVE" + +# Generate hashes +( + cd "$build/dist" + for file in *; do + sha256sum "$file" > "$file.sha256" + sha512sum "$file" > "$file.sha512" + done +) diff --git a/src/cmake/projectVersionDetails.cmake b/src/cmake/projectVersionDetails.cmake index 362759913..96e0de095 100644 --- a/src/cmake/projectVersionDetails.cmake +++ b/src/cmake/projectVersionDetails.cmake @@ -1,5 +1,27 @@ -# This file should contain nothing but the following line -# setting the project version. The variable name must not -# clash with the log4cxx_VERSION* variables automatically +# +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +## +# This file should contain only the current version and the build timestamp + +# The variable name must not clash with the log4cxx_VERSION* variables automatically # defined by the project() command. set(log4cxx_VER 1.3.0.0) + +# Timestamp used for the source archive to guarantee reproducible builds in ISO 8601 format. +# +# To generate use: date -u +%Y-%m-%dT%H:%M:%SZ +set(log4cxx_OUTPUT_TIMESTAMP "2024-10-11T14:35:14Z")