Doc: update the NOTICE and LICENSE files #371
Workflow file for this run
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# -------------------------------------------------------------------- | |
# | |
# 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. | |
# | |
# -------------------------------------------------------------------- | |
# GitHub Actions Workflow: Apache Cloudberry Build Pipeline | |
# -------------------------------------------------------------------- | |
# Description: | |
# | |
# This workflow builds, tests, and packages Apache Cloudberry on | |
# Rocky Linux 9. It ensures artifact integrity, performs installation | |
# tests, validates key operations, and provides detailed test reports, | |
# including handling for ignored test cases. | |
# | |
# Workflow Overview: | |
# 1. **Check Skip**: | |
# - Dynamically determines if the workflow should run based on CI skip flags. | |
# - Evaluates the following fields for skip flags: | |
# - **Pull Request Events**: PR title and PR body. | |
# - **Push Events**: Commit message of the head commit. | |
# - Supports the following skip patterns (case-insensitive): | |
# - `[skip ci]` | |
# - `[ci skip]` | |
# - `[no ci]` | |
# - **Example Usage**: | |
# - Add `[skip ci]` to a commit message, PR title, or body to skip the workflow. | |
# | |
# 2. **Build Job**: | |
# - Configures and builds Apache Cloudberry. | |
# - Supports debug build configuration via ENABLE_DEBUG flag. | |
# - Runs unit tests and verifies build artifacts. | |
# - Creates RPM packages (regular or debug), source tarballs, and logs. | |
# - **Key Artifacts**: RPM package, source tarball, build logs. | |
# | |
# 3. **RPM Install Test Job**: | |
# - Verifies RPM integrity and installs Cloudberry. | |
# - Validates successful installation. | |
# - **Key Artifacts**: Installation logs, verification results. | |
# | |
# 4. **Test Job (Matrix)**: | |
# - Executes a test matrix to validate different scenarios. | |
# - Creates a demo cluster and runs installcheck tests. | |
# - Parses and reports test results, including failed and ignored tests. | |
# - Detects and analyzes any core dumps generated during tests. | |
# - **Key Features**: | |
# - Regression diffs are displayed if found, aiding quick debugging. | |
# - Both failed and ignored test names are logged and reported. | |
# - Core dumps are analyzed using GDB for stack traces. | |
# - **Key Artifacts**: Test logs, regression files, test summaries, core analyses. | |
# | |
# 5. **Report Job**: | |
# - Aggregates job results into a final report. | |
# - Sends failure notifications if any step fails. | |
# | |
# Execution Environment: | |
# - **Runs On**: ubuntu-22.04 with Rocky Linux 9 containers. | |
# - **Resource Requirements**: | |
# - Disk: Minimum 20GB free space. | |
# - Memory: Minimum 8GB RAM. | |
# - CPU: Recommended 4+ cores. | |
# | |
# Triggers: | |
# - Push to `main` branch. | |
# - Pull requests to `main` branch. | |
# - Manual workflow dispatch. | |
# | |
# Container Images: | |
# - **Build**: `apache/incubator-cloudberry:cbdb-build-rocky9-latest` | |
# - **Test**: `apache/incubator-cloudberry:cbdb-test-rocky9-latest` | |
# | |
# Artifacts: | |
# - RPM Package (retention: ${{ env.LOG_RETENTION_DAYS }} days). | |
# - Source Tarball (retention: ${{ env.LOG_RETENTION_DAYS }} days). | |
# - Logs and Test Results (retention: ${{ env.LOG_RETENTION_DAYS }} days). | |
# - Regression Diffs (retention: ${{ env.LOG_RETENTION_DAYS }} days). | |
# - Core Dump Analyses (retention: ${{ env.LOG_RETENTION_DAYS }} days). | |
# | |
# Notes: | |
# - Supports concurrent job execution. | |
# - Includes robust skip logic for pull requests and pushes. | |
# - Handles ignored test cases, ensuring results are comprehensive. | |
# - Provides detailed logs and error handling for failed and ignored tests. | |
# - Analyzes core dumps generated during test execution. | |
# - Supports debug builds with preserved symbols. | |
# -------------------------------------------------------------------- | |
name: Apache Cloudberry Build | |
on: | |
push: | |
branches: [main] | |
pull_request: | |
branches: [main] | |
types: [opened, synchronize, reopened, edited] | |
workflow_dispatch: | |
inputs: | |
test_selection: | |
description: 'Select tests to run (comma-separated). Examples: ic-good-opt-off,ic-contrib' | |
required: false | |
default: 'all' | |
type: string | |
concurrency: | |
group: ${{ github.workflow }}-${{ github.ref }} | |
cancel-in-progress: false | |
# Note: Step details, logs, and artifacts require users to be logged into GitHub | |
# even for public repositories. This is a GitHub security feature and cannot | |
# be overridden by permissions. | |
permissions: | |
# READ permissions allow viewing repository contents | |
contents: read # Required for checking out code and reading repository files | |
# READ permissions for packages (Container registry, etc) | |
packages: read # Allows reading from GitHub package registry | |
# WRITE permissions for actions includes read access to: | |
# - Workflow runs | |
# - Artifacts (requires GitHub login) | |
# - Logs (requires GitHub login) | |
actions: write | |
# READ permissions for checks API: | |
# - Step details visibility (requires GitHub login) | |
# - Check run status and details | |
checks: read | |
# READ permissions for pull request metadata: | |
# - PR status | |
# - Associated checks | |
# - Review states | |
pull-requests: read | |
env: | |
LOG_RETENTION_DAYS: 7 | |
ENABLE_DEBUG: false | |
jobs: | |
## ====================================================================== | |
## Job: check-skip | |
## ====================================================================== | |
check-skip: | |
runs-on: ubuntu-22.04 | |
outputs: | |
should_skip: ${{ steps.skip-check.outputs.should_skip }} | |
steps: | |
- id: skip-check | |
shell: bash | |
env: | |
EVENT_NAME: ${{ github.event_name }} | |
PR_TITLE: ${{ github.event.pull_request.title || '' }} | |
PR_BODY: ${{ github.event.pull_request.body || '' }} | |
run: | | |
# Default to not skipping | |
echo "should_skip=false" >> "$GITHUB_OUTPUT" | |
# Apply skip logic only for pull_request events | |
if [[ "$EVENT_NAME" == "pull_request" ]]; then | |
# Combine PR title and body for skip check | |
MESSAGE="${PR_TITLE}\n${PR_BODY}" | |
# Escape special characters using printf %s | |
ESCAPED_MESSAGE=$(printf "%s" "$MESSAGE") | |
echo "Checking PR title and body (escaped): $ESCAPED_MESSAGE" | |
# Check for skip patterns | |
if echo -e "$ESCAPED_MESSAGE" | grep -qEi '\[skip[ -]ci\]|\[ci[ -]skip\]|\[no[ -]ci\]'; then | |
echo "should_skip=true" >> "$GITHUB_OUTPUT" | |
fi | |
else | |
echo "Skip logic is not applied for $EVENT_NAME events." | |
fi | |
- name: Report Skip Status | |
if: steps.skip-check.outputs.should_skip == 'true' | |
run: | | |
echo "CI Skip flag detected in PR - skipping all checks." | |
exit 0 | |
## ====================================================================== | |
## Job: prepare-test-matrix | |
## ====================================================================== | |
prepare-test-matrix: | |
runs-on: ubuntu-22.04 | |
needs: [check-skip] | |
if: needs.check-skip.outputs.should_skip != 'true' | |
outputs: | |
test-matrix: ${{ steps.set-matrix.outputs.matrix }} | |
steps: | |
- id: set-matrix | |
run: | | |
echo "=== Matrix Preparation Diagnostics ===" | |
echo "Event type: ${{ github.event_name }}" | |
echo "Test selection input: '${{ github.event.inputs.test_selection }}'" | |
# Define defaults | |
DEFAULT_NUM_PRIMARY_MIRROR_PAIRS=3 | |
DEFAULT_ENABLE_CGROUPS=false | |
DEFAULT_ENABLE_CORE_CHECK=true | |
DEFAULT_PG_SETTINGS_OPTIMIZER="" | |
# Define base test configurations | |
ALL_TESTS='{ | |
"include": [ | |
{"test":"ic-good-opt-off", | |
"make_configs":["src/test/regress:installcheck-good"], | |
"pg_settings":{"optimizer":"off"} | |
}, | |
{"test":"ic-good-opt-on", | |
"make_configs":["src/test/regress:installcheck-good"], | |
"pg_settings":{"optimizer":"on"} | |
}, | |
{"test":"ic-expandshrink", | |
"make_configs":["src/test/isolation2:installcheck-expandshrink"] | |
}, | |
{"test":"ic-singlenode", | |
"make_configs":["src/test/isolation:installcheck-singlenode", | |
"src/test/singlenode_regress:installcheck-singlenode", | |
"src/test/singlenode_isolation2:installcheck-singlenode"], | |
"num_primary_mirror_pairs":0 | |
}, | |
{"test":"ic-resgroup-v2", | |
"make_configs":["src/test/isolation2:installcheck-resgroup-v2"], | |
"enable_cgroups":true | |
}, | |
{"test":"ic-contrib", | |
"make_configs":["contrib/auto_explain:installcheck", | |
"contrib/citext:installcheck", | |
"contrib/btree_gin:installcheck", | |
"contrib/file_fdw:installcheck", | |
"contrib/formatter_fixedwidth:installcheck", | |
"contrib/extprotocol:installcheck", | |
"contrib/dblink:installcheck", | |
"contrib/pg_trgm:installcheck", | |
"contrib/indexscan:installcheck", | |
"contrib/hstore:installcheck", | |
"contrib/pgcrypto:installcheck", | |
"contrib/tablefunc:installcheck", | |
"contrib/passwordcheck:installcheck", | |
"contrib/sslinfo:installcheck"] | |
}, | |
{"test":"ic-gpcontrib", | |
"make_configs":["gpcontrib/orafce:installcheck", | |
"gpcontrib/pxf_fdw:installcheck", | |
"gpcontrib/zstd:installcheck", | |
"gpcontrib/gp_sparse_vector:installcheck", | |
"gpcontrib/gp_toolkit:installcheck"] | |
}, | |
{"test":"ic-fixme", | |
"make_configs":["src/test/regress:installcheck-fixme"], | |
"enable_core_check":false | |
}, | |
{"test":"ic-isolation2", | |
"make_configs":["src/test/isolation2:installcheck-isolation2"] | |
}, | |
{"test":"ic-isolation2-crash", | |
"make_configs":["src/test/isolation2:installcheck-isolation2-crash"], | |
"enable_core_check":false | |
}, | |
{"test":"ic-parallel-retrieve-cursor", | |
"make_configs":["src/test/isolation2:installcheck-parallel-retrieve-cursor"] | |
} | |
] | |
}' | |
# Function to apply defaults | |
apply_defaults() { | |
echo "$1" | jq --arg npm "$DEFAULT_NUM_PRIMARY_MIRROR_PAIRS" \ | |
--argjson ec "$DEFAULT_ENABLE_CGROUPS" \ | |
--argjson ecc "$DEFAULT_ENABLE_CORE_CHECK" \ | |
--arg opt "$DEFAULT_PG_SETTINGS_OPTIMIZER" \ | |
'def get_defaults: | |
{ | |
num_primary_mirror_pairs: ($npm|tonumber), | |
enable_cgroups: $ec, | |
enable_core_check: $ecc, | |
pg_settings: { | |
optimizer: $opt | |
} | |
}; | |
get_defaults * .' | |
} | |
# Extract all valid test names from ALL_TESTS | |
VALID_TESTS=$(echo "$ALL_TESTS" | jq -r '.include[].test') | |
# Parse input test selection | |
IFS=',' read -ra SELECTED_TESTS <<< "${{ github.event.inputs.test_selection }}" | |
# Default to all tests if selection is empty or 'all' | |
if [[ "${SELECTED_TESTS[*]}" == "all" || -z "${SELECTED_TESTS[*]}" ]]; then | |
mapfile -t SELECTED_TESTS <<< "$VALID_TESTS" | |
fi | |
# Validate and filter selected tests | |
INVALID_TESTS=() | |
FILTERED_TESTS=() | |
for TEST in "${SELECTED_TESTS[@]}"; do | |
TEST=$(echo "$TEST" | tr -d '[:space:]') # Trim whitespace | |
if echo "$VALID_TESTS" | grep -qw "$TEST"; then | |
FILTERED_TESTS+=("$TEST") | |
else | |
INVALID_TESTS+=("$TEST") | |
fi | |
done | |
# Handle invalid tests | |
if [[ ${#INVALID_TESTS[@]} -gt 0 ]]; then | |
echo "::error::Invalid test(s) selected: ${INVALID_TESTS[*]}" | |
echo "Valid tests are: $(echo "$VALID_TESTS" | tr '\n' ', ')" | |
exit 1 | |
fi | |
# Build result JSON with defaults applied | |
RESULT='{"include":[' | |
FIRST=true | |
for TEST in "${FILTERED_TESTS[@]}"; do | |
CONFIG=$(jq -c --arg test "$TEST" '.include[] | select(.test == $test)' <<< "$ALL_TESTS") | |
FILTERED_WITH_DEFAULTS=$(apply_defaults "$CONFIG") | |
if [[ "$FIRST" == true ]]; then | |
FIRST=false | |
else | |
RESULT="${RESULT}," | |
fi | |
RESULT="${RESULT}${FILTERED_WITH_DEFAULTS}" | |
done | |
RESULT="${RESULT}]}" | |
# Output the matrix for GitHub Actions | |
echo "Final matrix configuration:" | |
echo "$RESULT" | jq . | |
# Fix: Use block redirection | |
{ | |
echo "matrix<<EOF" | |
echo "$RESULT" | |
echo "EOF" | |
} >> "$GITHUB_OUTPUT" | |
echo "=== Matrix Preparation Complete ===" | |
## ====================================================================== | |
## Job: build | |
## ====================================================================== | |
build: | |
name: Build Apache Cloudberry | |
env: | |
JOB_TYPE: build | |
needs: [check-skip] | |
runs-on: ubuntu-22.04 | |
timeout-minutes: 120 | |
outputs: | |
build_timestamp: ${{ steps.set_timestamp.outputs.timestamp }} | |
container: | |
image: apache/incubator-cloudberry:cbdb-build-rocky9-latest | |
options: >- | |
--user root | |
-h cdw | |
steps: | |
- name: Skip Check | |
if: needs.check-skip.outputs.should_skip == 'true' | |
run: | | |
echo "Build skipped via CI skip flag" >> "$GITHUB_STEP_SUMMARY" | |
exit 0 | |
- name: Set build timestamp | |
if: needs.check-skip.outputs.should_skip != 'true' | |
id: set_timestamp # Add an ID to reference this step | |
run: | | |
timestamp=$(date +'%Y%m%d_%H%M%S') | |
echo "timestamp=$timestamp" | tee -a "$GITHUB_OUTPUT" # Use GITHUB_OUTPUT for job outputs | |
echo "BUILD_TIMESTAMP=$timestamp" | tee -a "$GITHUB_ENV" # Also set as environment variable | |
- name: Checkout Apache Cloudberry | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/checkout@v4 | |
with: | |
fetch-depth: 1 | |
- name: Checkout CI Build/Test Scripts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/checkout@v4 | |
with: | |
repository: apache/cloudberry-devops-release | |
ref: main | |
path: cloudberry-devops-release | |
fetch-depth: 1 | |
- name: Move cloudberry-devops-release directory | |
if: needs.check-skip.outputs.should_skip != 'true' | |
run: | | |
set -eo pipefail | |
if ! mv "${GITHUB_WORKSPACE}"/cloudberry-devops-release "${GITHUB_WORKSPACE}"/..; then | |
echo "::error::Container initialization failed" | |
exit 1 | |
fi | |
- name: Cloudberry Environment Initialization | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
LOGS_DIR: build-logs | |
run: | | |
set -eo pipefail | |
if ! su - gpadmin -c "/tmp/init_system.sh"; then | |
echo "::error::Container initialization failed" | |
exit 1 | |
fi | |
mkdir -p "${LOGS_DIR}/details" | |
chown -R gpadmin:gpadmin . | |
chmod -R 755 . | |
chmod 777 "${LOGS_DIR}" | |
df -kh / | |
rm -rf /__t/* | |
df -kh / | |
df -h | tee -a "${LOGS_DIR}/details/disk-usage.log" | |
free -h | tee -a "${LOGS_DIR}/details/memory-usage.log" | |
{ | |
echo "=== Environment Information ===" | |
uname -a | |
df -h | |
free -h | |
env | |
} | tee -a "${LOGS_DIR}/details/environment.log" | |
echo "SRC_DIR=${GITHUB_WORKSPACE}" | tee -a "$GITHUB_ENV" | |
- name: Generate Build Job Summary Start | |
if: needs.check-skip.outputs.should_skip != 'true' | |
run: | | |
{ | |
echo "# Build Job Summary" | |
echo "## Environment" | |
echo "- Start Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
echo "- ENABLE_DEBUG: ${{ env.ENABLE_DEBUG }}" | |
echo "- OS Version: $(cat /etc/redhat-release)" | |
echo "- GCC Version: $(gcc --version | head -n1)" | |
} >> "$GITHUB_STEP_SUMMARY" | |
- name: Run Apache Cloudberry configure script | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
chmod +x "${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/configure-cloudberry.sh | |
if ! time su - gpadmin -c "cd ${SRC_DIR} && SRC_DIR=${SRC_DIR} ENABLE_DEBUG=${{ env.ENABLE_DEBUG }} ${SRC_DIR}/../cloudberry-devops-release/build_automation/cloudberry/scripts/configure-cloudberry.sh"; then | |
echo "::error::Configure script failed" | |
exit 1 | |
fi | |
- name: Run Apache Cloudberry build script | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
chmod +x "${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/build-cloudberry.sh | |
if ! time su - gpadmin -c "cd ${SRC_DIR} && SRC_DIR=${SRC_DIR} ${SRC_DIR}/../cloudberry-devops-release/build_automation/cloudberry/scripts/build-cloudberry.sh"; then | |
echo "::error::Build script failed" | |
exit 1 | |
fi | |
- name: Verify build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
run: | | |
set -eo pipefail | |
echo "Verifying build artifacts..." | |
{ | |
echo "=== Build Artifacts Verification ===" | |
echo "Timestamp: $(date -u)" | |
if [ ! -d "/usr/local/cloudberry-db" ]; then | |
echo "::error::Build artifacts directory not found" | |
exit 1 | |
fi | |
# Verify critical binaries | |
critical_binaries=( | |
"/usr/local/cloudberry-db/bin/postgres" | |
"/usr/local/cloudberry-db/bin/psql" | |
) | |
echo "Checking critical binaries..." | |
for binary in "${critical_binaries[@]}"; do | |
if [ ! -f "$binary" ]; then | |
echo "::error::Critical binary missing: $binary" | |
exit 1 | |
fi | |
if [ ! -x "$binary" ]; then | |
echo "::error::Binary not executable: $binary" | |
exit 1 | |
fi | |
echo "Binary verified: $binary" | |
ls -l "$binary" | |
done | |
# Test binary execution | |
echo "Testing binary execution..." | |
if ! /usr/local/cloudberry-db/bin/postgres --version; then | |
echo "::error::postgres binary verification failed" | |
exit 1 | |
fi | |
if ! /usr/local/cloudberry-db/bin/psql --version; then | |
echo "::error::psql binary verification failed" | |
exit 1 | |
fi | |
echo "All build artifacts verified successfully" | |
} 2>&1 | tee -a build-logs/details/build-verification.log | |
- name: Create Source tarball, create RPM and verify artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
CBDB_VERSION: 99.0.0 | |
BUILD_NUMBER: 1 | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
{ | |
echo "=== Artifact Creation Log ===" | |
echo "Timestamp: $(date -u)" | |
# Create source tarball | |
echo "Creating source tarball..." | |
tar czf "${SRC_DIR}"/../apache-cloudberry-incubating-src.tgz -C "${SRC_DIR}"/.. ./cloudberry | |
mv "${SRC_DIR}"/../apache-cloudberry-incubating-src.tgz "${SRC_DIR}" | |
# Verify tarball contents | |
echo "Verifying source tarball contents..." | |
if ! tar tzf "${SRC_DIR}"/apache-cloudberry-incubating-src.tgz > /dev/null; then | |
echo "::error::Source tarball verification failed" | |
exit 1 | |
fi | |
# Create RPM | |
echo "Creating RPM package..." | |
rpmdev-setuptree | |
ln -s "${SRC_DIR}"/../cloudberry-devops-release/packaging/rpm/el/SPECS/apache-cloudberry-db-incubating.spec "${HOME}"/rpmbuild/SPECS/apache-cloudberry-db-incubating.spec | |
cp "${SRC_DIR}"/LICENSE /usr/local/cloudberry-db | |
DEBUG_RPMBUILD_OPT="" | |
DEBUG_IDENTIFIER="" | |
if [ "${{ env.ENABLE_DEBUG }}" = "true" ]; then | |
DEBUG_RPMBUILD_OPT="--with-debug" | |
DEBUG_IDENTIFIER=".debug" | |
fi | |
"${SRC_DIR}"/../cloudberry-devops-release/scripts/build-rpm.sh --version "${CBDB_VERSION}" --release "${BUILD_NUMBER}" "${DEBUG_RPMBUILD_OPT}" | |
# Get OS version and move RPM | |
os_version=$(grep -oP '(?<=^VERSION_ID=")[0-9]' /etc/os-release) | |
RPM_FILE="${HOME}"/rpmbuild/RPMS/x86_64/apache-cloudberry-db-incubating-"${CBDB_VERSION}"-"${BUILD_NUMBER}""${DEBUG_IDENTIFIER}".el"${os_version}".x86_64.rpm | |
cp "${RPM_FILE}" "${SRC_DIR}" | |
# Get package information | |
echo "Package Information:" | |
rpm -qip "${RPM_FILE}" | |
# Verify critical files in RPM | |
echo "Verifying critical files in RPM..." | |
for binary in "bin/postgres" "bin/psql"; do | |
if ! rpm -qlp "${RPM_FILE}" | grep -q "${binary}$"; then | |
echo "::error::Critical binary '${binary}' not found in RPM" | |
exit 1 | |
fi | |
done | |
# Record checksums | |
echo "Calculating checksums..." | |
sha256sum "${RPM_FILE}" | tee -a build-logs/details/checksums.log | |
sha256sum "${SRC_DIR}"/apache-cloudberry-incubating-src.tgz | tee -a build-logs/details/checksums.log | |
echo "Artifacts created and verified successfully" | |
} 2>&1 | tee -a build-logs/details/artifact-creation.log | |
- name: Run Apache Cloudberry unittest script | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
chmod +x "${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/unittest-cloudberry.sh | |
if ! time su - gpadmin -c "cd ${SRC_DIR} && SRC_DIR=${SRC_DIR} ${SRC_DIR}/../cloudberry-devops-release/build_automation/cloudberry/scripts/unittest-cloudberry.sh"; then | |
echo "::error::Unittest script failed" | |
exit 1 | |
fi | |
- name: Generate Build Job Summary End | |
if: always() | |
run: | | |
{ | |
echo "## Build Results" | |
echo "- End Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
} >> "$GITHUB_STEP_SUMMARY" | |
- name: Upload build logs | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/upload-artifact@v4 | |
with: | |
name: build-logs-${{ env.BUILD_TIMESTAMP }} | |
path: | | |
build-logs/ | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
- name: Upload Cloudberry RPM build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/upload-artifact@v4 | |
with: | |
name: apache-cloudberry-db-incubating-rpm-build-artifacts | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
if-no-files-found: error | |
path: | | |
*.rpm | |
- name: Upload Cloudberry source build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/upload-artifact@v4 | |
with: | |
name: apache-cloudberry-db-incubating-source-build-artifacts | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
if-no-files-found: error | |
path: | | |
apache-cloudberry-incubating-src.tgz | |
## ====================================================================== | |
## Job: rpm-install-test | |
## ====================================================================== | |
rpm-install-test: | |
name: RPM Install Test Apache Cloudberry | |
needs: [check-skip, build] | |
runs-on: ubuntu-22.04 | |
timeout-minutes: 120 | |
container: | |
image: apache/incubator-cloudberry:cbdb-test-rocky9-latest | |
options: >- | |
--user root | |
-h cdw | |
steps: | |
- name: Skip Check | |
if: needs.check-skip.outputs.should_skip == 'true' | |
run: | | |
echo "RPM install test skipped via CI skip flag" >> "$GITHUB_STEP_SUMMARY" | |
exit 0 | |
- name: Download Cloudberry RPM build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/download-artifact@v4 | |
with: | |
name: apache-cloudberry-db-incubating-rpm-build-artifacts | |
path: ${{ github.workspace }}/rpm_build_artifacts | |
merge-multiple: false | |
- name: Cloudberry Environment Initialization | |
if: needs.check-skip.outputs.should_skip != 'true' | |
env: | |
LOGS_DIR: install-logs | |
run: | | |
set -eo pipefail | |
if ! su - gpadmin -c "/tmp/init_system.sh"; then | |
echo "::error::Container initialization failed" | |
exit 1 | |
fi | |
mkdir -p "${LOGS_DIR}/details" | |
chown -R gpadmin:gpadmin . | |
chmod -R 755 . | |
chmod 777 "${LOGS_DIR}" | |
df -kh / | |
rm -rf /__t/* | |
df -kh / | |
df -h | tee -a "${LOGS_DIR}/details/disk-usage.log" | |
free -h | tee -a "${LOGS_DIR}/details/memory-usage.log" | |
{ | |
echo "=== Environment Information ===" | |
uname -a | |
df -h | |
free -h | |
env | |
} | tee -a "${LOGS_DIR}/details/environment.log" | |
echo "SRC_DIR=${GITHUB_WORKSPACE}" | tee -a "$GITHUB_ENV" | |
- name: Verify RPM artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
id: verify-artifacts | |
run: | | |
set -eo pipefail | |
RPM_FILE=$(ls "${GITHUB_WORKSPACE}"/rpm_build_artifacts/apache-cloudberry-db-incubating*.rpm) | |
if [ ! -f "${RPM_FILE}" ]; then | |
echo "::error::RPM file not found" | |
exit 1 | |
fi | |
echo "rpm_file=${RPM_FILE}" >> "$GITHUB_OUTPUT" | |
echo "Verifying RPM artifacts..." | |
{ | |
echo "=== RPM Verification Summary ===" | |
echo "Timestamp: $(date -u)" | |
echo "RPM File: ${RPM_FILE}" | |
# Get RPM metadata and verify contents | |
echo "Package Information:" | |
rpm -qip "${RPM_FILE}" | |
# Get key RPM attributes for verification | |
RPM_VERSION=$(rpm -qp --queryformat "%{VERSION}" "${RPM_FILE}") | |
RPM_RELEASE=$(rpm -qp --queryformat "%{RELEASE}" "${RPM_FILE}") | |
echo "version=${RPM_VERSION}" >> "$GITHUB_OUTPUT" | |
echo "release=${RPM_RELEASE}" >> "$GITHUB_OUTPUT" | |
# Verify expected binaries are in the RPM | |
echo "Verifying critical files in RPM..." | |
for binary in "bin/postgres" "bin/psql"; do | |
if ! rpm -qlp "${RPM_FILE}" | grep -q "${binary}$"; then | |
echo "::error::Critical binary '${binary}' not found in RPM" | |
exit 1 | |
fi | |
done | |
echo "RPM Details:" | |
echo "- Version: ${RPM_VERSION}" | |
echo "- Release: ${RPM_RELEASE}" | |
# Calculate and store checksum | |
echo "Checksum:" | |
sha256sum "${RPM_FILE}" | |
} 2>&1 | tee -a install-logs/details/rpm-verification.log | |
- name: Install Cloudberry RPM | |
if: success() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
RPM_FILE: ${{ steps.verify-artifacts.outputs.rpm_file }} | |
RPM_VERSION: ${{ steps.verify-artifacts.outputs.version }} | |
RPM_RELEASE: ${{ steps.verify-artifacts.outputs.release }} | |
run: | | |
set -eo pipefail | |
if [ -z "${RPM_FILE}" ]; then | |
echo "::error::RPM_FILE environment variable is not set" | |
exit 1 | |
fi | |
{ | |
echo "=== RPM Installation Log ===" | |
echo "Timestamp: $(date -u)" | |
echo "RPM File: ${RPM_FILE}" | |
echo "Version: ${RPM_VERSION}" | |
echo "Release: ${RPM_RELEASE}" | |
# Clean install location | |
rm -rf /usr/local/cloudberry-db | |
# Install RPM | |
echo "Starting installation..." | |
if ! time dnf install -y "${RPM_FILE}"; then | |
echo "::error::RPM installation failed" | |
exit 1 | |
fi | |
echo "Installation completed successfully" | |
rpm -qi apache-cloudberry-db-incubating | |
echo "Installed files:" | |
rpm -ql apache-cloudberry-db-incubating | |
} 2>&1 | tee -a install-logs/details/rpm-installation.log | |
- name: Upload install logs | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/upload-artifact@v4 | |
with: | |
name: install-logs-${{ needs.build.outputs.build_timestamp }} | |
path: | | |
install-logs/ | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
- name: Generate Install Test Job Summary End | |
if: always() | |
shell: bash {0} | |
run: | | |
{ | |
echo "# Installed Package Summary" | |
echo "\`\`\`" | |
rpm -qi apache-cloudberry-db-incubating | |
echo "\`\`\`" | |
} >> "$GITHUB_STEP_SUMMARY" || true | |
## ====================================================================== | |
## Job: test | |
## ====================================================================== | |
test: | |
name: ${{ matrix.test }} | |
needs: [check-skip, build, prepare-test-matrix] | |
runs-on: ubuntu-22.04 | |
timeout-minutes: 120 | |
# actionlint-allow matrix[*].pg_settings | |
strategy: | |
fail-fast: false # Continue with other tests if one fails | |
matrix: ${{ fromJson(needs.prepare-test-matrix.outputs.test-matrix) }} | |
container: | |
image: apache/incubator-cloudberry:cbdb-build-rocky9-latest | |
options: >- | |
--privileged | |
--user root | |
--hostname cdw | |
--shm-size=2gb | |
--ulimit core=-1 | |
--cgroupns=host | |
-v /sys/fs/cgroup:/sys/fs/cgroup:rw | |
steps: | |
- name: Skip Check | |
if: needs.check-skip.outputs.should_skip == 'true' | |
run: | | |
echo "Test ${{ matrix.test }} skipped via CI skip flag" >> "$GITHUB_STEP_SUMMARY" | |
exit 0 | |
- name: Use timestamp from previous job | |
if: needs.check-skip.outputs.should_skip != 'true' | |
run: | | |
echo "Timestamp from output: ${{ needs.build.outputs.build_timestamp }}" | |
- name: Checkout CI Build/Test Scripts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/checkout@v4 | |
with: | |
repository: apache/cloudberry-devops-release | |
ref: main | |
path: cloudberry-devops-release | |
fetch-depth: 1 | |
- name: Move cloudberry-devops-release directory | |
if: needs.check-skip.outputs.should_skip != 'true' | |
run: | | |
set -eo pipefail | |
if ! mv "${GITHUB_WORKSPACE}"/cloudberry-devops-release "${GITHUB_WORKSPACE}"/..; then | |
echo "::error::Container initialization failed" | |
exit 1 | |
fi | |
- name: Cloudberry Environment Initialization | |
env: | |
LOGS_DIR: build-logs | |
run: | | |
set -eo pipefail | |
if ! su - gpadmin -c "/tmp/init_system.sh"; then | |
echo "::error::Container initialization failed" | |
exit 1 | |
fi | |
mkdir -p "${LOGS_DIR}/details" | |
chown -R gpadmin:gpadmin . | |
chmod -R 755 . | |
chmod 777 "${LOGS_DIR}" | |
df -kh / | |
rm -rf /__t/* | |
df -kh / | |
df -h | tee -a "${LOGS_DIR}/details/disk-usage.log" | |
free -h | tee -a "${LOGS_DIR}/details/memory-usage.log" | |
{ | |
echo "=== Environment Information ===" | |
uname -a | |
df -h | |
free -h | |
env | |
} | tee -a "${LOGS_DIR}/details/environment.log" | |
echo "SRC_DIR=${GITHUB_WORKSPACE}" | tee -a "$GITHUB_ENV" | |
- name: Setup cgroups | |
if: needs.check-skip.outputs.should_skip != 'true' | |
shell: bash | |
run: | | |
set -uxo pipefail | |
if [ "${{ matrix.enable_cgroups }}" = "true" ]; then | |
echo "Current mounts:" | |
mount | grep cgroup | |
CGROUP_BASEDIR=/sys/fs/cgroup | |
# 1. Basic setup with permissions | |
sudo chmod -R 777 ${CGROUP_BASEDIR}/ | |
sudo mkdir -p ${CGROUP_BASEDIR}/gpdb | |
sudo chmod -R 777 ${CGROUP_BASEDIR}/gpdb | |
sudo chown -R gpadmin:gpadmin ${CGROUP_BASEDIR}/gpdb | |
# 2. Enable controllers | |
sudo bash -c "echo '+cpu +cpuset +memory +io' > ${CGROUP_BASEDIR}/cgroup.subtree_control" || true | |
sudo bash -c "echo '+cpu +cpuset +memory +io' > ${CGROUP_BASEDIR}/gpdb/cgroup.subtree_control" || true | |
# 3. CPU settings | |
sudo bash -c "echo 'max 100000' > ${CGROUP_BASEDIR}/gpdb/cpu.max" || true | |
sudo bash -c "echo '100' > ${CGROUP_BASEDIR}/gpdb/cpu.weight" || true | |
sudo bash -c "echo '0' > ${CGROUP_BASEDIR}/gpdb/cpu.weight.nice" || true | |
sudo bash -c "echo 0-$(( $(nproc) - 1 )) > ${CGROUP_BASEDIR}/gpdb/cpuset.cpus" || true | |
sudo bash -c "echo '0' > ${CGROUP_BASEDIR}/gpdb/cpuset.mems" || true | |
# 4. Memory settings | |
sudo bash -c "echo 'max' > ${CGROUP_BASEDIR}/gpdb/memory.max" || true | |
sudo bash -c "echo '0' > ${CGROUP_BASEDIR}/gpdb/memory.min" || true | |
sudo bash -c "echo 'max' > ${CGROUP_BASEDIR}/gpdb/memory.high" || true | |
# 5. IO settings | |
echo "Available block devices:" | |
lsblk | |
sudo bash -c " | |
if [ -f \${CGROUP_BASEDIR}/gpdb/io.stat ]; then | |
echo 'Detected IO devices:' | |
cat \${CGROUP_BASEDIR}/gpdb/io.stat | |
fi | |
echo '' > \${CGROUP_BASEDIR}/gpdb/io.max || true | |
" | |
# 6. Fix permissions again after all writes | |
sudo chmod -R 777 ${CGROUP_BASEDIR}/gpdb | |
sudo chown -R gpadmin:gpadmin ${CGROUP_BASEDIR}/gpdb | |
# 7. Check required files | |
echo "Checking required files:" | |
required_files=( | |
"cgroup.procs" | |
"cpu.max" | |
"cpu.pressure" | |
"cpu.weight" | |
"cpu.weight.nice" | |
"cpu.stat" | |
"cpuset.cpus" | |
"cpuset.mems" | |
"cpuset.cpus.effective" | |
"cpuset.mems.effective" | |
"memory.current" | |
"io.max" | |
) | |
for file in "${required_files[@]}"; do | |
if [ -f "${CGROUP_BASEDIR}/gpdb/$file" ]; then | |
echo "✓ $file exists" | |
ls -l "${CGROUP_BASEDIR}/gpdb/$file" | |
else | |
echo "✗ $file missing" | |
fi | |
done | |
# 8. Test subdirectory creation | |
echo "Testing subdirectory creation..." | |
sudo -u gpadmin bash -c " | |
TEST_DIR=\${CGROUP_BASEDIR}/gpdb/test6448 | |
if mkdir -p \$TEST_DIR; then | |
echo 'Created test directory' | |
sudo chmod -R 777 \$TEST_DIR | |
if echo \$\$ > \$TEST_DIR/cgroup.procs; then | |
echo 'Successfully wrote to cgroup.procs' | |
cat \$TEST_DIR/cgroup.procs | |
# Move processes back to parent before cleanup | |
echo \$\$ > \${CGROUP_BASEDIR}/gpdb/cgroup.procs | |
else | |
echo 'Failed to write to cgroup.procs' | |
ls -la \$TEST_DIR/cgroup.procs | |
fi | |
ls -la \$TEST_DIR/ | |
rmdir \$TEST_DIR || { | |
echo 'Moving all processes to parent before cleanup' | |
cat \$TEST_DIR/cgroup.procs | while read pid; do | |
echo \$pid > \${CGROUP_BASEDIR}/gpdb/cgroup.procs 2>/dev/null || true | |
done | |
rmdir \$TEST_DIR | |
} | |
else | |
echo 'Failed to create test directory' | |
fi | |
" | |
# 9. Verify setup as gpadmin user | |
echo "Testing cgroup access as gpadmin..." | |
sudo -u gpadmin bash -c " | |
echo 'Checking mounts...' | |
mount | grep cgroup | |
echo 'Checking /proc/self/mounts...' | |
cat /proc/self/mounts | grep cgroup | |
if ! grep -q cgroup2 /proc/self/mounts; then | |
echo 'ERROR: cgroup2 mount NOT visible to gpadmin' | |
exit 1 | |
fi | |
echo 'SUCCESS: cgroup2 mount visible to gpadmin' | |
if ! [ -w ${CGROUP_BASEDIR}/gpdb ]; then | |
echo 'ERROR: gpadmin cannot write to gpdb cgroup' | |
exit 1 | |
fi | |
echo 'SUCCESS: gpadmin can write to gpdb cgroup' | |
echo 'Verifying key files content:' | |
echo 'cpu.max:' | |
cat ${CGROUP_BASEDIR}/gpdb/cpu.max || echo 'Failed to read cpu.max' | |
echo 'cpuset.cpus:' | |
cat ${CGROUP_BASEDIR}/gpdb/cpuset.cpus || echo 'Failed to read cpuset.cpus' | |
echo 'cgroup.subtree_control:' | |
cat ${CGROUP_BASEDIR}/gpdb/cgroup.subtree_control || echo 'Failed to read cgroup.subtree_control' | |
" | |
# 10. Show final state | |
echo "Final cgroup state:" | |
ls -la ${CGROUP_BASEDIR}/gpdb/ | |
echo "Cgroup setup completed successfully" | |
else | |
echo "Cgroup setup skipped" | |
fi | |
- name: "Generate Test Job Summary Start: ${{ matrix.test }}" | |
if: always() | |
run: | | |
{ | |
echo "# Test Job Summary: ${{ matrix.test }}" | |
echo "## Environment" | |
echo "- Start Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
if [[ "${{ needs.check-skip.outputs.should_skip }}" == "true" ]]; then | |
echo "## Skip Status" | |
echo "✓ Test execution skipped via CI skip flag" | |
else | |
echo "- OS Version: $(cat /etc/redhat-release)" | |
fi | |
} >> "$GITHUB_STEP_SUMMARY" | |
- name: Download Cloudberry RPM build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/download-artifact@v4 | |
with: | |
name: apache-cloudberry-db-incubating-rpm-build-artifacts | |
path: ${{ github.workspace }}/rpm_build_artifacts | |
merge-multiple: false | |
- name: Download Cloudberry Source build artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
uses: actions/download-artifact@v4 | |
with: | |
name: apache-cloudberry-db-incubating-source-build-artifacts | |
path: ${{ github.workspace }}/source_build_artifacts | |
merge-multiple: false | |
- name: Verify downloaded artifacts | |
if: needs.check-skip.outputs.should_skip != 'true' | |
id: verify-artifacts | |
run: | | |
set -eo pipefail | |
SRC_TARBALL_FILE=$(ls "${GITHUB_WORKSPACE}"/source_build_artifacts/apache-cloudberry-incubating-src.tgz) | |
if [ ! -f "${SRC_TARBALL_FILE}" ]; then | |
echo "::error::SRC TARBALL file not found" | |
exit 1 | |
fi | |
echo "src_tarball_file=${SRC_TARBALL_FILE}" >> "$GITHUB_OUTPUT" | |
echo "Verifying SRC TARBALL artifacts..." | |
{ | |
echo "=== SRC TARBALL Verification Summary ===" | |
echo "Timestamp: $(date -u)" | |
echo "SRC TARBALL File: ${SRC_TARBALL_FILE}" | |
# Calculate and store checksum | |
echo "Checksum:" | |
sha256sum "${SRC_TARBALL_FILE}" | |
} 2>&1 | tee -a build-logs/details/src-tarball-verification.log | |
RPM_FILE=$(ls "${GITHUB_WORKSPACE}"/rpm_build_artifacts/apache-cloudberry-db-incubating*.rpm) | |
if [ ! -f "${RPM_FILE}" ]; then | |
echo "::error::RPM file not found" | |
exit 1 | |
fi | |
echo "rpm_file=${RPM_FILE}" >> "$GITHUB_OUTPUT" | |
echo "Verifying RPM artifacts..." | |
{ | |
echo "=== RPM Verification Summary ===" | |
echo "Timestamp: $(date -u)" | |
echo "RPM File: ${RPM_FILE}" | |
# Get RPM metadata and verify contents | |
echo "Package Information:" | |
rpm -qip "${RPM_FILE}" | |
# Get key RPM attributes for verification | |
RPM_VERSION=$(rpm -qp --queryformat "%{VERSION}" "${RPM_FILE}") | |
RPM_RELEASE=$(rpm -qp --queryformat "%{RELEASE}" "${RPM_FILE}") | |
echo "version=${RPM_VERSION}" >> "$GITHUB_OUTPUT" | |
echo "release=${RPM_RELEASE}" >> "$GITHUB_OUTPUT" | |
# Verify expected binaries are in the RPM | |
echo "Verifying critical files in RPM..." | |
for binary in "bin/postgres" "bin/psql"; do | |
if ! rpm -qlp "${RPM_FILE}" | grep -q "${binary}$"; then | |
echo "::error::Critical binary '${binary}' not found in RPM" | |
exit 1 | |
fi | |
done | |
echo "RPM Details:" | |
echo "- Version: ${RPM_VERSION}" | |
echo "- Release: ${RPM_RELEASE}" | |
# Calculate and store checksum | |
echo "Checksum:" | |
sha256sum "${RPM_FILE}" | |
} 2>&1 | tee -a build-logs/details/rpm-verification.log | |
- name: Install Cloudberry RPM | |
if: success() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
RPM_FILE: ${{ steps.verify-artifacts.outputs.rpm_file }} | |
RPM_VERSION: ${{ steps.verify-artifacts.outputs.version }} | |
RPM_RELEASE: ${{ steps.verify-artifacts.outputs.release }} | |
run: | | |
set -eo pipefail | |
if [ -z "${RPM_FILE}" ]; then | |
echo "::error::RPM_FILE environment variable is not set" | |
exit 1 | |
fi | |
{ | |
echo "=== RPM Installation Log ===" | |
echo "Timestamp: $(date -u)" | |
echo "RPM File: ${RPM_FILE}" | |
echo "Version: ${RPM_VERSION}" | |
echo "Release: ${RPM_RELEASE}" | |
# Clean install location | |
rm -rf /usr/local/cloudberry-db | |
# Install RPM | |
echo "Starting installation..." | |
if ! time dnf install -y "${RPM_FILE}"; then | |
echo "::error::RPM installation failed" | |
exit 1 | |
fi | |
echo "Installation completed successfully" | |
rpm -qi apache-cloudberry-db-incubating | |
} 2>&1 | tee -a build-logs/details/rpm-installation.log | |
- name: Extract source tarball | |
if: success() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_TARBALL_FILE: ${{ steps.verify-artifacts.outputs.src_tarball_file }} | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
{ | |
echo "=== Source Extraction Log ===" | |
echo "Timestamp: $(date -u)" | |
echo "Starting extraction..." | |
if ! time tar zxf "${SRC_TARBALL_FILE}" -C "${SRC_DIR}"/.. ; then | |
echo "::error::Source extraction failed" | |
exit 1 | |
fi | |
echo "Extraction completed successfully" | |
echo "Extracted contents:" | |
ls -la "${SRC_DIR}/../cloudberry" | |
echo "Directory size:" | |
du -sh "${SRC_DIR}/../cloudberry" | |
} 2>&1 | tee -a build-logs/details/source-extraction.log | |
- name: Create Apache Cloudberry demo cluster | |
if: success() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
run: | | |
set -eo pipefail | |
{ | |
chmod +x "${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/create-cloudberry-demo-cluster.sh | |
if ! time su - gpadmin -c "cd ${SRC_DIR} && NUM_PRIMARY_MIRROR_PAIRS='${{ matrix.num_primary_mirror_pairs }}' SRC_DIR=${SRC_DIR} ${SRC_DIR}/../cloudberry-devops-release/build_automation/cloudberry/scripts/create-cloudberry-demo-cluster.sh"; then | |
echo "::error::Demo cluster creation failed" | |
exit 1 | |
fi | |
} 2>&1 | tee -a build-logs/details/create-cloudberry-demo-cluster.log | |
- name: "Run Tests: ${{ matrix.test }}" | |
if: success() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
shell: bash {0} | |
run: | | |
set -o pipefail | |
# Initialize test status | |
overall_status=0 | |
# Create logs directory structure | |
mkdir -p build-logs/details | |
# Core file config | |
mkdir -p "/tmp/cloudberry-cores" | |
chmod 1777 "/tmp/cloudberry-cores" | |
sysctl -w kernel.core_pattern="/tmp/cloudberry-cores/core-%e-%s-%u-%g-%p-%t" | |
sysctl kernel.core_pattern | |
su - gpadmin -c "ulimit -c" | |
# WARNING: PostgreSQL Settings | |
# When adding new pg_settings key/value pairs: | |
# 1. Add a new check below for the setting | |
# 2. Follow the same pattern as optimizer | |
# 3. Update matrix entries to include the new setting | |
# Set PostgreSQL options if defined | |
PG_OPTS="" | |
if [[ "${{ matrix.pg_settings.optimizer != '' }}" == "true" ]]; then | |
PG_OPTS="$PG_OPTS -c optimizer=${{ matrix.pg_settings.optimizer }}" | |
fi | |
# Read configs into array | |
IFS=' ' read -r -a configs <<< "${{ join(matrix.make_configs, ' ') }}" | |
echo "=== Starting test execution for ${{ matrix.test }} ===" | |
echo "Number of configurations to execute: ${#configs[@]}" | |
echo "" | |
# Execute each config separately | |
for ((i=0; i<${#configs[@]}; i++)); do | |
config="${configs[$i]}" | |
IFS=':' read -r dir target <<< "$config" | |
echo "=== Executing configuration $((i+1))/${#configs[@]} ===" | |
echo "Make command: make -C $dir $target" | |
echo "Environment:" | |
echo "- PGOPTIONS: ${PG_OPTS}" | |
# Create unique log file for this configuration | |
config_log="build-logs/details/make-${{ matrix.test }}-config$i.log" | |
# Clean up any existing core files | |
echo "Cleaning up existing core files..." | |
rm -f /tmp/cloudberry-cores/core-* | |
# Execute test script with proper environment setup | |
if ! time su - gpadmin -c "cd ${SRC_DIR} && \ | |
MAKE_NAME='${{ matrix.test }}-config$i' \ | |
MAKE_TARGET='$target' \ | |
MAKE_DIRECTORY='-C $dir' \ | |
PGOPTIONS='${PG_OPTS}' \ | |
SRC_DIR='${SRC_DIR}' \ | |
${SRC_DIR}/../cloudberry-devops-release/build_automation/cloudberry/scripts/test-cloudberry.sh" \ | |
2>&1 | tee "$config_log"; then | |
echo "::warning::Test execution failed for configuration $((i+1)): make -C $dir $target" | |
overall_status=1 | |
fi | |
# Check for results directory | |
results_dir="${dir}/results" | |
if [[ -d "$results_dir" ]]; then | |
echo "-----------------------------------------" | tee -a build-logs/details/make-${{ matrix.test }}-config$i-results.log | |
echo "Found results directory: $results_dir" | tee -a build-logs/details/make-${{ matrix.test }}-config$i-results.log | |
echo "Contents of results directory:" | tee -a build-logs/details/make-${{ matrix.test }}-config$i-results.log | |
find "$results_dir" -type f -ls >> "$log_file" 2>&1 | tee -a build-logs/details/make-${{ matrix.test }}-config$i-results.log | |
echo "-----------------------------------------" | tee -a build-logs/details/make-${{ matrix.test }}-config$i-results.log | |
else | |
echo "-----------------------------------------" | |
echo "Results directory $results_dir does not exit" | |
echo "-----------------------------------------" | |
fi | |
# Analyze any core files generated by this test configuration | |
echo "Analyzing core files for configuration ${{ matrix.test }}-config$i..." | |
test_id="${{ matrix.test }}-config$i" | |
# List the cores directory | |
echo "-----------------------------------------" | |
echo "Cores directory: /tmp/cloudberry-cores" | |
echo "Contents of cores directory:" | |
ls -Rl "/tmp/cloudberry-cores" | |
echo "-----------------------------------------" | |
"${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/analyze_core_dumps.sh "$test_id" | |
core_analysis_rc=$? | |
case "$core_analysis_rc" in | |
0) echo "No core dumps found for this configuration" ;; | |
1) echo "Core dumps were found and analyzed successfully" ;; | |
2) echo "::warning::Issues encountered during core dump analysis" ;; | |
*) echo "::error::Unexpected return code from core dump analysis: $core_analysis_rc" ;; | |
esac | |
echo "Log file: $config_log" | |
echo "=== End configuration $((i+1)) execution ===" | |
echo "" | |
done | |
echo "=== Test execution completed ===" | |
echo "Log files:" | |
ls -l build-logs/details/ | |
# Store number of configurations for parsing step | |
echo "NUM_CONFIGS=${#configs[@]}" >> "$GITHUB_ENV" | |
# Report overall status | |
if [ $overall_status -eq 0 ]; then | |
echo "All test executions completed successfully" | |
else | |
echo "::warning::Some test executions failed, check individual logs for details" | |
fi | |
exit $overall_status | |
- name: "Parse Test Results: ${{ matrix.test }}" | |
id: test-results | |
if: always() && needs.check-skip.outputs.should_skip != 'true' | |
env: | |
SRC_DIR: ${{ github.workspace }} | |
shell: bash {0} | |
run: | | |
set -o pipefail | |
overall_status=0 | |
# Get configs array to create context for results | |
IFS=' ' read -r -a configs <<< "${{ join(matrix.make_configs, ' ') }}" | |
echo "=== Starting results parsing for ${{ matrix.test }} ===" | |
echo "Number of configurations to parse: ${#configs[@]}" | |
echo "" | |
# Parse each configuration's results independently | |
for ((i=0; i<NUM_CONFIGS; i++)); do | |
config="${configs[$i]}" | |
IFS=':' read -r dir target <<< "$config" | |
config_log="build-logs/details/make-${{ matrix.test }}-config$i.log" | |
echo "=== Parsing results for configuration $((i+1))/${NUM_CONFIGS} ===" | |
echo "Make command: make -C $dir $target" | |
echo "Log file: $config_log" | |
if [ ! -f "$config_log" ]; then | |
echo "::error::Log file not found: $config_log" | |
{ | |
echo "MAKE_COMMAND=make -C $dir $target" | |
echo "STATUS=missing_log" | |
echo "TOTAL_TESTS=0" | |
echo "FAILED_TESTS=0" | |
echo "PASSED_TESTS=0" | |
echo "IGNORED_TESTS=0" | |
} > "test_results.$i.txt" | |
overall_status=1 | |
continue | |
fi | |
# Parse this configuration's results | |
MAKE_NAME="${{ matrix.test }}-config$i" \ | |
"${SRC_DIR}"/../cloudberry-devops-release/build_automation/cloudberry/scripts/parse-test-results.sh "$config_log" | |
status_code=$? | |
{ | |
echo "SUITE_NAME=${{ matrix.test }}" | |
echo "DIR=${dir}" | |
echo "TARGET=${target}" | |
} >> test_results.txt | |
# Process return code | |
case $status_code in | |
0) # All tests passed | |
echo "All tests passed successfully" | |
if [ -f test_results.txt ]; then | |
(echo "MAKE_COMMAND=\"make -C $dir $target\""; cat test_results.txt) | tee "test_results.${{ matrix.test }}.$i.txt" | |
rm test_results.txt | |
fi | |
;; | |
1) # Tests failed but parsed successfully | |
echo "Test failures detected but properly parsed" | |
if [ -f test_results.txt ]; then | |
(echo "MAKE_COMMAND=\"make -C $dir $target\""; cat test_results.txt) | tee "test_results.${{ matrix.test }}.$i.txt" | |
rm test_results.txt | |
fi | |
overall_status=1 | |
;; | |
2) # Parse error or missing file | |
echo "::warning::Could not parse test results properly for configuration $((i+1))" | |
{ | |
echo "MAKE_COMMAND=\"make -C $dir $target\"" | |
echo "STATUS=parse_error" | |
echo "TOTAL_TESTS=0" | |
echo "FAILED_TESTS=0" | |
echo "PASSED_TESTS=0" | |
echo "IGNORED_TESTS=0" | |
} | tee "test_results.${{ matrix.test }}.$i.txt" | |
overall_status=1 | |
;; | |
*) # Unexpected error | |
echo "::warning::Unexpected error during test results parsing for configuration $((i+1))" | |
{ | |
echo "MAKE_COMMAND=\"make -C $dir $target\"" | |
echo "STATUS=unknown_error" | |
echo "TOTAL_TESTS=0" | |
echo "FAILED_TESTS=0" | |
echo "PASSED_TESTS=0" | |
echo "IGNORED_TESTS=0" | |
} | tee "test_results.${{ matrix.test }}.$i.txt" | |
overall_status=1 | |
;; | |
esac | |
echo "Results stored in test_results.$i.txt" | |
echo "=== End parsing for configuration $((i+1)) ===" | |
echo "" | |
done | |
# Report status of results files | |
echo "=== Results file status ===" | |
echo "Generated results files:" | |
for ((i=0; i<NUM_CONFIGS; i++)); do | |
if [ -f "test_results.${{ matrix.test }}.$i.txt" ]; then | |
echo "- test_results.${{ matrix.test }}.$i.txt exists" | |
echo "" | |
else | |
echo "::error::Missing results file: test_results.${{ matrix.test }}.$i.txt" | |
overall_status=1 | |
fi | |
done | |
exit $overall_status | |
- name: Check and Display Regression Diffs | |
if: always() | |
run: | | |
# Search for regression.diffs recursively | |
found_file=$(find . -type f -name "regression.diffs" | head -n 1) | |
if [[ -n "$found_file" ]]; then | |
echo "Found regression.diffs at: $found_file" | |
cat "$found_file" | |
else | |
echo "No regression.diffs file found in the hierarchy." | |
fi | |
- name: "Check for Core Dumps Across All Configurations: ${{ matrix.test }}" | |
if: always() && needs.check-skip.outputs.should_skip != 'true' | |
shell: bash {0} | |
run: | | |
# Look for any core analysis files from this test matrix entry | |
core_files=$(find "${SRC_DIR}/build-logs" -name "core_analysis_*.log") | |
if [ -n "$core_files" ]; then | |
echo "::error::Core dumps were found during test execution:" | |
echo "$core_files" | while read -r file; do | |
echo "Core analysis file: $file" | |
echo "=== Content ===" | |
cat "$file" | |
echo "==============" | |
done | |
if [ "${{ matrix.enable_core_check }}" = "true" ]; then | |
exit 1 | |
else | |
echo "::warning::Special case - core checks will generate a warning" | |
fi | |
else | |
echo "No core dumps were found during test execution" | |
fi | |
- name: "Generate Test Job Summary End: ${{ matrix.test }}" | |
if: always() | |
shell: bash {0} | |
run: | | |
{ | |
if [[ "${{ needs.check-skip.outputs.should_skip }}" == "true" ]]; then | |
echo "## Test Results - SKIPPED" | |
echo "- End Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
exit 0 | |
fi | |
echo "## Test Results" | |
echo "- End Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
# Check if job was cancelled | |
if [[ "${{ job.status }}" == "cancelled" ]]; then | |
echo "### Test Status" | |
echo "🚫 Test execution was cancelled" | |
echo "" | |
echo "### Execution Summary" | |
echo "Test run was interrupted and did not complete. No test results are available." | |
exit 0 | |
fi | |
# Check for core analysis files | |
core_files=$(find "${SRC_DIR}/build-logs" -name "core_analysis_*.log") | |
if [ -n "$core_files" ]; then | |
if [ "${{ matrix.enable_core_check }}" = "true" ]; then | |
echo "❌ Core dumps were detected" | |
else | |
echo "⚠️ Core dumps were detected - enable_core_check: false" | |
fi | |
echo "" | |
echo "#### Core Analysis Files" | |
echo "\`\`\`" | |
echo "$core_files" | |
echo "\`\`\`" | |
echo "" | |
echo "#### Analysis Details" | |
echo "\`\`\`" | |
while read -r file; do | |
echo "=== $file ===" | |
cat "$file" | |
echo "" | |
done <<< "$core_files" | |
echo "\`\`\`" | |
else | |
echo "✅ No core dumps detected" | |
fi | |
# Process results for each configuration | |
IFS=' ' read -r -a configs <<< "${{ join(matrix.make_configs, ' ') }}" | |
for ((i=0; i<NUM_CONFIGS; i++)); do | |
config="${configs[$i]}" | |
IFS=':' read -r dir target <<< "$config" | |
echo "### Configuration $((i+1)): \`make -C $dir $target\`" | |
if [[ ! -f "test_results.${{ matrix.test }}.$i.txt" ]]; then | |
echo "⚠️ No results file found for this configuration" | |
continue | |
fi | |
# Source configuration results | |
# shellcheck source=/dev/null | |
. "test_results.${{ matrix.test }}.$i.txt" | |
# Rest of the code remains the same... | |
# Display status with emoji | |
echo "#### Status" | |
case "${STATUS:-unknown}" in | |
passed) | |
echo "✅ All tests passed" | |
;; | |
failed) | |
echo "❌ Some tests failed" | |
;; | |
parse_error) | |
echo "⚠️ Could not parse test results" | |
;; | |
unknown_error) | |
echo "⚠️ Unexpected error during test execution/parsing" | |
;; | |
missing_log) | |
echo "⚠️ Test log file missing" | |
;; | |
*) | |
echo "⚠️ Unknown status: ${status:-unknown}" | |
;; | |
esac | |
echo "" | |
echo "#### Test Counts" | |
echo "| Metric | Count |" | |
echo "|--------|-------|" | |
echo "| Total Tests | ${TOTAL_TESTS:-0} |" | |
echo "| Passed Tests | ${PASSED_TESTS:-0} |" | |
echo "| Failed Tests | ${FAILED_TESTS:-0} |" | |
echo "| Ignored Tests | ${IGNORED_TESTS:-0} |" | |
# Add failed tests if any | |
if [[ -n "${FAILED_TEST_NAMES:-}" && "${FAILED_TESTS:-0}" != "0" ]]; then | |
echo "" | |
echo "#### Failed Tests" | |
echo "${FAILED_TEST_NAMES}" | tr ',' '\n' | while read -r test; do | |
if [[ -n "$test" ]]; then | |
echo "* \`${test}\`" | |
fi | |
done | |
fi | |
# Add ignored tests if any | |
if [[ -n "${IGNORED_TEST_NAMES:-}" && "${IGNORED_TESTS:-0}" != "0" ]]; then | |
echo "" | |
echo "#### Ignored Tests" | |
echo "${IGNORED_TEST_NAMES}" | tr ',' '\n' | while read -r test; do | |
if [[ -n "$test" ]]; then | |
echo "* \`${test}\`" | |
fi | |
done | |
fi | |
echo "" | |
echo "---" | |
done | |
} >> "$GITHUB_STEP_SUMMARY" || true | |
- name: Upload test logs | |
if: always() | |
uses: actions/upload-artifact@v4 | |
with: | |
name: test-logs-${{ matrix.test }}-${{ needs.build.outputs.build_timestamp }} | |
path: | | |
build-logs/ | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
- name: Upload Test Metadata | |
if: always() | |
uses: actions/upload-artifact@v4 | |
with: | |
name: test-metadata-${{ matrix.test }} | |
path: | | |
test_results*.txt | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
- name: Upload test results files | |
uses: actions/upload-artifact@v4 | |
with: | |
name: results-${{ matrix.test }}-${{ needs.build.outputs.build_timestamp }} | |
path: | | |
**/regression.out | |
**/regression.diffs | |
**/results/ | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
- name: Upload test regression logs | |
if: failure() || cancelled() | |
uses: actions/upload-artifact@v4 | |
with: | |
name: regression-logs-${{ matrix.test }}-${{ needs.build.outputs.build_timestamp }} | |
path: | | |
**/regression.out | |
**/regression.diffs | |
**/results/ | |
gpAux/gpdemo/datadirs/standby/log/ | |
gpAux/gpdemo/datadirs/qddir/demoDataDir-1/log/ | |
gpAux/gpdemo/datadirs/dbfast1/demoDataDir0/log/ | |
gpAux/gpdemo/datadirs/dbfast2/demoDataDir1/log/ | |
gpAux/gpdemo/datadirs/dbfast3/demoDataDir2/log/ | |
gpAux/gpdemo/datadirs/dbfast_mirror1/demoDataDir0/log/ | |
gpAux/gpdemo/datadirs/dbfast_mirror2/demoDataDir1/log/ | |
gpAux/gpdemo/datadirs/dbfast_mirror3/demoDataDir2/log/ | |
retention-days: ${{ env.LOG_RETENTION_DAYS }} | |
## ====================================================================== | |
## Job: report | |
## ====================================================================== | |
report: | |
name: Generate Apache Cloudberry Build Report | |
needs: [check-skip, build, prepare-test-matrix, rpm-install-test, test] | |
if: always() | |
runs-on: ubuntu-22.04 | |
steps: | |
- name: Generate Final Report | |
run: | | |
{ | |
echo "# Apache Cloudberry Build Pipeline Report" | |
if [[ "${{ needs.check-skip.outputs.should_skip }}" == "true" ]]; then | |
echo "## CI Skip Status" | |
echo "✅ CI checks skipped via skip flag" | |
echo "- Completion Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
else | |
echo "## Job Status" | |
echo "- Build Job: ${{ needs.build.result }}" | |
echo "- Test Job: ${{ needs.test.result }}" | |
echo "- Completion Time: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
if [[ "${{ needs.build.result }}" == "success" && "${{ needs.test.result }}" == "success" ]]; then | |
echo "✅ Pipeline completed successfully" | |
else | |
echo "⚠️ Pipeline completed with failures" | |
if [[ "${{ needs.build.result }}" != "success" ]]; then | |
echo "### Build Job Failure" | |
echo "Check build logs for details" | |
fi | |
if [[ "${{ needs.test.result }}" != "success" ]]; then | |
echo "### Test Job Failure" | |
echo "Check test logs and regression files for details" | |
fi | |
fi | |
fi | |
} >> "$GITHUB_STEP_SUMMARY" | |
- name: Notify on failure | |
if: | | |
needs.check-skip.outputs.should_skip != 'true' && | |
(needs.build.result != 'success' || needs.test.result != 'success') | |
run: | | |
echo "::error::Build/Test pipeline failed! Check job summaries and logs for details" | |
echo "Timestamp: $(date -u +'%Y-%m-%d %H:%M:%S UTC')" | |
echo "Build Result: ${{ needs.build.result }}" | |
echo "Test Result: ${{ needs.test.result }}" |