| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274 |
- name: Publish Docker Images
- on:
- #push:
- # branches:
- # - main-next
- # - dev-next
- release:
- types:
- - published
- workflow_dispatch:
- inputs:
- tag:
- description: "The tag version you want to build"
- env:
- REGISTRY_IMAGE: ghcr.io/sagernet/sing-box
- jobs:
- build_binary:
- name: Build binary
- runs-on: ubuntu-latest
- strategy:
- fail-fast: true
- matrix:
- include:
- # Naive-enabled builds (musl)
- - { arch: amd64, naive: true, docker_platform: "linux/amd64" }
- - { arch: arm64, naive: true, docker_platform: "linux/arm64" }
- - { arch: "386", naive: true, docker_platform: "linux/386" }
- - { arch: arm, goarm: "7", naive: true, docker_platform: "linux/arm/v7" }
- # Non-naive builds
- - { arch: arm, goarm: "6", docker_platform: "linux/arm/v6" }
- - { arch: ppc64le, docker_platform: "linux/ppc64le" }
- - { arch: riscv64, docker_platform: "linux/riscv64" }
- - { arch: s390x, docker_platform: "linux/s390x" }
- steps:
- - name: Get commit to build
- id: ref
- run: |-
- if [[ -z "${{ github.event.inputs.tag }}" ]]; then
- ref="${{ github.ref_name }}"
- else
- ref="${{ github.event.inputs.tag }}"
- fi
- echo "ref=$ref"
- echo "ref=$ref" >> $GITHUB_OUTPUT
- - name: Checkout
- uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5
- with:
- ref: ${{ steps.ref.outputs.ref }}
- fetch-depth: 0
- - name: Setup Go
- uses: actions/setup-go@v5
- with:
- go-version: ^1.25.4
- - name: Clone cronet-go
- if: matrix.naive
- run: |
- set -xeuo pipefail
- CRONET_GO_VERSION=$(cat .github/CRONET_GO_VERSION)
- git init ~/cronet-go
- git -C ~/cronet-go remote add origin https://github.com/sagernet/cronet-go.git
- git -C ~/cronet-go fetch --depth=1 origin "$CRONET_GO_VERSION"
- git -C ~/cronet-go checkout FETCH_HEAD
- git -C ~/cronet-go submodule update --init --recursive --depth=1
- - name: Cache Chromium toolchain
- if: matrix.naive
- id: cache-chromium-toolchain
- uses: actions/cache@v4
- with:
- path: |
- ~/cronet-go/naiveproxy/src/third_party/llvm-build/Release+Asserts
- ~/cronet-go/naiveproxy/src/out/sysroot-build
- key: chromium-toolchain-${{ matrix.arch }}-musl-${{ hashFiles('.github/CRONET_GO_VERSION') }}
- - name: Download Chromium toolchain
- if: matrix.naive
- run: |
- set -xeuo pipefail
- cd ~/cronet-go
- go run ./cmd/build-naive --target=linux/${{ matrix.arch }} --libc=musl download-toolchain
- - name: Set version
- run: |
- set -xeuo pipefail
- VERSION=$(go run ./cmd/internal/read_tag)
- echo "VERSION=${VERSION}" >> "${GITHUB_ENV}"
- - name: Set Chromium toolchain environment
- if: matrix.naive
- run: |
- set -xeuo pipefail
- cd ~/cronet-go
- go run ./cmd/build-naive --target=linux/${{ matrix.arch }} --libc=musl env >> $GITHUB_ENV
- - name: Set build tags
- run: |
- set -xeuo pipefail
- TAGS='with_gvisor,with_quic,with_dhcp,with_wireguard,with_utls,with_acme,with_clash_api,with_tailscale,with_ccm,with_ocm,badlinkname,tfogo_checklinkname0'
- if [[ "${{ matrix.naive }}" == "true" ]]; then
- TAGS="${TAGS},with_naive_outbound,with_musl"
- fi
- echo "BUILD_TAGS=${TAGS}" >> "${GITHUB_ENV}"
- - name: Build (naive)
- if: matrix.naive
- run: |
- set -xeuo pipefail
- go build -v -trimpath -o sing-box -tags "${BUILD_TAGS}" \
- -ldflags "-X \"github.com/sagernet/sing-box/constant.Version=${VERSION}\" -s -w -buildid= -checklinkname=0" \
- ./cmd/sing-box
- env:
- CGO_ENABLED: "1"
- GOOS: linux
- GOARCH: ${{ matrix.arch }}
- GOARM: ${{ matrix.goarm }}
- - name: Build (non-naive)
- if: ${{ ! matrix.naive }}
- run: |
- set -xeuo pipefail
- go build -v -trimpath -o sing-box -tags "${BUILD_TAGS}" \
- -ldflags "-X \"github.com/sagernet/sing-box/constant.Version=${VERSION}\" -s -w -buildid= -checklinkname=0" \
- ./cmd/sing-box
- env:
- CGO_ENABLED: "0"
- GOOS: linux
- GOARCH: ${{ matrix.arch }}
- GOARM: ${{ matrix.goarm }}
- - name: Prepare artifact
- run: |
- platform=${{ matrix.docker_platform }}
- echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
- # Rename binary to include arch info for Dockerfile.binary
- BINARY_NAME="sing-box-${{ matrix.arch }}"
- if [[ -n "${{ matrix.goarm }}" ]]; then
- BINARY_NAME="${BINARY_NAME}v${{ matrix.goarm }}"
- fi
- mv sing-box "${BINARY_NAME}"
- echo "BINARY_NAME=${BINARY_NAME}" >> $GITHUB_ENV
- - name: Upload binary
- uses: actions/upload-artifact@v4
- with:
- name: binary-${{ env.PLATFORM_PAIR }}
- path: ${{ env.BINARY_NAME }}
- if-no-files-found: error
- retention-days: 1
- build_docker:
- name: Build Docker image
- runs-on: ubuntu-latest
- needs:
- - build_binary
- strategy:
- fail-fast: true
- matrix:
- platform:
- - linux/amd64
- - linux/arm/v6
- - linux/arm/v7
- - linux/arm64
- - linux/386
- - linux/ppc64le
- - linux/riscv64
- - linux/s390x
- steps:
- - name: Get commit to build
- id: ref
- run: |-
- if [[ -z "${{ github.event.inputs.tag }}" ]]; then
- ref="${{ github.ref_name }}"
- else
- ref="${{ github.event.inputs.tag }}"
- fi
- echo "ref=$ref"
- echo "ref=$ref" >> $GITHUB_OUTPUT
- - name: Checkout
- uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5
- with:
- ref: ${{ steps.ref.outputs.ref }}
- fetch-depth: 0
- - name: Prepare
- run: |
- platform=${{ matrix.platform }}
- echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
- - name: Download binary
- uses: actions/download-artifact@v5
- with:
- name: binary-${{ env.PLATFORM_PAIR }}
- path: .
- - name: Prepare binary
- run: |
- # Find and make the binary executable
- chmod +x sing-box-*
- ls -la sing-box-*
- - name: Setup QEMU
- uses: docker/setup-qemu-action@v3
- - name: Setup Docker Buildx
- uses: docker/setup-buildx-action@v3
- - name: Login to GitHub Container Registry
- uses: docker/login-action@v3
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Docker meta
- id: meta
- uses: docker/metadata-action@v5
- with:
- images: ${{ env.REGISTRY_IMAGE }}
- - name: Build and push by digest
- id: build
- uses: docker/build-push-action@v6
- with:
- platforms: ${{ matrix.platform }}
- context: .
- file: Dockerfile.binary
- labels: ${{ steps.meta.outputs.labels }}
- outputs: type=image,name=${{ env.REGISTRY_IMAGE }},push-by-digest=true,name-canonical=true,push=true
- - name: Export digest
- run: |
- mkdir -p /tmp/digests
- digest="${{ steps.build.outputs.digest }}"
- touch "/tmp/digests/${digest#sha256:}"
- - name: Upload digest
- uses: actions/upload-artifact@v4
- with:
- name: digests-${{ env.PLATFORM_PAIR }}
- path: /tmp/digests/*
- if-no-files-found: error
- retention-days: 1
- merge:
- runs-on: ubuntu-latest
- needs:
- - build_docker
- steps:
- - name: Get commit to build
- id: ref
- run: |-
- if [[ -z "${{ github.event.inputs.tag }}" ]]; then
- ref="${{ github.ref_name }}"
- else
- ref="${{ github.event.inputs.tag }}"
- fi
- echo "ref=$ref"
- echo "ref=$ref" >> $GITHUB_OUTPUT
- if [[ $ref == *"-"* ]]; then
- latest=latest-beta
- else
- latest=latest
- fi
- echo "latest=$latest"
- echo "latest=$latest" >> $GITHUB_OUTPUT
- - name: Download digests
- uses: actions/download-artifact@v5
- with:
- path: /tmp/digests
- pattern: digests-*
- merge-multiple: true
- - name: Set up Docker Buildx
- uses: docker/setup-buildx-action@v3
- - name: Login to GitHub Container Registry
- uses: docker/login-action@v3
- with:
- registry: ghcr.io
- username: ${{ github.repository_owner }}
- password: ${{ secrets.GITHUB_TOKEN }}
- - name: Create manifest list and push
- if: github.event_name != 'push'
- working-directory: /tmp/digests
- run: |
- docker buildx imagetools create \
- -t "${{ env.REGISTRY_IMAGE }}:${{ steps.ref.outputs.latest }}" \
- -t "${{ env.REGISTRY_IMAGE }}:${{ steps.ref.outputs.ref }}" \
- $(printf '${{ env.REGISTRY_IMAGE }}@sha256:%s ' *)
- - name: Inspect image
- if: github.event_name != 'push'
- run: |
- docker buildx imagetools inspect ${{ env.REGISTRY_IMAGE }}:${{ steps.ref.outputs.latest }}
- docker buildx imagetools inspect ${{ env.REGISTRY_IMAGE }}:${{ steps.ref.outputs.ref }}
|