From 48fce7fe4594188216163b1687300bd252d16ceb Mon Sep 17 00:00:00 2001 From: Michal Date: Sat, 21 Feb 2026 14:00:24 +0000 Subject: [PATCH] feat: add RPM packaging with nfpm and Gitea registry publishing Bundles the CLI into a standalone binary via bun compile, packages as RPM with nfpm, and publishes to Gitea's built-in package registry. Users install with: dnf config-manager --add-repo /rpm.repo Co-Authored-By: Claude Opus 4.6 --- .gitea/workflows/ci.yml | 48 +++++++++++++++++++++++++++++++++++++++++ nfpm.yaml | 12 +++++++++++ package.json | 4 +++- scripts/build-rpm.sh | 23 ++++++++++++++++++++ scripts/publish-rpm.sh | 34 +++++++++++++++++++++++++++++ 5 files changed, 120 insertions(+), 1 deletion(-) create mode 100644 nfpm.yaml create mode 100755 scripts/build-rpm.sh create mode 100755 scripts/publish-rpm.sh diff --git a/.gitea/workflows/ci.yml b/.gitea/workflows/ci.yml index 5bfb7ad..9b1d6de 100644 --- a/.gitea/workflows/ci.yml +++ b/.gitea/workflows/ci.yml @@ -92,3 +92,51 @@ jobs: - name: Build all packages run: pnpm build + + package: + runs-on: ubuntu-latest + needs: [build] + if: github.ref == 'refs/heads/main' && github.event_name == 'push' + steps: + - uses: actions/checkout@v4 + + - uses: pnpm/action-setup@v4 + with: + version: 9 + + - uses: actions/setup-node@v4 + with: + node-version: 20 + cache: pnpm + + - run: pnpm install --frozen-lockfile + + - name: Generate Prisma client + run: pnpm --filter @mcpctl/db exec prisma generate + + - name: Build TypeScript + run: pnpm build + + - name: Install bun + uses: oven-sh/setup-bun@v2 + + - name: Install nfpm + run: | + curl -sL -o /tmp/nfpm.tar.gz "https://github.com/goreleaser/nfpm/releases/download/v2.45.0/nfpm_2.45.0_Linux_x86_64.tar.gz" + tar xzf /tmp/nfpm.tar.gz -C /usr/local/bin nfpm + + - name: Bundle standalone binary + run: bun build src/cli/src/index.ts --compile --outfile dist/mcpctl + + - name: Build RPM + run: nfpm pkg --packager rpm --target dist/ + + - name: Publish to Gitea packages + env: + GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} + run: | + RPM_FILE=$(ls dist/mcpctl-*.rpm | head -1) + curl --fail -X PUT \ + -H "Authorization: token ${GITEA_TOKEN}" \ + --upload-file "$RPM_FILE" \ + "${{ github.server_url }}/api/packages/${{ github.repository_owner }}/rpm/upload" diff --git a/nfpm.yaml b/nfpm.yaml new file mode 100644 index 0000000..4fec7f4 --- /dev/null +++ b/nfpm.yaml @@ -0,0 +1,12 @@ +name: mcpctl +arch: amd64 +version: 0.1.0 +release: "1" +maintainer: michal +description: kubectl-like CLI for managing MCP servers +license: MIT +contents: + - src: ./dist/mcpctl + dst: /usr/bin/mcpctl + file_info: + mode: 0755 diff --git a/package.json b/package.json index 7f17b6d..da018bc 100644 --- a/package.json +++ b/package.json @@ -15,7 +15,9 @@ "clean": "pnpm -r run clean && rimraf node_modules", "db:up": "docker compose -f deploy/docker-compose.yml up -d", "db:down": "docker compose -f deploy/docker-compose.yml down", - "typecheck": "tsc --build" + "typecheck": "tsc --build", + "rpm:build": "bash scripts/build-rpm.sh", + "rpm:publish": "bash scripts/publish-rpm.sh" }, "engines": { "node": ">=20.0.0", diff --git a/scripts/build-rpm.sh b/scripts/build-rpm.sh new file mode 100755 index 0000000..a81a1c5 --- /dev/null +++ b/scripts/build-rpm.sh @@ -0,0 +1,23 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)" +PROJECT_ROOT="$(dirname "$SCRIPT_DIR")" +cd "$PROJECT_ROOT" + +export VERSION="${VERSION:-0.1.0}" + +echo "==> Building TypeScript..." +pnpm build + +echo "==> Bundling standalone binary..." +mkdir -p dist +bun build src/cli/src/index.ts --compile --outfile dist/mcpctl + +echo "==> Packaging RPM..." +nfpm pkg --packager rpm --target dist/ + +RPM_FILE=$(ls dist/mcpctl-*.rpm 2>/dev/null | head -1) +echo "==> Built: $RPM_FILE" +echo " Size: $(du -h "$RPM_FILE" | cut -f1)" +rpm -qpi "$RPM_FILE" diff --git a/scripts/publish-rpm.sh b/scripts/publish-rpm.sh new file mode 100755 index 0000000..2966518 --- /dev/null +++ b/scripts/publish-rpm.sh @@ -0,0 +1,34 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)" +PROJECT_ROOT="$(dirname "$SCRIPT_DIR")" +cd "$PROJECT_ROOT" + +GITEA_URL="${GITEA_URL:-http://10.0.0.194:3012}" +GITEA_OWNER="${GITEA_OWNER:-michal}" + +if [ -z "$GITEA_TOKEN" ]; then + echo "Error: GITEA_TOKEN environment variable is required" + echo "Create one at: ${GITEA_URL}/user/settings/applications" + exit 1 +fi + +RPM_FILE=$(ls dist/mcpctl-*.rpm 2>/dev/null | head -1) +if [ -z "$RPM_FILE" ]; then + echo "Error: No RPM file found in dist/. Run scripts/build-rpm.sh first." + exit 1 +fi + +echo "==> Publishing $RPM_FILE to ${GITEA_URL}..." +curl --fail -X PUT \ + -H "Authorization: token ${GITEA_TOKEN}" \ + --upload-file "$RPM_FILE" \ + "${GITEA_URL}/api/packages/${GITEA_OWNER}/rpm/upload" + +echo "" +echo "==> Published successfully!" +echo "" +echo "Users can install with:" +echo " dnf config-manager --add-repo ${GITEA_URL}/api/packages/${GITEA_OWNER}/rpm.repo" +echo " dnf install mcpctl"