name: build on: workflow_dispatch: inputs: use_cache: description: 'Restore cache for host tools' required: true default: 'true' type: choice options: - true - false cache_ver: description: 'Use cache version' required: false default: '0' type: string test_build: description: 'Test build (only spec target) ' required: true default: 'false' type: choice options: - true - false target_name: description: 'Target Name for test build' required: false default: 'r3d' type: string package_dir: description: 'Package Dir for test build' required: false default: '' type: string fake_build: description: 'Fake build' required: true default: 'false' type: choice options: - true - false trx_build: description: 'TRX images build' required: true default: 'false' type: choice options: - true - false suffix: description: 'TAG suffix' required: false default: '' type: string env: TEST_BUILD: ${{ github.event.inputs.test_build == 'true' }} TAG_SUFFIX: ${{ github.event.inputs.fake_build == 'true' && '-fake' || github.event.inputs.test_build == 'true' && '-test' || '' }} TARGET_NAME: ${{ github.event.inputs.target_name }} G_PKG_DIR: ${{ github.event.inputs.package_dir }} REPO_URL: https://github.com/openwrt-xiaomi/openwrt REPO_LNK: openwrt-xiaomi/openwrt REPO_BRANCH: xq-24.10 TAG_PREFIX: v24- TZ: UTC DEVICE_NAME: unknown BUILD_DATE: unknown REPO_DATE: unknown FW_DATE: unknown FW_VER: unknown jobs: check: runs-on: ubuntu-24.04 outputs: tag: ${{ steps.gh.outputs.tag }} date: ${{ steps.gh.outputs.date }} sha: ${{ steps.gh.outputs.sha }} url: ${{ steps.gh.outputs.url }} message: ${{ steps.gh.outputs.message }} build_date: ${{ steps.gh.outputs.build_date }} fw_date: ${{ steps.gh.outputs.fw_date }} fw_ver: ${{ steps.gh.outputs.fw_ver }} is_active: ${{ steps.activity.outputs.is_active }} test_build: ${{ env.TEST_BUILD }} target_name: ${{ env.TARGET_NAME }} fake_build: ${{ env.FAKE_BUILD }} steps: - name: Get repo data via GH API id: gh env: GH_TOKEN: ${{ secrets.GITHUB_TOKEN }} run: | echo "Tag name from GITHUB_REF_NAME: $GITHUB_REF_NAME" echo "Tag name from github.ref_name: ${{ github.ref_name }}" BRANCH=$REPO_BRANCH REPO_DATE=$(gh api repos/$REPO_LNK/commits/$BRANCH --jq '.commit.committer.date') BUILD_DATE=$( date --utc +'%y%m%d' ) FW_DATE=$( date --utc +'%Y-%m-%d' ) TAG=$TAG_PREFIX$BUILD_DATE echo "REPO_DATE=$REPO_DATE" >> $GITHUB_ENV echo "BUILD_DATE=$BUILD_DATE" >> $GITHUB_ENV echo "FW_DATE=$FW_DATE" >> $GITHUB_ENV echo "TAG=$TAG" >> $GITHUB_ENV wget $REPO_URL/raw/refs/heads/$BRANCH/include/version.mk FW_VERSION=$( grep -s '^VERSION_NUMBER:=$(if' version.mk 2>/dev/null ) [ -z "$FW_VERSION" ] && { echo "ERROR: Cannot find VERSION_NUMBER"; exit 90; } FW_VER=$( echo $FW_VERSION | cut -d"," -f3 | cut -d")" -f1 ) echo 'FW_VER = "'$FW_VER'"' echo "FW_VER=$FW_VER" >> $GITHUB_ENV rm -f version.mk { echo "tag=$TAG" echo "date=$(date --utc -d $REPO_DATE +%Y%m%d)" echo "sha=$(gh api repos/$REPO_LNK/commits/$BRANCH --jq '.sha[0:7]')" echo "url=$(gh api repos/$REPO_LNK/commits/$BRANCH --jq '.html_url')" echo "message<> $GITHUB_OUTPUT - name: Check for repo activity id: activity env: REPO_DATE: ${{ env.REPO_DATE }} URL: ${{ steps.gh.outputs.url }} run: | TIMESTAMP=$(date --utc -d $REPO_DATE +%s) DAYS=$(( ( $(date --utc +%s) - $TIMESTAMP ) / 86400 )) echo "Repository activity: $(date --utc -d $REPO_DATE)" echo "Commit: $URL" if [ "${{ github.event_name }}" != "schedule" ]; then is_active=true elif [[ $DAYS -lt 1 ]] ; then is_active=true else echo "Repository not updated within last 24 hours." is_active=false fi echo "is_active=$is_active" >> $GITHUB_OUTPUT build: needs: check #if: needs.check.outputs.is_active == 'true' runs-on: ubuntu-24.04 strategy: fail-fast: false matrix: target: - kng_re - r3d - r3g - r3p - rb03 - rb06 - rd03 - rt-ax52 - rt-ax57m - rt-ax59u - rt-ax89x - tuf-ax4200 - tuf-ax4200q - tuf-ax6000 - zen-bt8 isTest: - ${{ needs.check.outputs.test_build == 'true' }} exclude: - { isTest: true } include: - target: ${{ github.event.inputs.target_name }} steps: - name: Initialization environment env: DEBIAN_FRONTEND: noninteractive run: | sudo apt-get -qq update #sudo apt-get -qq update && sudo apt-get -qq upgrade sudo apt-get -qq install build-essential gawk flex rsync swig unzip zlib1g-dev file wget sudo apt-get -qq install libc6-dev clang bison g++ gcc-multilib g++-multilib sudo apt-get -qq install gettext git libncurses5-dev libssl-dev python3-setuptools sudo apt-get -qq install libc6-dev pkg-config squashfs-tools sudo apt-get -qq autoremove --purge sudo apt-get -qq clean sudo timedatectl set-timezone "$TZ" - name: Maximize build disk space uses: easimon/maximize-build-space@master with: remove-dotnet: true remove-android: true remove-haskell: true remove-codeql: true remove-docker-images: true - name: Set fixed workspace env: GITHUB_WORKSPACE: ${{ github.workspace }} run: | sudo mkdir -p $GITHUB_WORKSPACE/_temp #sudo chown -R $USER:$USER $GITHUB_WORKSPACE echo "GITHUB_WORKSPACE = $GITHUB_WORKSPACE" echo "GITHUB_WORKSPACE=$GITHUB_WORKSPACE" >> $GITHUB_ENV echo "RUNNER_TEMP=$GITHUB_WORKSPACE/_temp" >> $GITHUB_ENV - name: Checkout builder uses: actions/checkout@main - name: Prepare env params env: INP_FAKE_BUILD: ${{ github.event.inputs.fake_build == 'true' }} INP_TRX_BUILD: ${{ github.event.inputs.trx_build == 'true' }} TARGET: ${{ matrix.target }} run: | FAKE_BUILD=$INP_FAKE_BUILD TRX_BUILD=false if [ "$INP_TRX_BUILD" = true ]; then CFG="./_initramfs/$TARGET.config" if [ -f "$CFG" ]; then TRX_BUILD=true echo "Start TRX build for $CFG" echo "OPT_INITRAMFS=-i" >> $GITHUB_ENV else FAKE_BUILD=true echo "Start FAKE build for $TARGET" echo "SKIP_ME=true" >> $GITHUB_ENV fi fi echo "FAKE_BUILD = $FAKE_BUILD" echo "TRX_BUILD = $TRX_BUILD" echo "FAKE_BUILD=$FAKE_BUILD" >> $GITHUB_ENV echo "TRX_BUILD=$TRX_BUILD" >> $GITHUB_ENV - name: Download OpenWrt sources run: | git clone --branch $REPO_BRANCH $REPO_URL.git openwrt rm -rf .git rsync -a openwrt/ ./ rm -rf openwrt - name: Init builder run: | chmod +x *.sh wget https://github.com/fantastic-packages/packages/raw/refs/heads/24.10/keys/usign/53FF2B6672243D28.pub - name: Update OpenWrt packages id: update run: | ./xupdate.sh -f mkdir -p logs echo "status=success" >> $GITHUB_OUTPUT - name: Prepare for cache id: cache_prepare env: TARGET: ${{ matrix.target }} USE_CACHE: ${{ github.event.inputs.use_cache }} CACHE_VER: ${{ github.event.inputs.cache_ver }} run: | if [ "$USE_CACHE" = "true" ]; then rm -rf staging_dir/ mkdir -p staging_dir/host rm -rf build_dir/ mkdir -p build_dir/host fi if [[ " kng_re r3g r3p " == *" $TARGET "* ]]; then CACHE_ARCH=mips else CACHE_ARCH=arm fi echo "CACHE_ARCH = $CACHE_ARCH" echo "USE_CACHE=$USE_CACHE" >> $GITHUB_ENV echo "CACHE_VER=$CACHE_VER" >> $GITHUB_ENV echo "CACHE_ARCH=$CACHE_ARCH" >> $GITHUB_ENV echo "status=success" >> $GITHUB_OUTPUT - name: Restore cache for host tools id: tools_cache_restore if: github.event.inputs.use_cache == 'true' uses: actions/cache/restore@v4 with: key: host-tools-${{ env.CACHE_VER }}-${{ env.CACHE_ARCH }}-${{ hashFiles('tools/**') }} path: | staging_dir/host build_dir/host - name: Fix time stamps into restored dirs if: github.event.inputs.use_cache == 'true' && steps.tools_cache_restore.outputs.cache-hit == 'true' run: | touch staging_dir/host/.prepared touch staging_dir/host/.prereq-build touch staging_dir/host/stamp/.* 2>/dev/null find build_dir/host -name ".prepared*" -exec touch {} + find build_dir/host -name ".configured" -exec touch {} + find build_dir/host -name ".built" -exec touch {} + - name: Inspect restored host tools if: github.event.inputs.use_cache == 'true' run: | echo "====== openwrt-native.txt ======" cat build_dir/host/pkgconf-*/openwrt-native.txt || true echo "====== staging_dir/host =====" du -sh staging_dir/host || true ls -la staging_dir/host || true echo "====== staging_dir/host/bin =====" du -sh staging_dir/host/bin || true ls -1 staging_dir/host/bin | head -20 || true echo "====== build_dir/host =======" du -sh build_dir/host || true ls -1 build_dir/host | head -20 || true echo "====== path info ========" grep -R "/home/runner" build_dir/host | head - name: Configure id: configure if: steps.update.outputs.status == 'success' env: TARGET: ${{ matrix.target }} FW_VER: ${{ needs.check.outputs.fw_ver }} run: | ./xmake.sh $OPT_INITRAMFS -I -t $TARGET G_DEVICE_NAME=$( grep -so '^CONFIG_TARGET.*DEVICE.*=y' .config | sed -r 's/.*DEVICE_(.*)=y/\1/' ) [ -z "$G_DEVICE_NAME" ] && exit 31 G_BOARD_NAME=$( grep -so '^CONFIG_TARGET_BOARD=.*' .config | cut -d'"' -f2 ) [ -z "$G_BOARD_NAME" ] && exit 32 G_SUBTARGET_NAME=$( grep -so '^CONFIG_TARGET_SUBTARGET=.*' .config | cut -d'"' -f2 ) [ -z "$G_SUBTARGET_NAME" ] && exit 33 G_TARGET_ARCH=$( grep -so '^CONFIG_TARGET_ARCH_PACKAGES=.*' .config | cut -d'"' -f2 ) echo "G_TARGET_ARCH = $G_TARGET_ARCH" VERMAGIC_LIST="vermagic-$G_BOARD_NAME-$G_SUBTARGET_NAME-*.list" if [ $(find . -maxdepth 1 -name "$VERMAGIC_LIST" | wc -l) = "0" ]; then ./vermagic_update.sh $G_BOARD_NAME $G_SUBTARGET_NAME fi if [ $(find . -maxdepth 1 -name "$VERMAGIC_LIST" | wc -l) = "0" ]; then exit 41 fi FILE_DATE=$(date --utc +'%y%m%d') echo "status=success" >> $GITHUB_OUTPUT echo "FW_VER=$FW_VER" >> $GITHUB_ENV echo "G_DEVICE_NAME=$G_DEVICE_NAME" >> $GITHUB_ENV echo "G_BOARD_NAME=$G_BOARD_NAME" >> $GITHUB_ENV echo "G_SUBTARGET_NAME=$G_SUBTARGET_NAME" >> $GITHUB_ENV echo "G_TARGET_ARCH=$G_TARGET_ARCH" >> $GITHUB_ENV echo "FILE_DATE=$FILE_DATE" >> $GITHUB_ENV - name: Dump .config file run: | echo "====== .config =========" cat .config echo "========================" cp -f .config logs/config1 cp -f .config config1 - name: Monitor memory run: | echo "CPU cores = $(nproc)" echo "Memory and swap:" sudo free -h echo sudo swapon --show echo echo "Available storage:" sudo df -h - name: Download sources (dl) id: download if: ${{ steps.configure.outputs.status == 'success' && env.FAKE_BUILD != 'true' }} run: | make -j$(nproc) download echo "status=success" >> $GITHUB_OUTPUT - name: Fix time stamps into restored dirs if: github.event.inputs.use_cache == 'true' && steps.tools_cache_restore.outputs.cache-hit == 'true' run: | touch staging_dir/host/.prepared touch staging_dir/host/.prereq-build touch staging_dir/host/stamp/.* 2>/dev/null find build_dir/host -name ".prepared*" -exec touch {} + find build_dir/host -name ".configured" -exec touch {} + find build_dir/host -name ".built" -exec touch {} + - name: Build tools and toolchain id: tools if: ${{ steps.configure.outputs.status == 'success' && env.FAKE_BUILD != 'true' }} run: | make toolchain/install -j$(nproc) echo "status=success" >> $GITHUB_OUTPUT - name: Save cache for host tools id: tools_cache_save if: steps.tools.outputs.status == 'success' uses: actions/cache/save@v4 with: key: host-tools-${{ env.CACHE_VER }}-${{ env.CACHE_ARCH }}-${{ hashFiles('tools/**') }} path: | staging_dir/host build_dir/host - name: Configure 2 id: configure2 if: steps.tools.outputs.status == 'success' env: TARGET: ${{ matrix.target }} run: | make clean ./xmake.sh $OPT_INITRAMFS -I -t $TARGET mkdir -p logs cp -f .config logs/config2 [ -f config1 ] && cp -f config1 logs/config1 - name: Kernel compile id: kernel if: ${{ steps.tools.outputs.status == 'success' && env.FAKE_BUILD != 'true' }} run: | make target/linux/compile -j$(nproc) echo "status=success" >> $GITHUB_OUTPUT - name: Monitor memory 2 run: | echo "Memory and swap:" sudo free -h echo sudo swapon --show echo echo "Available storage:" sudo df -h - name: Build the firmware image id: compile if: steps.tools.outputs.status == 'success' env: TARGET: ${{ matrix.target }} run: | echo "Run $(nproc) thread compile" if [ "$FAKE_BUILD" != "true" -a "$TEST_BUILD" = "true" -a "$G_PKG_DIR" != "" ]; then make target/linux/compile -j$(npoc) echo "=============== BUILD PACKAGE: $G_PKG_DIR ===============" make $G_PKG_DIR/compile V=sc BUILD_LOG=1 exit 99 fi if [ "$FAKE_BUILD" != "true" ]; then if [ "$TEST_BUILD" != "true" ]; then make -j$(nproc) world else make V=sc world fi fi echo "status=success" >> $GITHUB_OUTPUT - name: Process out results id: install if: steps.compile.outputs.status == 'success' || env.FAKE_BUILD == 'true' env: TARGET: ${{ matrix.target }} OUT_DIR: ./bin/targets/${{ env.G_BOARD_NAME }}/${{ env.G_SUBTARGET_NAME }} run: | if [ "$FAKE_BUILD" = true ]; then mkdir -p logs mkdir -p "$OUT_DIR" cd "$OUT_DIR" echo "$G_BOARD_NAME $G_SUBTARGET_NAME $G_DEVICE_NAME" > "$G_DEVICE_NAME.txt" touch FAKE_BUILD.bin touch kernel-debug.tar.zst BLD_VER=$FW_VER else if [ ! -d "$OUT_DIR" ]; then echo "OUT_DIR not found!" exit 99 fi cd "$OUT_DIR" BLD_VER=$( cat profiles.json | grep -so '"version_number":"[^"]*' | grep -so '[^"]*$' ) fi if [ "$SKIP_ME" = true ]; then touch FLAG-SKIP_ME fi if [ "$TRX_BUILD" = true ]; then touch FLAG-TRX_BUILD fi echo "Firmware $BLD_VER [$FILE_DATE] builded!" if [ ! -f kernel-debug.tar.zst ]; then echo "File kernel-debug.tar.zst not found!" exit 100 fi if [ "$TRX_BUILD" = true ]; then find . -maxdepth 1 -type f ! -name '*-initramfs*' ! -name 'FLAG-*' -delete VER_MAJOR=$( echo "$BLD_VER" | cut -d. -f1 ) STD_IMG_DIR="openwrt-initramfs-bin-for-luci" mkdir $STD_IMG_DIR find . -maxdepth 1 -type f -name '*-initramfs*.bin' -exec mv -f {} $STD_IMG_DIR/openwrt-${VER_MAJOR}_${TARGET}_initramfs.bin \; find . -maxdepth 1 -type f -name '*-initramfs*.trx' -exec mv -f {} openwrt-${VER_MAJOR}_${TARGET}_initramfs.trx \; fi echo "status=success" >> $GITHUB_OUTPUT echo "FW_BUILD_OK=1" >> $GITHUB_ENV echo "BLD_VER=$BLD_VER" >> $GITHUB_ENV echo "OUT_DIR=$OUT_DIR" >> $GITHUB_ENV - name: Mark firmware build failed if: false # steps.install.outcome == 'failure' || steps.compile.outcome == 'failure' run: | echo "Firmware build failed, continuing pipeline" echo "FW_BUILD_OK=0" >> $GITHUB_ENV OUT_DIR=_bin_fake mkdir -p $OUT_DIR echo "$G_BOARD_NAME $G_SUBTARGET_NAME $G_DEVICE_NAME" > "$OUT_DIR/$G_DEVICE_NAME.txt" touch "$OUT_DIR/kernel-debug.tar.zst" echo "BUILD FAILED" > $OUT_DIR/FAILED.txt echo "BLD_VER=$FW_VER" >> $GITHUB_ENV echo "OUT_DIR=$OUT_DIR" >> $GITHUB_ENV - name: Check space usage run: df -hT - name: Check OOM error if: failure() run: | dmesg | grep -i -E 'oom|killed process' || true echo "================== Memory =================" free -h - name: Compress build logs if: always() env: TARGET: ${{ matrix.target }} FW_DATE: ${{ needs.check.outputs.fw_date }} run: | tar -cJvf logs-$TARGET-$FW_DATE.tar.xz logs - name: Cleanup OUT directory if: steps.install.outputs.status == 'success' env: OUT_DIR: ${{ env.OUT_DIR }} run: | cd $OUT_DIR echo "---------------------------" ls -la echo "---------------------------" rm -rf packages rm -f kernel-* rm -f *uImage rm -f *ubi-cleaner* rm -f *-ram-*.bin rm -f *-preloader* rm -f *-bl31-uboot* rm -f *-initramfs-recovery* rm -f *-bt8-factory.bin - name: Upload OUT directory uses: actions/upload-artifact@main if: steps.install.outputs.status == 'success' with: name: openwrt-${{ env.BLD_VER }}-${{ needs.check.outputs.build_date }}-${{ env.G_DEVICE_NAME }} path: ${{ env.OUT_DIR }} if-no-files-found: error - name: Upload build logs if: always() uses: actions/upload-artifact@v4 with: name: logs-${{ matrix.target }}-${{ needs.check.outputs.fw_date }} path: logs-*.tar.xz release: needs: [ check, build ] permissions: contents: write runs-on: ubuntu-24.04 strategy: max-parallel: 1 steps: - name: Download artifacts uses: actions/download-artifact@v4 with: pattern: openwrt-* - name: Check for failed builds run: | echo "Checking for FAILED.txt in openwrt-* directories..." FAILED_DIRS=$( find . -maxdepth 1 -type d -name 'openwrt-*' -exec test -f '{}/FAILED.txt' \; -print ) if [ -n "$FAILED_DIRS" ]; then echo "❌ Build failures detected in:" exit 201 fi echo "✅ No FAILED.txt found, proceeding to release" - name: Put images into zip run: | mkdir -p public for DIR in openwrt-*; do [ -d "$DIR" ] || continue if [ -f "$DIR/FLAG-SKIP_ME" ]; then echo "Skip $DIR" continue fi ZIP_SUFFIX="" if [ -f "$DIR/FLAG-TRX_BUILD" ]; then echo "TRX_BUILD=true" >> $GITHUB_ENV ZIP_SUFFIX=".trx" fi rm -f "$DIR"/FLAG-* zip_name="./public/$(basename "$DIR")${ZIP_SUFFIX}.zip" echo "Creating $zip_name" ( cd "$DIR"; zip -0 -r "../$zip_name" . ) done - name: Show all zip files run: | ls -lh ./public/*.zip - name: Generate names env: TAG: ${{ needs.check.outputs.tag }} FW_VER: ${{ needs.check.outputs.fw_ver }} FW_DATE: ${{ needs.check.outputs.fw_date }} run: | NAME="$TAG" DESC="OpenWrt $FW_VER [$FW_DATE]" if [ "$TRX_BUILD" = true ]; then NAME="TRX-$TAG" TAG="$TAG-trx" DESC="TRX images based on $DESC" fi echo "REL_TAG=$TAG" >> $GITHUB_ENV echo "REL_NAME=$NAME" >> $GITHUB_ENV echo "REL_DESC=$DESC" >> $GITHUB_ENV - name: Upload assets uses: softprops/action-gh-release@v2 with: draft: ${{ env.TEST_BUILD == 'true' || github.event.inputs.fake_build == 'true' }} prerelease: true tag_name: ${{ env.REL_TAG }}${{ env.TAG_SUFFIX }} name: '${{ env.REL_NAME }}' body: | ${{ env.REL_DESC }} author: [remittor](https://github.com/remittor) files: ./public/*.zip