This commit is contained in:
commit
464b733e67
6 changed files with 163 additions and 0 deletions
83
.forgejo/workflows/deploy.yml
Normal file
83
.forgejo/workflows/deploy.yml
Normal file
|
|
@ -0,0 +1,83 @@
|
|||
# FnKit Deploy — Forgejo Actions
|
||||
# Builds and deploys this function container on every push to main
|
||||
# Requires: Forgejo runner with Docker socket access (fnkit deploy runner)
|
||||
#
|
||||
# Pipeline: git push → build image → deploy container → health check
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [main]
|
||||
|
||||
jobs:
|
||||
deploy:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Build image
|
||||
run: |
|
||||
FUNCTION_NAME="fn-python"
|
||||
IMAGE_NAME="fnkit-fn-${FUNCTION_NAME}:latest"
|
||||
IMAGE_PREV="fnkit-fn-${FUNCTION_NAME}:prev"
|
||||
|
||||
echo "🔨 Building ${FUNCTION_NAME}..."
|
||||
docker build -t $IMAGE_NAME .
|
||||
|
||||
echo "IMAGE_NAME=$IMAGE_NAME" >> $GITHUB_ENV
|
||||
echo "IMAGE_PREV=$IMAGE_PREV" >> $GITHUB_ENV
|
||||
echo "FUNCTION_NAME=$FUNCTION_NAME" >> $GITHUB_ENV
|
||||
|
||||
- name: Deploy container
|
||||
run: |
|
||||
echo "🌐 Ensuring fnkit-network exists..."
|
||||
docker network create fnkit-network 2>/dev/null || true
|
||||
|
||||
# Tag current image as :prev for rollback
|
||||
docker tag $IMAGE_NAME $IMAGE_PREV 2>/dev/null || true
|
||||
|
||||
echo "♻️ Replacing running container..."
|
||||
docker stop $FUNCTION_NAME 2>/dev/null || true
|
||||
docker rm $FUNCTION_NAME 2>/dev/null || true
|
||||
|
||||
echo "🚀 Starting ${FUNCTION_NAME}..."
|
||||
docker run -d \
|
||||
--name $FUNCTION_NAME \
|
||||
--network fnkit-network \
|
||||
--label fnkit.fn=true \
|
||||
--label fnkit.deployed="$(date -u +%Y-%m-%dT%H:%M:%SZ)" \
|
||||
--restart unless-stopped \
|
||||
$IMAGE_NAME
|
||||
|
||||
- name: Health check
|
||||
run: |
|
||||
echo "🏥 Checking container health..."
|
||||
sleep 3
|
||||
|
||||
if docker ps --filter "name=$FUNCTION_NAME" --filter "status=running" -q | grep -q .; then
|
||||
echo "✅ ${FUNCTION_NAME} is running"
|
||||
echo "🌐 Available at gateway: /${FUNCTION_NAME}"
|
||||
else
|
||||
echo "❌ Container failed to start — rolling back..."
|
||||
docker logs $FUNCTION_NAME 2>&1 || true
|
||||
|
||||
# Rollback to previous image if available
|
||||
if docker image inspect $IMAGE_PREV >/dev/null 2>&1; then
|
||||
echo "🔄 Rolling back to previous image..."
|
||||
docker rm $FUNCTION_NAME 2>/dev/null || true
|
||||
docker run -d \
|
||||
--name $FUNCTION_NAME \
|
||||
--network fnkit-network \
|
||||
--label fnkit.fn=true \
|
||||
--label fnkit.deployed="$(date -u +%Y-%m-%dT%H:%M:%SZ)" \
|
||||
--label fnkit.rollback=true \
|
||||
--restart unless-stopped \
|
||||
$IMAGE_PREV
|
||||
echo "⚠️ Rolled back to previous version"
|
||||
fi
|
||||
exit 1
|
||||
fi
|
||||
|
||||
- name: Cleanup old images
|
||||
run: |
|
||||
echo "🧹 Cleaning up dangling images..."
|
||||
docker image prune -f --filter "label=fnkit.fn=true" 2>/dev/null || true
|
||||
18
.gitignore
vendored
Normal file
18
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,18 @@
|
|||
# IDE
|
||||
.idea/
|
||||
.vscode/
|
||||
*.swp
|
||||
*.swo
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Docker
|
||||
Dockerfile.local
|
||||
# Python
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
.env
|
||||
venv/
|
||||
.venv/
|
||||
7
Dockerfile
Normal file
7
Dockerfile
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
FROM python:3.11-slim
|
||||
LABEL fnkit.fn="true"
|
||||
WORKDIR /app
|
||||
COPY requirements.txt .
|
||||
RUN pip install --no-cache-dir -r requirements.txt
|
||||
COPY . .
|
||||
CMD ["functions-framework", "--target=hello", "--port=8080"]
|
||||
36
docker-compose.yml
Normal file
36
docker-compose.yml
Normal file
|
|
@ -0,0 +1,36 @@
|
|||
# Docker Compose for FnKit function with gateway integration
|
||||
# Requires: docker network create fnkit-network
|
||||
# Requires: fnkit-gateway running (fnkit gateway init && fnkit gateway build && fnkit gateway start)
|
||||
|
||||
version: '3.8'
|
||||
|
||||
services:
|
||||
fn-python:
|
||||
build: .
|
||||
container_name: fn-python
|
||||
networks:
|
||||
- fnkit-network
|
||||
depends_on:
|
||||
fnkit-gateway:
|
||||
condition: service_started
|
||||
restart: unless-stopped
|
||||
|
||||
# Uncomment to include gateway in this compose file
|
||||
# fnkit-gateway:
|
||||
# image: fnkit-gateway:latest
|
||||
# container_name: fnkit-gateway
|
||||
# ports:
|
||||
# - "8080:8080"
|
||||
# environment:
|
||||
# - FNKIT_AUTH_TOKEN=${FNKIT_AUTH_TOKEN:-}
|
||||
# networks:
|
||||
# - fnkit-network
|
||||
|
||||
networks:
|
||||
fnkit-network:
|
||||
name: fnkit-network
|
||||
external: true
|
||||
|
||||
# Usage:
|
||||
# docker-compose up -d
|
||||
# curl -H "Authorization: Bearer <token>" http://localhost:8080/fn-python
|
||||
18
main.py
Normal file
18
main.py
Normal file
|
|
@ -0,0 +1,18 @@
|
|||
import flask
|
||||
import functions_framework
|
||||
|
||||
|
||||
@functions_framework.http
|
||||
def hello(request: flask.Request) -> flask.typing.ResponseReturnValue:
|
||||
"""HTTP Cloud Function.
|
||||
|
||||
Args:
|
||||
request: The request object.
|
||||
<https://flask.palletsprojects.com/en/latest/api/#flask.Request>
|
||||
|
||||
Returns:
|
||||
The response text, or any set of values that can be turned into a
|
||||
Response object using `make_response`.
|
||||
<https://flask.palletsprojects.com/en/latest/api/#flask.make_response>
|
||||
"""
|
||||
return "Hello, World!"
|
||||
1
requirements.txt
Normal file
1
requirements.txt
Normal file
|
|
@ -0,0 +1 @@
|
|||
functions-framework==3.*
|
||||
Loading…
Add table
Reference in a new issue