This commit is contained in:
@@ -11,7 +11,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: https://gitea.com/actions/checkout@v4
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3
|
uses: docker/setup-buildx-action@v3
|
||||||
@@ -25,32 +25,45 @@ jobs:
|
|||||||
|
|
||||||
- name: Build and push postgres image
|
- name: Build and push postgres image
|
||||||
uses: docker/build-push-action@v5
|
uses: docker/build-push-action@v5
|
||||||
if: |
|
# if: |
|
||||||
contains(github.event.commits[0].modified, 'infra/docker/postgres-pg-cron') ||
|
# contains(github.event.commits[0].modified, 'infra/docker/postgres-pg-cron') ||
|
||||||
contains(github.event.commits[0].added, 'infra/docker/postgres-pg-cron')
|
# contains(github.event.commits[0].added, 'infra/docker/postgres-pg-cron')
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: infra/docker/postgres-pg-cron/Dockerfile
|
file: infra/docker/postgres-pg-cron/Dockerfile
|
||||||
push: true
|
push: true
|
||||||
tags: |
|
tags: |
|
||||||
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub-postgres:latest
|
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub-postgres:latest
|
||||||
cache-from: type=gha
|
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:${{ github.sha }}
|
||||||
cache-to: type=gha,mode=max
|
cache-from: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache
|
||||||
|
cache-to: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache,mode=max
|
||||||
|
|
||||||
|
# - name: Build and push webhook image
|
||||||
|
# uses: docker/build-push-action@v5
|
||||||
|
# with:
|
||||||
|
# context: .
|
||||||
|
# file: infra/webhook/Dockerfile
|
||||||
|
# push: true
|
||||||
|
# tags: |
|
||||||
|
# ${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub-webhook:latest
|
||||||
|
# ${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:${{ github.sha }}
|
||||||
|
# cache-from: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache
|
||||||
|
# cache-to: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache,mode=max
|
||||||
|
|
||||||
- name: Build and push app image
|
- name: Build and push app image
|
||||||
uses: docker/build-push-action@v5
|
uses: docker/build-push-action@v5
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: infra/docker/application/Dockerfile
|
file: infra/application/Dockerfile
|
||||||
push: true
|
push: true
|
||||||
tags: |
|
tags: |
|
||||||
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:latest
|
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:latest
|
||||||
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:${{ github.sha }}
|
${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:${{ github.sha }}
|
||||||
cache-from: type=gha
|
cache-from: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache
|
||||||
cache-to: type=gha,mode=max
|
cache-to: type=registry,ref=${{ secrets.REGISTRY_URL }}/${{ secrets.REGISTRY_USER }}/familyhub:cache,mode=max
|
||||||
|
|
||||||
- name: Trigger deploy
|
# - name: Trigger deploy
|
||||||
run: |
|
# run: |
|
||||||
curl -s -X POST \
|
# curl -s -X POST \
|
||||||
-H "X-Webhook-Secret: ${{ secrets.WEBHOOK_SECRET }}" \
|
# -H "X-Webhook-Secret: ${{ secrets.WEBHOOK_SECRET }}" \
|
||||||
"http://10.0.0.2:9001/deploy?container=familyhub"
|
# "http://10.0.0.2:9001/deploy?container=familyhub"
|
||||||
@@ -14,7 +14,7 @@ RUN npm run build
|
|||||||
# ================================
|
# ================================
|
||||||
# Stage 2: сборка Go
|
# Stage 2: сборка Go
|
||||||
# ================================
|
# ================================
|
||||||
FROM golang:1.25-alpine AS backend
|
FROM golang:1.26-bookworm AS backend
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
@@ -27,22 +27,19 @@ COPY backend/ ./
|
|||||||
|
|
||||||
# встраиваем собранную статику Vue
|
# встраиваем собранную статику Vue
|
||||||
COPY --from=frontend /app/dist ./src/api/dist
|
COPY --from=frontend /app/dist ./src/api/dist
|
||||||
|
# Миграции кладём туда, откуда Go их ищет
|
||||||
|
COPY backend/migrations ./migrations
|
||||||
# сборка бинарника
|
# сборка бинарника
|
||||||
RUN CGO_ENABLED=0 GOOS=linux go build -o server ./src/
|
RUN CGO_ENABLED=0 GOOS=linux go build -o server ./src/
|
||||||
|
|
||||||
# ================================
|
# ================================
|
||||||
# Stage 3: финальный образ
|
# Stage 3: финальный образ
|
||||||
# ================================
|
# ================================
|
||||||
FROM alpine:3.19
|
FROM scratch
|
||||||
|
|
||||||
# нужен для корректной работы TLS и временных зон
|
COPY --from=backend /app/server /server
|
||||||
RUN apk add --no-cache ca-certificates tzdata
|
COPY --from=backend /app/migrations /migrations
|
||||||
|
COPY --from=backend /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/
|
||||||
|
COPY --from=backend /usr/share/zoneinfo /usr/share/zoneinfo
|
||||||
|
|
||||||
WORKDIR /app
|
ENTRYPOINT ["/server"]
|
||||||
|
|
||||||
COPY --from=backend /app/server ./server
|
|
||||||
|
|
||||||
EXPOSE 8080
|
|
||||||
|
|
||||||
ENTRYPOINT ["./server"]
|
|
||||||
+22
-13
@@ -3,16 +3,23 @@ version: '3.9'
|
|||||||
services:
|
services:
|
||||||
app:
|
app:
|
||||||
image: git.myhomecloud.tech/admin/familyhub:latest
|
image: git.myhomecloud.tech/admin/familyhub:latest
|
||||||
container_name: familyhub
|
container_name: application
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
ports:
|
ports:
|
||||||
- "10.0.0.2:8000:8000" # только через WireGuard
|
- "8000:8000"
|
||||||
environment:
|
environment:
|
||||||
- DB_HOST=db
|
- DB_HOST=${DB_HOST}
|
||||||
- DB_PORT=5432
|
- DB_PORT=${DB_PORT}
|
||||||
- DB_USER=familyUser
|
- DB_USER=${DB_USER}
|
||||||
- DB_PASSWORD=familyPass
|
- DB_PASSWORD=${DB_PASSWORD}
|
||||||
- DB_NAME=familyHubDB
|
- DB_NAME=${DB_NAME}
|
||||||
|
- BOT_TOKEN=${BOT_TOKEN}
|
||||||
|
- GOOGLE_APPLICATION_CREDENTIALS=${GOOGLE_APPLICATION_CREDENTIALS}
|
||||||
|
- RUN_MODE=${RUN_MODE}
|
||||||
|
- API_SECRET=${API_SECRET}
|
||||||
|
- DB_PATH=postgres://${DB_USER}:${DB_PASSWORD}@${DB_HOST}:${DB_PORT}/${DB_NAME}?sslmode=disable
|
||||||
|
- OPEN_API_ENABLED=${OPEN_API_ENABLED}
|
||||||
|
- DEBUG_MODE=${DEBUG_MODE}
|
||||||
depends_on:
|
depends_on:
|
||||||
- db
|
- db
|
||||||
networks:
|
networks:
|
||||||
@@ -23,6 +30,8 @@ services:
|
|||||||
container_name: postgres
|
container_name: postgres
|
||||||
restart: always
|
restart: always
|
||||||
pull_policy: always
|
pull_policy: always
|
||||||
|
ports:
|
||||||
|
- "5432:5432"
|
||||||
command:
|
command:
|
||||||
- postgres
|
- postgres
|
||||||
- -c
|
- -c
|
||||||
@@ -30,9 +39,9 @@ services:
|
|||||||
- -c
|
- -c
|
||||||
- cron.database_name=familyHubDB
|
- cron.database_name=familyHubDB
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: familyUser
|
POSTGRES_USER: ${DB_USER}
|
||||||
POSTGRES_PASSWORD: familyPass
|
POSTGRES_PASSWORD: ${DB_PASSWORD}
|
||||||
POSTGRES_DB: familyHubDB
|
POSTGRES_DB: ${DB_NAME}
|
||||||
volumes:
|
volumes:
|
||||||
- postgres-data:/var/lib/postgresql/data
|
- postgres-data:/var/lib/postgresql/data
|
||||||
- ./init:/docker-entrypoint-initdb.d
|
- ./init:/docker-entrypoint-initdb.d
|
||||||
@@ -45,14 +54,14 @@ services:
|
|||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
pull_policy: always
|
pull_policy: always
|
||||||
ports:
|
ports:
|
||||||
- "10.0.0.2:9001:9001"
|
- "9001:9001"
|
||||||
environment:
|
environment:
|
||||||
- WEBHOOK_SECRET=${WEBHOOK_SECRET}
|
- WEBHOOK_SECRET=${WEBHOOK_SECRET}
|
||||||
- COMPOSE_FILE=/compose/docker-compose.yml
|
# - COMPOSE_FILE=/compose/docker-compose.yml
|
||||||
- COMPOSE_PROJECT=familyhub
|
- COMPOSE_PROJECT=familyhub
|
||||||
volumes:
|
volumes:
|
||||||
- /var/run/docker.sock:/var/run/docker.sock
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
- ./docker-compose.yml:/compose/docker-compose.yml:ro
|
# - ./docker-compose.yml:/compose/docker-compose.yml:ro
|
||||||
networks:
|
networks:
|
||||||
- family-hub-net
|
- family-hub-net
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
FROM golang:1.25-alpine AS builder
|
FROM golang:1.26-bookworm AS builder
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY infra/webhook/ .
|
COPY infra/webhook/ .
|
||||||
RUN go mod download && \
|
RUN go mod download && \
|
||||||
CGO_ENABLED=0 GOOS=linux go build -o webhook .
|
CGO_ENABLED=0 GOOS=linux go build -o webhook .
|
||||||
|
|
||||||
FROM alpine:3.19
|
FROM scratch
|
||||||
RUN apk add --no-cache docker-cli ca-certificates
|
|
||||||
COPY --from=builder /app/webhook /webhook
|
COPY --from=builder /app/webhook /webhook
|
||||||
|
COPY --from=builder /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/
|
||||||
EXPOSE 9001
|
EXPOSE 9001
|
||||||
ENTRYPOINT ["/webhook"]
|
ENTRYPOINT ["/webhook"]
|
||||||
@@ -1,5 +1,3 @@
|
|||||||
module webhook
|
module webhook
|
||||||
|
|
||||||
go 1.25
|
go 1.25.0
|
||||||
|
|
||||||
require github.com/docker/docker v26.1.0+incompatible
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
github.com/docker/docker v26.1.0+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
|
|
||||||
|
|||||||
+78
-32
@@ -3,79 +3,125 @@ package main
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
|
|
||||||
"github.com/docker/docker/client"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func newDockerClient() *http.Client {
|
||||||
|
return &http.Client{
|
||||||
|
Transport: &http.Transport{
|
||||||
|
DialContext: func(ctx context.Context, _, _ string) (net.Conn, error) {
|
||||||
|
return net.Dial("unix", "/var/run/docker.sock")
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getImageName(containerName string) (string, error) {
|
||||||
|
client := newDockerClient()
|
||||||
|
resp, err := client.Get("http://localhost/containers/" + containerName + "/json")
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
var result struct {
|
||||||
|
Config struct {
|
||||||
|
Image string `json:"Image"`
|
||||||
|
} `json:"Config"`
|
||||||
|
}
|
||||||
|
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return result.Config.Image, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func pullImage(imageName string) error {
|
||||||
|
client := newDockerClient()
|
||||||
|
resp, err := client.Post(
|
||||||
|
"http://localhost/images/create?fromImage="+url.QueryEscape(imageName),
|
||||||
|
"application/json",
|
||||||
|
nil,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
// читаем до конца чтобы pull завершился полностью
|
||||||
|
io.Copy(io.Discard, resp.Body)
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return fmt.Errorf("pull failed with status: %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func restartContainer(containerName string) error {
|
||||||
|
client := newDockerClient()
|
||||||
|
resp, err := client.Post(
|
||||||
|
"http://localhost/containers/"+containerName+"/restart",
|
||||||
|
"application/json",
|
||||||
|
nil,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusNoContent {
|
||||||
|
return fmt.Errorf("restart failed with status: %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
secret := os.Getenv("WEBHOOK_SECRET")
|
secret := os.Getenv("WEBHOOK_SECRET")
|
||||||
|
|
||||||
cli, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation())
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal("failed to create docker client:", err)
|
|
||||||
}
|
|
||||||
defer cli.Close()
|
|
||||||
|
|
||||||
http.HandleFunc("/deploy", func(w http.ResponseWriter, r *http.Request) {
|
http.HandleFunc("/deploy", func(w http.ResponseWriter, r *http.Request) {
|
||||||
if r.Method != http.MethodPost {
|
if r.Method != http.MethodPost {
|
||||||
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
http.Error(w, "method not allowed", http.StatusMethodNotAllowed)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if r.Header.Get("X-Webhook-Secret") != secret {
|
if r.Header.Get("X-Webhook-Secret") != secret {
|
||||||
http.Error(w, "unauthorized", http.StatusUnauthorized)
|
http.Error(w, "unauthorized", http.StatusUnauthorized)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// имя контейнера берём из query параметра
|
|
||||||
containerName := r.URL.Query().Get("container")
|
containerName := r.URL.Query().Get("container")
|
||||||
if containerName == "" {
|
if containerName == "" {
|
||||||
http.Error(w, "container parameter is required", http.StatusBadRequest)
|
http.Error(w, "container parameter is required", http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx := context.Background()
|
imageName, err := getImageName(containerName)
|
||||||
|
|
||||||
// получаем инфо о контейнере чтобы узнать имя образа
|
|
||||||
inspect, err := cli.ContainerInspect(ctx, containerName)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("failed to inspect container %s: %v", containerName, err)
|
log.Printf("failed to inspect container %s: %v", containerName, err)
|
||||||
http.Error(w, "container not found", http.StatusNotFound)
|
http.Error(w, "container not found", http.StatusNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
imageName := inspect.Config.Image
|
|
||||||
log.Printf("Container: %s, Image: %s", containerName, imageName)
|
log.Printf("Container: %s, Image: %s", containerName, imageName)
|
||||||
|
|
||||||
// тянем новый образ
|
log.Printf("Pulling image %s...", imageName)
|
||||||
log.Println("Pulling new image...")
|
if err := pullImage(imageName); err != nil {
|
||||||
pull := exec.Command("docker", "pull", imageName)
|
log.Printf("pull failed: %v", err)
|
||||||
pull.Stdout = os.Stdout
|
|
||||||
pull.Stderr = os.Stderr
|
|
||||||
if err := pull.Run(); err != nil {
|
|
||||||
log.Println("pull failed:", err)
|
|
||||||
http.Error(w, "pull failed", http.StatusInternalServerError)
|
http.Error(w, "pull failed", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// перезапускаем контейнер
|
|
||||||
log.Printf("Restarting container %s...", containerName)
|
log.Printf("Restarting container %s...", containerName)
|
||||||
restart := exec.Command("docker", "restart", containerName)
|
if err := restartContainer(containerName); err != nil {
|
||||||
restart.Stdout = os.Stdout
|
log.Printf("restart failed: %v", err)
|
||||||
restart.Stderr = os.Stderr
|
|
||||||
if err := restart.Run(); err != nil {
|
|
||||||
log.Println("restart failed:", err)
|
|
||||||
http.Error(w, "restart failed", http.StatusInternalServerError)
|
http.Error(w, "restart failed", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Printf("Deploy of %s completed", containerName)
|
log.Printf("Deploy of %s completed", containerName)
|
||||||
w.WriteHeader(http.StatusOK)
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
|
||||||
json.NewEncoder(w).Encode(map[string]string{
|
json.NewEncoder(w).Encode(map[string]string{
|
||||||
"status": "ok",
|
"status": "ok",
|
||||||
"container": containerName,
|
"container": containerName,
|
||||||
|
|||||||
Reference in New Issue
Block a user