mirror of
https://github.com/1technophile/OpenMQTTGateway.git
synced 2026-03-11 18:00:48 +01:00
* Refactor GitHub Actions workflows for build, documentation, and linting - Consolidated build logic into reusable workflows (`task-build.yml` and `task-docs.yml`) to reduce duplication across multiple workflows. - Introduced `environments.json` to centralize the list of PlatformIO build environments, improving maintainability and clarity. - Updated `build.yml` and `build_and_docs_to_dev.yml` to utilize the new reusable workflows and environment definitions. - Enhanced `release.yml` to streamline the release process and integrate documentation generation. - Created reusable linting workflow (`task-lint.yml`) to standardize code formatting checks across the repository. - Simplified manual documentation workflow by leveraging the new reusable documentation workflow. - Improved artifact management and retention policies across workflows. - Updated dependencies and versions in workflows to ensure compatibility and performance. CI/CD pipeline agnostic of Workflow Engine and integrated on github actions - Implemented ci.sh for orchestrating the complete build pipeline. - Created ci_00_config.sh for centralized configuration of build scripts. - Created ci_build_firmware.sh for building firmware for specified PlatformIO environments. - Created ci_prepare_artifacts.sh for preparing firmware artifacts for upload or deployment. - Created ci_set_version.sh for updating version tags in firmware configuration files. - Created ci_build.sh to orchestrate the complete build pipeline. - Created ci_qa.sh for code linting and formatting checks using clang-format. - Created ci_site.sh for building and deploying VuePress documentation with version management. - Implemented checks for required tools and dependencies in the new scripts. - Improved internal scripts for better error handling and logging. UPDATE the web installer manifest generation and update documentation structure - Enhanced ci_list-env.sh to list environments from a JSON file. - Replaced common_wu.py and gen_wu.py scripts with new npm scripts for site generation and previewing on docsgen/gen_wu.js - Replaced generate_board_docs.py with docsgen/generated_board_docs.js - Added new npm scripts for integration of site generation on build phase. - Created preview_site.js to serve locally generated site over HTTPS with improved error handling. - Added new CI environments for CI builds in environments.json. - Deleted lint.yml as part of workflow cleanup. - Enhanced task-build.yml to include linting as a job and added support for specifying PlatformIO version. - Improved task-docs.yml to handle versioning more effectively and added clean option. Enhance documentation - ADD CLEAR Mark of development version of site - Updated README.md to include detailed workflow dependencies and relationships using mermaid diagrams. - Improved development.md with a quick checklist for contributors and clarified the code style guide. - Enhanced quick_start.md with tips for contributors and streamlined the workflow explanation. LINT FIX - Refined User_config.h for better formatting consistency. - Adjusted blufi.cpp and gatewayBT.cpp for improved code readability and consistency in formatting. - Updated gatewaySERIAL.cpp and mqttDiscovery.cpp to enhance logging error messages. - Improved sensorDS1820.cpp for better logging of device information. Add security scan workflows for vulnerability detection Add SBOM generation and upload to release workflow; update security scan summary handling Add shellcheck suppor + FIX shellcheck warning Enhance documentation for CI/CD scripts and workflows, adding details for security scanning and SBOM generation processes Fix formatting and alignment in BLE connection handling Reviewed the full web board presentation and the ESP32 web upload. The project uses a modern pattern where data is divided from the presentation layer. - Removed the `generate_board_docs` script. - Updated the `gen_wu` script in order to generate `boards-info.json`: the fail that containe all information about the configuration - Created and isolate the file `boards-info.js` to streamline the parsing of PlatformIO dependencies, modules, environments and improve the handling of library information. - Introduced vuepress component `BoardEnvironmentTable.vue` that render `boards-info.json` as UI card component - Introduced vuepress component `FlashEnvironmentSelector.vue` that render a selectred environment from `boards-info.json` and provide esp-web-upload feature on it - Introduced a new board page `board-selector.md` for improved firmware selection. - Updated `web-install.md` to enhance the firmware upload process, including a new board environment table. - Enhanced custom descriptions in `environments.ini` to include HTML links for better user guidance and board image link Add CC1101 initialization improvements and logging enhancements Add installation step for PlatformIO dependencies in documentation workflow Remove ci_set_version.sh script and associated versioning functionality * Fix comment provisined Fix PlatformIO version input reference in documentation workflow Remove outdated Squeezelite-ESP32 installer documentation
475 lines
13 KiB
Bash
Executable File
475 lines
13 KiB
Bash
Executable File
#!/bin/bash
|
|
# CI/CD Quality Assurance (QA) - Code Linting and Formatting
|
|
# Checks and fixes code formatting using clang-format
|
|
# Usage: ./scripts/ci_qa.sh [OPTIONS]
|
|
|
|
set -euo pipefail
|
|
|
|
# Constants
|
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
PROJECT_ROOT="$(dirname "$SCRIPT_DIR")"
|
|
readonly SCRIPT_DIR
|
|
readonly PROJECT_ROOT
|
|
|
|
# Load shared configuration
|
|
if [[ -f "${SCRIPT_DIR}/ci_00_config.sh" ]]; then
|
|
source "${SCRIPT_DIR}/ci_00_config.sh"
|
|
else
|
|
echo "ERROR: ci_00_config.sh not found" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Default values
|
|
FIX_MODE=false
|
|
FORMAT_ONLY=false
|
|
SOURCE_DIR="main"
|
|
EXTENSIONS="h,ino,cpp"
|
|
CLANG_FORMAT_VERSION="9"
|
|
VERBOSE=false
|
|
|
|
# Function to check if clang-format is available
|
|
check_clang_format() {
|
|
local version="$1"
|
|
local cmd="clang-format-${version}"
|
|
|
|
if command -v "$cmd" >/dev/null 2>&1; then
|
|
echo "$cmd"
|
|
return 0
|
|
fi
|
|
|
|
# Try without version suffix
|
|
if command -v clang-format >/dev/null 2>&1; then
|
|
echo "clang-format"
|
|
return 0
|
|
fi
|
|
|
|
return 1
|
|
}
|
|
|
|
# Function to find files to check
|
|
find_files() {
|
|
local source="$1"
|
|
local extensions="$2"
|
|
|
|
if [[ ! -d "${PROJECT_ROOT}/${source}" ]]; then
|
|
return 1
|
|
fi
|
|
|
|
local find_patterns=()
|
|
IFS=',' read -ra exts <<< "$extensions"
|
|
for ext in "${exts[@]}"; do
|
|
find_patterns+=(-name "*.${ext}" -o)
|
|
done
|
|
# Remove last -o
|
|
unset 'find_patterns[-1]'
|
|
|
|
local files
|
|
files=$(find "${PROJECT_ROOT}/${source}" -type f \( "${find_patterns[@]}" \) 2>/dev/null || true)
|
|
|
|
if [[ -z "$files" ]]; then
|
|
return 1
|
|
fi
|
|
|
|
echo "$files"
|
|
}
|
|
|
|
# Function to check formatting
|
|
check_formatting() {
|
|
local clang_format_cmd="$1"
|
|
local files="$2"
|
|
|
|
log_info "Checking code formatting..."
|
|
|
|
local failed_files=()
|
|
local checked_count=0
|
|
local has_issues=false
|
|
|
|
while IFS= read -r file; do
|
|
if [[ -z "$file" ]]; then
|
|
continue
|
|
fi
|
|
|
|
checked_count=$((checked_count + 1))
|
|
|
|
if [[ "$VERBOSE" == true ]]; then
|
|
log_info "Checking: $file"
|
|
fi
|
|
|
|
# Check if file needs formatting and capture diff
|
|
local diff_output
|
|
diff_output=$("$clang_format_cmd" --dry-run --Werror "$file" 2>&1)
|
|
local format_result=$?
|
|
|
|
if [[ $format_result -ne 0 ]]; then
|
|
failed_files+=("$file")
|
|
has_issues=true
|
|
|
|
# Show the actual formatting differences
|
|
echo ""
|
|
log_warn "⚠ Formatting issues in: $file"
|
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
|
|
|
# Generate and show diff with colors
|
|
local actual_diff
|
|
actual_diff=$(diff -u "$file" <("$clang_format_cmd" "$file") 2>/dev/null || true)
|
|
|
|
if [[ -n "$actual_diff" ]]; then
|
|
echo "$actual_diff" | head -50 | while IFS= read -r line; do
|
|
if [[ "$line" =~ ^-[^-] ]]; then
|
|
echo -e "\033[31m$line\033[0m" # Red for removed lines
|
|
elif [[ "$line" =~ ^+[^+] ]]; then
|
|
echo -e "\033[32m$line\033[0m" # Green for added lines
|
|
elif [[ "$line" =~ ^@@ ]]; then
|
|
echo -e "\033[36m$line\033[0m" # Cyan for line numbers
|
|
else
|
|
echo "$line"
|
|
fi
|
|
done
|
|
else
|
|
echo "$diff_output"
|
|
fi
|
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
|
else
|
|
if [[ "$VERBOSE" == true ]]; then
|
|
log_info " ✓ OK"
|
|
fi
|
|
fi
|
|
done <<< "$files"
|
|
|
|
echo ""
|
|
log_info "Checked ${checked_count} files"
|
|
|
|
if [[ $has_issues == true ]]; then
|
|
echo ""
|
|
log_error "Found ${#failed_files[@]} files with formatting issues:"
|
|
for file in "${failed_files[@]}"; do
|
|
log_error " - $file"
|
|
done
|
|
echo ""
|
|
log_error "To fix these issues automatically, run:"
|
|
log_error " $0 --fix"
|
|
echo ""
|
|
return 1
|
|
fi
|
|
|
|
log_info "✓ All files are properly formatted"
|
|
return 0
|
|
}
|
|
|
|
# Function to fix formatting
|
|
fix_formatting() {
|
|
local clang_format_cmd="$1"
|
|
local files="$2"
|
|
|
|
log_info "Fixing code formatting..."
|
|
|
|
local fixed_count=0
|
|
local total_count=0
|
|
|
|
while IFS= read -r file; do
|
|
if [[ -z "$file" ]]; then
|
|
continue
|
|
fi
|
|
|
|
total_count=$((total_count + 1))
|
|
|
|
if [[ "$VERBOSE" == true ]]; then
|
|
log_info "Processing: $file"
|
|
fi
|
|
|
|
# Apply formatting in-place
|
|
if "$clang_format_cmd" -i "$file" 2>/dev/null; then
|
|
fixed_count=$((fixed_count + 1))
|
|
if [[ "$VERBOSE" == true ]]; then
|
|
log_info " ✓ Formatted"
|
|
fi
|
|
else
|
|
if [[ "$VERBOSE" == true ]]; then
|
|
log_warn " ✗ Failed to format"
|
|
fi
|
|
fi
|
|
done <<< "$files"
|
|
|
|
echo ""
|
|
log_info "Processed ${total_count} files"
|
|
log_info "✓ Formatting applied to ${fixed_count} files"
|
|
|
|
if [[ $fixed_count -gt 0 ]]; then
|
|
log_warn ""
|
|
log_warn "Files have been modified. Please review and commit the changes:"
|
|
log_warn " git diff"
|
|
log_warn " git add -u"
|
|
log_warn " git commit -m 'style: apply clang-format'"
|
|
fi
|
|
}
|
|
|
|
# Function to run shellcheck on scripts
|
|
run_shellcheck() {
|
|
log_info "Checking shell scripts with ShellCheck..."
|
|
|
|
# Check if shellcheck is installed
|
|
if ! command -v shellcheck >/dev/null 2>&1; then
|
|
log_warn "ShellCheck not found, skipping shell script checks"
|
|
log_info "To install ShellCheck:"
|
|
log_info " Ubuntu/Debian: sudo apt-get install shellcheck"
|
|
log_info " macOS: brew install shellcheck"
|
|
return 0 # Don't fail, just skip
|
|
fi
|
|
|
|
log_info "✓ ShellCheck found: $(shellcheck --version | head -n2 | tail -n1)"
|
|
|
|
# Find all .sh files in scripts directory
|
|
local scripts_dir="${PROJECT_ROOT}/scripts"
|
|
if [[ ! -d "$scripts_dir" ]]; then
|
|
log_warn "Scripts directory not found: $scripts_dir"
|
|
return 0
|
|
fi
|
|
|
|
local shell_files
|
|
shell_files=$(find "$scripts_dir" -type f -name "*.sh" 2>/dev/null)
|
|
|
|
if [[ -z "$shell_files" ]]; then
|
|
log_warn "No shell scripts found in $scripts_dir"
|
|
return 0
|
|
fi
|
|
|
|
local file_count
|
|
file_count=$(echo "$shell_files" | wc -l)
|
|
log_info "Found $file_count shell script(s) to check"
|
|
|
|
# Run shellcheck on all files at once to allow cross-file analysis
|
|
log_info "Running ShellCheck on scripts directory..."
|
|
|
|
local shellcheck_output
|
|
# shellcheck disable=SC2086
|
|
shellcheck_output=$(shellcheck -f gcc $shell_files 2>&1)
|
|
local shellcheck_result=$?
|
|
|
|
if [[ $shellcheck_result -ne 0 ]]; then
|
|
echo ""
|
|
log_warn "⚠ ShellCheck found issues:"
|
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
|
echo "$shellcheck_output"
|
|
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
|
echo ""
|
|
log_error "Please fix the ShellCheck warnings/errors"
|
|
return 1
|
|
fi
|
|
|
|
log_success "✓ All $file_count shell script(s) passed ShellCheck"
|
|
return 0
|
|
}
|
|
|
|
# Function to run all QA checks
|
|
run_all_checks() {
|
|
log_info "Running all quality assurance checks..."
|
|
|
|
local all_passed=true
|
|
|
|
# Format check
|
|
log_info "═══ Code Formatting ═══"
|
|
if ! run_format_check; then
|
|
all_passed=false
|
|
fi
|
|
echo ""
|
|
|
|
# ShellCheck
|
|
log_info "═══ Shell Scripts Check ═══"
|
|
if ! run_shellcheck; then
|
|
all_passed=false
|
|
fi
|
|
echo ""
|
|
|
|
# Future: Add more checks here
|
|
# - cppcheck static analysis
|
|
# - code complexity metrics
|
|
# - TODO/FIXME detection
|
|
# - license header validation
|
|
|
|
if [[ "$all_passed" == false ]]; then
|
|
log_error "Some QA checks failed"
|
|
return 1
|
|
fi
|
|
|
|
log_info "✓ All QA checks passed"
|
|
return 0
|
|
}
|
|
|
|
# Function to run format check
|
|
run_format_check() {
|
|
log_info "Checking for clang-format version ${CLANG_FORMAT_VERSION}..."
|
|
|
|
local clang_format_cmd
|
|
clang_format_cmd=$(check_clang_format "$CLANG_FORMAT_VERSION")
|
|
|
|
# shellcheck disable=SC2181
|
|
if [[ $? -ne 0 ]] || [[ -z "$clang_format_cmd" ]]; then
|
|
log_error "clang-format not found"
|
|
log_error "Please install clang-format:"
|
|
log_error " Ubuntu/Debian: sudo apt-get install clang-format-${CLANG_FORMAT_VERSION}"
|
|
log_error " macOS: brew install clang-format"
|
|
return 1
|
|
fi
|
|
|
|
if [[ "$clang_format_cmd" == "clang-format-${CLANG_FORMAT_VERSION}" ]]; then
|
|
log_info "✓ clang-format-${CLANG_FORMAT_VERSION} found"
|
|
else
|
|
local installed_version
|
|
installed_version=$(clang-format --version | grep -oP '\d+\.\d+' | head -1 || echo "unknown")
|
|
log_warn "clang-format-${CLANG_FORMAT_VERSION} not found, using clang-format (version ${installed_version})"
|
|
fi
|
|
|
|
log_info "Finding files in '${SOURCE_DIR}' with extensions: ${EXTENSIONS}"
|
|
|
|
local files
|
|
files=$(find_files "$SOURCE_DIR" "$EXTENSIONS")
|
|
|
|
# shellcheck disable=SC2181
|
|
if [[ $? -ne 0 ]] || [[ -z "$files" ]]; then
|
|
log_error "Source directory not found: ${PROJECT_ROOT}/${SOURCE_DIR}"
|
|
return 1
|
|
fi
|
|
|
|
local file_count
|
|
file_count=$(echo "$files" | wc -l)
|
|
log_info "Found ${file_count} files to check"
|
|
|
|
if [[ "$FIX_MODE" == true ]]; then
|
|
fix_formatting "$clang_format_cmd" "$files"
|
|
else
|
|
check_formatting "$clang_format_cmd" "$files"
|
|
fi
|
|
}
|
|
|
|
# Function to show usage
|
|
usage() {
|
|
cat << EOF
|
|
Usage: $0 [OPTIONS]
|
|
|
|
Run quality assurance checks on OpenMQTTGateway code.
|
|
|
|
Options:
|
|
--check Check formatting only (CI mode) [default]
|
|
--fix Fix formatting issues automatically
|
|
--format Run only format checks
|
|
--all Run all QA checks [default]
|
|
--source DIR Source directory to check [default: main]
|
|
--extensions EXTS File extensions (comma-separated) [default: h,ino,cpp]
|
|
--clang-format-version V clang-format version [default: 9]
|
|
--verbose Enable verbose output
|
|
--help Show this help message
|
|
|
|
Examples:
|
|
# Check formatting (CI mode)
|
|
$0 --check
|
|
|
|
# Fix formatting issues
|
|
$0 --fix
|
|
|
|
# Check specific directory
|
|
$0 --check --source lib/LEDManager
|
|
|
|
# Check with custom extensions
|
|
$0 --check --extensions h,cpp
|
|
|
|
# Verbose output
|
|
$0 --check --verbose
|
|
|
|
EOF
|
|
exit 0
|
|
}
|
|
|
|
# Parse command line arguments
|
|
parse_args() {
|
|
while [[ $# -gt 0 ]]; do
|
|
case "$1" in
|
|
--check)
|
|
FIX_MODE=false
|
|
shift
|
|
;;
|
|
--fix)
|
|
FIX_MODE=true
|
|
shift
|
|
;;
|
|
--format)
|
|
FORMAT_ONLY=true
|
|
shift
|
|
;;
|
|
--all)
|
|
FORMAT_ONLY=false
|
|
shift
|
|
;;
|
|
--source)
|
|
SOURCE_DIR="$2"
|
|
shift 2
|
|
;;
|
|
--extensions)
|
|
EXTENSIONS="$2"
|
|
shift 2
|
|
;;
|
|
--clang-format-version)
|
|
CLANG_FORMAT_VERSION="$2"
|
|
shift 2
|
|
;;
|
|
--verbose)
|
|
VERBOSE=true
|
|
shift
|
|
;;
|
|
--help)
|
|
usage
|
|
;;
|
|
*)
|
|
log_error "Unknown option: $1"
|
|
usage
|
|
;;
|
|
esac
|
|
done
|
|
}
|
|
|
|
# Main execution
|
|
main() {
|
|
local start_time
|
|
start_time=$(date +%s)
|
|
|
|
parse_args "$@"
|
|
|
|
log_info "Starting QA pipeline..."
|
|
|
|
if [[ "$FIX_MODE" == true ]]; then
|
|
log_info "Mode: FIX (will modify files)"
|
|
else
|
|
log_info "Mode: CHECK (read-only)"
|
|
fi
|
|
|
|
# Run checks
|
|
local result=0
|
|
if [[ "$FORMAT_ONLY" == true ]]; then
|
|
run_format_check || result=$?
|
|
else
|
|
run_all_checks || result=$?
|
|
fi
|
|
|
|
local end_time
|
|
end_time=$(date +%s)
|
|
local duration=$((end_time - start_time))
|
|
|
|
echo ""
|
|
echo "╔════════════════════════════════════════╗"
|
|
echo "║ QA Pipeline Summary ║"
|
|
echo "╚════════════════════════════════════════╝"
|
|
echo " Duration: ${duration}s"
|
|
|
|
if [[ $result -eq 0 ]]; then
|
|
echo " Status: SUCCESS ✓"
|
|
echo "╚════════════════════════════════════════╝"
|
|
return 0
|
|
else
|
|
echo " Status: FAILED ✗"
|
|
echo "╚════════════════════════════════════════╝"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# Execute main function
|
|
main "$@"
|