|
| 1 | +name: Setup spiceio |
| 2 | +description: Install a released version of spiceio and start the S3-to-SMB proxy |
| 3 | + |
| 4 | +inputs: |
| 5 | + version: |
| 6 | + description: Release tag to install (e.g. "v0.1.0"). Use "latest" for the most recent release. |
| 7 | + required: false |
| 8 | + default: latest |
| 9 | + smb-url: |
| 10 | + description: > |
| 11 | + SMB connection URL: smb://user:pass@server/share or smb://user:pass@server:port/share. |
| 12 | + Password can also be provided separately via smb-pass. |
| 13 | + required: true |
| 14 | + smb-pass: |
| 15 | + description: SMB password (overrides password in smb-url if both provided) |
| 16 | + required: false |
| 17 | + default: "" |
| 18 | + bucket: |
| 19 | + description: Virtual S3 bucket name |
| 20 | + required: false |
| 21 | + default: spiceio |
| 22 | + region: |
| 23 | + description: AWS region to advertise |
| 24 | + required: false |
| 25 | + default: us-east-1 |
| 26 | + bind: |
| 27 | + description: Listen address for the S3 endpoint |
| 28 | + required: false |
| 29 | + default: "127.0.0.1:8333" |
| 30 | + token: |
| 31 | + description: GitHub token for downloading release assets from private repos |
| 32 | + required: true |
| 33 | + |
| 34 | +outputs: |
| 35 | + endpoint: |
| 36 | + description: The S3-compatible endpoint URL |
| 37 | + value: ${{ steps.start.outputs.endpoint }} |
| 38 | + pid: |
| 39 | + description: PID of the spiceio background process (empty if skipped) |
| 40 | + value: ${{ steps.start.outputs.pid }} |
| 41 | + |
| 42 | +runs: |
| 43 | + using: composite |
| 44 | + steps: |
| 45 | + - name: Download and install spiceio |
| 46 | + id: install |
| 47 | + shell: bash |
| 48 | + env: |
| 49 | + GH_TOKEN: ${{ inputs.token }} |
| 50 | + run: | |
| 51 | + set -euo pipefail |
| 52 | +
|
| 53 | + REPO="spiceai/spiceio" |
| 54 | + INSTALL_DIR="${RUNNER_TEMP}/spiceio-bin" |
| 55 | + ASSET="spiceio-${RUNNER_OS}-${RUNNER_ARCH}.tar.gz" |
| 56 | + VERSION="${{ inputs.version }}" |
| 57 | +
|
| 58 | + mkdir -p "$INSTALL_DIR" |
| 59 | +
|
| 60 | + if [[ "$VERSION" == "latest" ]]; then |
| 61 | + gh release download --repo "$REPO" --pattern "$ASSET" --pattern "${ASSET}.sha256" --dir "$INSTALL_DIR" |
| 62 | + else |
| 63 | + gh release download "$VERSION" --repo "$REPO" --pattern "$ASSET" --pattern "${ASSET}.sha256" --dir "$INSTALL_DIR" |
| 64 | + fi |
| 65 | +
|
| 66 | + # Verify integrity |
| 67 | + cd "$INSTALL_DIR" |
| 68 | + shasum -a 256 -c "${ASSET}.sha256" |
| 69 | +
|
| 70 | + tar xzf "$ASSET" -C "$INSTALL_DIR" |
| 71 | + chmod +x "$INSTALL_DIR/spiceio" |
| 72 | + echo "$INSTALL_DIR" >> "$GITHUB_PATH" |
| 73 | + echo "pid_file=${RUNNER_TEMP}/spiceio.pid" >> "$GITHUB_OUTPUT" |
| 74 | +
|
| 75 | + - name: Start spiceio |
| 76 | + id: start |
| 77 | + shell: bash |
| 78 | + env: |
| 79 | + SMB_URL: ${{ inputs.smb-url }} |
| 80 | + SMB_PASS_OVERRIDE: ${{ inputs.smb-pass }} |
| 81 | + SPICEIO_BUCKET: ${{ inputs.bucket }} |
| 82 | + SPICEIO_REGION: ${{ inputs.region }} |
| 83 | + SPICEIO_BIND: ${{ inputs.bind }} |
| 84 | + run: | |
| 85 | + set -euo pipefail |
| 86 | +
|
| 87 | + ENDPOINT="http://${SPICEIO_BIND}" |
| 88 | + PID_FILE="${{ steps.install.outputs.pid_file }}" |
| 89 | +
|
| 90 | + # Skip if spiceio is already listening on the requested address |
| 91 | + # Verify it's spiceio by checking for the "spiceio" server header |
| 92 | + if curl -sf -I "$ENDPOINT/" 2>/dev/null | grep -qi "server: spiceio"; then |
| 93 | + echo "spiceio already running at $ENDPOINT — skipping start" |
| 94 | + echo "endpoint=$ENDPOINT" >> "$GITHUB_OUTPUT" |
| 95 | + echo "pid=" >> "$GITHUB_OUTPUT" |
| 96 | + echo "skipped=true" >> "$GITHUB_OUTPUT" |
| 97 | + exit 0 |
| 98 | + fi |
| 99 | +
|
| 100 | + # Parse smb://user:pass@server:port/share |
| 101 | + # Strips the smb:// prefix, then splits on :, @, / |
| 102 | + URL="${SMB_URL#smb://}" |
| 103 | +
|
| 104 | + USERINFO="${URL%%@*}" |
| 105 | + HOSTPATH="${URL#*@}" |
| 106 | +
|
| 107 | + SPICEIO_SMB_USER="${USERINFO%%:*}" |
| 108 | + # Extract password from URL only if user:pass@ format is used |
| 109 | + if [[ "$USERINFO" == *:* ]]; then |
| 110 | + URL_PASS="${USERINFO#*:}" |
| 111 | + else |
| 112 | + URL_PASS="" |
| 113 | + fi |
| 114 | + # smb-pass input takes priority, then URL password |
| 115 | + if [[ -n "$SMB_PASS_OVERRIDE" ]]; then |
| 116 | + SPICEIO_SMB_PASS="$SMB_PASS_OVERRIDE" |
| 117 | + elif [[ -n "$URL_PASS" ]]; then |
| 118 | + SPICEIO_SMB_PASS="$URL_PASS" |
| 119 | + else |
| 120 | + echo "::error::No SMB password provided (use smb-pass input or include in smb-url)" |
| 121 | + exit 1 |
| 122 | + fi |
| 123 | +
|
| 124 | + HOSTPORT="${HOSTPATH%%/*}" |
| 125 | + SPICEIO_SMB_SHARE="${HOSTPATH#*/}" |
| 126 | +
|
| 127 | + if [[ "$HOSTPORT" == *:* ]]; then |
| 128 | + SPICEIO_SMB_SERVER="${HOSTPORT%%:*}" |
| 129 | + SPICEIO_SMB_PORT="${HOSTPORT#*:}" |
| 130 | + else |
| 131 | + SPICEIO_SMB_SERVER="$HOSTPORT" |
| 132 | + SPICEIO_SMB_PORT="445" |
| 133 | + fi |
| 134 | +
|
| 135 | + export SPICEIO_SMB_SERVER SPICEIO_SMB_PORT SPICEIO_SMB_USER SPICEIO_SMB_PASS SPICEIO_SMB_SHARE |
| 136 | + export SPICEIO_BUCKET SPICEIO_REGION SPICEIO_BIND |
| 137 | +
|
| 138 | + spiceio & |
| 139 | + PID=$! |
| 140 | + echo "$PID" > "$PID_FILE" |
| 141 | + echo "pid=$PID" >> "$GITHUB_OUTPUT" |
| 142 | + echo "endpoint=$ENDPOINT" >> "$GITHUB_OUTPUT" |
| 143 | + echo "skipped=false" >> "$GITHUB_OUTPUT" |
| 144 | +
|
| 145 | + # Wait for readiness |
| 146 | + echo "Waiting for spiceio on ${SPICEIO_BIND}..." |
| 147 | + for i in $(seq 1 30); do |
| 148 | + if curl -sf -o /dev/null "$ENDPOINT/" 2>/dev/null; then |
| 149 | + echo "spiceio ready at $ENDPOINT (PID $PID)" |
| 150 | + exit 0 |
| 151 | + fi |
| 152 | + if ! kill -0 "$PID" 2>/dev/null; then |
| 153 | + echo "::error::spiceio exited unexpectedly" |
| 154 | + exit 1 |
| 155 | + fi |
| 156 | + sleep 1 |
| 157 | + done |
| 158 | + echo "::error::spiceio failed to start within 30s" |
| 159 | + exit 1 |
| 160 | +
|
| 161 | + - name: Register cleanup |
| 162 | + if: always() && steps.start.outputs.skipped != 'true' |
| 163 | + shell: bash |
| 164 | + run: | |
| 165 | + PID_FILE="${{ steps.install.outputs.pid_file }}" |
| 166 | + if [[ -f "$PID_FILE" ]]; then |
| 167 | + PID=$(cat "$PID_FILE") |
| 168 | + if kill -0 "$PID" 2>/dev/null; then |
| 169 | + echo "Stopping spiceio (PID $PID)" |
| 170 | + kill "$PID" 2>/dev/null || true |
| 171 | + wait "$PID" 2>/dev/null || true |
| 172 | + fi |
| 173 | + rm -f "$PID_FILE" |
| 174 | + fi |
0 commit comments