Compare commits
1 Commits
feat-ionos
...
75e527737c
| Author | SHA1 | Date | |
|---|---|---|---|
| 75e527737c |
@@ -9,27 +9,6 @@ env:
|
|||||||
DOCKER_REGISTRY: gitea.t000-n.de
|
DOCKER_REGISTRY: gitea.t000-n.de
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
check-changes:
|
|
||||||
name: Check changes
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
outputs:
|
|
||||||
code: ${{ steps.filter.outputs.code }}
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
- name: Get changes
|
|
||||||
id: filter
|
|
||||||
uses: dorny/paths-filter@v3
|
|
||||||
with:
|
|
||||||
filters: |
|
|
||||||
code:
|
|
||||||
- 'go.mod'
|
|
||||||
- 'go.sum'
|
|
||||||
- '**/*.go'
|
|
||||||
- 'config.example.yaml'
|
|
||||||
- 'Dockerfile'
|
|
||||||
- 'Makefile'
|
|
||||||
|
|
||||||
test:
|
test:
|
||||||
name: test
|
name: test
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
@@ -65,70 +44,41 @@ jobs:
|
|||||||
|
|
||||||
build_and_push:
|
build_and_push:
|
||||||
name: Build and push
|
name: Build and push
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
arch: [amd64, arm64]
|
|
||||||
needs:
|
needs:
|
||||||
- test
|
- test
|
||||||
- check-changes
|
|
||||||
if: ${{ needs.check-changes.outputs.code == 'true' }}
|
|
||||||
runs-on:
|
|
||||||
- ubuntu-latest
|
|
||||||
- linux_${{ matrix.arch }}
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
- name: Set up Docker Buildx
|
|
||||||
uses: docker/setup-buildx-action@v3
|
|
||||||
- name: Login to Registry
|
|
||||||
uses: docker/login-action@v3
|
|
||||||
with:
|
|
||||||
registry: ${{ env.DOCKER_REGISTRY }}
|
|
||||||
username: ${{ secrets.REGISTRY_USER }}
|
|
||||||
password: ${{ secrets.REGISTRY_PASSWORD }}
|
|
||||||
- name: Get Metadata
|
|
||||||
id: meta
|
|
||||||
run: |
|
|
||||||
echo REPO_NAME=$(echo ${GITHUB_REPOSITORY} | awk -F"/" '{print $2}' | tr '[:upper:]' '[:lower:]') >> $GITHUB_OUTPUT
|
|
||||||
echo REPO_VERSION=$(git describe --tags --always | sed 's/^v//') >> $GITHUB_OUTPUT
|
|
||||||
- name: Build and push
|
|
||||||
uses: docker/build-push-action@v6
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
file: ./Dockerfile
|
|
||||||
platforms: linux/${{ matrix.arch }}
|
|
||||||
push: true
|
|
||||||
provenance: false
|
|
||||||
build-args: GOARCH=${{ matrix.arch }}
|
|
||||||
tags: |
|
|
||||||
${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:${{ steps.meta.outputs.REPO_VERSION }}-${{ matrix.arch }}
|
|
||||||
|
|
||||||
create_manifest:
|
|
||||||
name: Create manifest
|
|
||||||
needs:
|
|
||||||
- build_and_push
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v2
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v2
|
||||||
|
|
||||||
|
- name: Login to Registry
|
||||||
|
uses: docker/login-action@v2
|
||||||
|
with:
|
||||||
|
registry: ${{ env.DOCKER_REGISTRY }}
|
||||||
|
username: ${{ secrets.REGISTRY_USER }}
|
||||||
|
password: ${{ secrets.REGISTRY_PASSWORD }}
|
||||||
|
|
||||||
- name: Get Metadata
|
- name: Get Metadata
|
||||||
id: meta
|
id: meta
|
||||||
run: |
|
run: |
|
||||||
echo REPO_NAME=$(echo ${GITHUB_REPOSITORY} | awk -F"/" '{print $2}' | tr '[:upper:]' '[:lower:]') >> $GITHUB_OUTPUT
|
echo REPO_NAME=$(echo ${GITHUB_REPOSITORY} | awk -F"/" '{print $2}' | tr '[:upper:]' '[:lower:]') >> $GITHUB_OUTPUT
|
||||||
echo REPO_VERSION=$(git describe --tags --always | sed 's/^v//') >> $GITHUB_OUTPUT
|
echo REPO_VERSION=$(git describe --tags --always | sed 's/^v//') >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Login to Registry
|
- name: Build and push
|
||||||
uses: docker/login-action@v3
|
uses: docker/build-push-action@v4
|
||||||
with:
|
with:
|
||||||
registry: ${{ env.DOCKER_REGISTRY }}
|
context: .
|
||||||
username: ${{ secrets.REGISTRY_USER }}
|
file: ./Dockerfile
|
||||||
password: ${{ secrets.REGISTRY_PASSWORD }}
|
platforms: |
|
||||||
|
linux/amd64
|
||||||
- name: Create manifest
|
linux/arm64
|
||||||
run: |
|
push: true
|
||||||
docker manifest create ${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:latest \
|
tags: |
|
||||||
${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:${{ steps.meta.outputs.REPO_VERSION }}-amd64 \
|
${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:${{ steps.meta.outputs.REPO_VERSION }}
|
||||||
${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:${{ steps.meta.outputs.REPO_VERSION }}-arm64
|
${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:latest
|
||||||
|
|
||||||
docker manifest push ${{ env.DOCKER_REGISTRY }}/t.behrendt/${{ steps.meta.outputs.REPO_NAME }}:latest
|
|
||||||
|
|||||||
21
Dockerfile
21
Dockerfile
@@ -1,14 +1,13 @@
|
|||||||
FROM golang:1.24-alpine as build
|
FROM golang:1.23-alpine
|
||||||
|
|
||||||
ARG GOARCH=amd64
|
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY go.mod go.sum ./
|
|
||||||
RUN go mod download
|
|
||||||
COPY . .
|
|
||||||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=${GOARCH} \
|
|
||||||
go build -trimpath -ldflags="-s -w" -o main .
|
|
||||||
|
|
||||||
FROM gcr.io/distroless/static-debian12
|
COPY go.mod go.sum ./
|
||||||
COPY --from=build /app/main /
|
|
||||||
CMD ["/main"]
|
RUN go mod download
|
||||||
|
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
RUN go build -o main .
|
||||||
|
|
||||||
|
CMD ["./main"]
|
||||||
|
|||||||
@@ -38,7 +38,6 @@ domains:
|
|||||||
- www
|
- www
|
||||||
check_interval: 0 0 0/6 * * * *
|
check_interval: 0 0 0/6 * * * *
|
||||||
mode: Scheduled
|
mode: Scheduled
|
||||||
log_level: info
|
|
||||||
```
|
```
|
||||||
|
|
||||||
The config file is expected to be in the same directory as the binary and called `config.yaml`. For the OCR image, the root directory is `/app`.
|
The config file is expected to be in the same directory as the binary and called `config.yaml`. For the OCR image, the root directory is `/app`.
|
||||||
|
|||||||
@@ -21,4 +21,3 @@ domains:
|
|||||||
- www
|
- www
|
||||||
check_interval: 0 0 0/6 * * * *
|
check_interval: 0 0 0/6 * * * *
|
||||||
mode: Scheduled
|
mode: Scheduled
|
||||||
log_level: info
|
|
||||||
|
|||||||
7
go.mod
7
go.mod
@@ -1,6 +1,6 @@
|
|||||||
module realdnydns
|
module realdnydns
|
||||||
|
|
||||||
go 1.24.0
|
go 1.23
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/go-co-op/gocron v1.37.0
|
github.com/go-co-op/gocron v1.37.0
|
||||||
@@ -8,11 +8,8 @@ require (
|
|||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/cespare/xxhash/v2 v2.2.0 // indirect
|
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
|
||||||
github.com/google/uuid v1.6.0 // indirect
|
github.com/google/uuid v1.6.0 // indirect
|
||||||
github.com/redis/go-redis/v9 v9.7.0 // indirect
|
|
||||||
github.com/robfig/cron/v3 v3.0.1 // indirect
|
github.com/robfig/cron/v3 v3.0.1 // indirect
|
||||||
github.com/stretchr/testify v1.10.0 // indirect
|
github.com/stretchr/testify v1.8.4 // indirect
|
||||||
go.uber.org/atomic v1.11.0 // indirect
|
go.uber.org/atomic v1.11.0 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
10
go.sum
10
go.sum
@@ -1,11 +1,7 @@
|
|||||||
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
|
|
||||||
github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
|
||||||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78=
|
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc=
|
|
||||||
github.com/go-co-op/gocron v1.37.0 h1:ZYDJGtQ4OMhTLKOKMIch+/CY70Brbb1dGdooLEhh7b0=
|
github.com/go-co-op/gocron v1.37.0 h1:ZYDJGtQ4OMhTLKOKMIch+/CY70Brbb1dGdooLEhh7b0=
|
||||||
github.com/go-co-op/gocron v1.37.0/go.mod h1:3L/n6BkO7ABj+TrfSVXLRzsP26zmikL4ISkLQ0O8iNY=
|
github.com/go-co-op/gocron v1.37.0/go.mod h1:3L/n6BkO7ABj+TrfSVXLRzsP26zmikL4ISkLQ0O8iNY=
|
||||||
github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
@@ -22,8 +18,6 @@ github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
|
|||||||
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
|
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/redis/go-redis/v9 v9.7.0 h1:HhLSs+B6O021gwzl+locl0zEDnyNkxMtf/Z3NNBMa9E=
|
|
||||||
github.com/redis/go-redis/v9 v9.7.0/go.mod h1:f6zhXITC7JUJIlPEiBOTXxJgPLdZcA93GewI7inzyWw=
|
|
||||||
github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs=
|
github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs=
|
||||||
github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro=
|
github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro=
|
||||||
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
||||||
@@ -36,8 +30,8 @@ github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UV
|
|||||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk=
|
||||||
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
||||||
go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
|
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
|
||||||
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
|
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
|
||||||
|
|||||||
48
main.go
48
main.go
@@ -25,7 +25,8 @@ func main() {
|
|||||||
configClient := config.Config{}
|
configClient := config.Config{}
|
||||||
err := configClient.Load("config.yaml")
|
err := configClient.Load("config.yaml")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to load config file", slog.String("error", err.Error()))
|
logger.Error("Failed to load config file",
|
||||||
|
slog.String("error", err.Error()))
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -38,64 +39,39 @@ func main() {
|
|||||||
var externalIpProvider externalIpProvider.ExternalIpProvider
|
var externalIpProvider externalIpProvider.ExternalIpProvider
|
||||||
switch configClient.ExternalIPProvider.Type {
|
switch configClient.ExternalIPProvider.Type {
|
||||||
case "plain":
|
case "plain":
|
||||||
logger.Info("Using plain external IP provider", slog.String("external_ip_provider", "plain"))
|
|
||||||
|
|
||||||
var plainConfig plainExternalIpProvider.PlainExternalIpProviderConfig
|
var plainConfig plainExternalIpProvider.PlainExternalIpProviderConfig
|
||||||
err := configClient.ExternalIPProvider.ProviderConfig.Decode(&plainConfig)
|
err := configClient.ExternalIPProvider.ProviderConfig.Decode(&plainConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create config",
|
|
||||||
slog.String("external_ip_provider", "plain"),
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
externalIpProvider, err = plainExternalIpProvider.New(plainConfig)
|
externalIpProvider, err = plainExternalIpProvider.New(plainConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create plain external IP provider",
|
|
||||||
slog.String("external_ip_provider", "plain"),
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
logger.Error("Unknown external IP provider", slog.String("external_ip_provider", configClient.ExternalIPProvider.Type))
|
|
||||||
panic(fmt.Errorf("unknown external IP provider: %s", configClient.ExternalIPProvider.Type))
|
panic(fmt.Errorf("unknown external IP provider: %s", configClient.ExternalIPProvider.Type))
|
||||||
}
|
}
|
||||||
|
|
||||||
var dnsProvider dnsProvider.DNSProvider
|
var dnsProvider dnsProvider.DNSProvider
|
||||||
switch configClient.DNSProvider.Type {
|
switch configClient.DNSProvider.Type {
|
||||||
case "ionos":
|
case "ionos":
|
||||||
logger.Info("Using IONOS DNS provider", slog.String("dns_provider", "ionos"))
|
|
||||||
|
|
||||||
var ionosConfig ionos.IONOSConfig
|
var ionosConfig ionos.IONOSConfig
|
||||||
err := configClient.DNSProvider.ProviderConfig.Decode(&ionosConfig)
|
err := configClient.DNSProvider.ProviderConfig.Decode(&ionosConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create IONOS DNS provider",
|
|
||||||
slog.String("dns_provider", "ionos"),
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
dnsProvider, err = ionos.NewIonos(&ionosConfig)
|
dnsProvider, err = ionos.NewIonos(&ionosConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create IONOS DNS provider",
|
|
||||||
slog.String("dns_provider", "ionos"),
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
logger.Error("Unknown DNS provider", slog.String("dns_provider", configClient.DNSProvider.Type))
|
|
||||||
panic(fmt.Errorf("unknown DNS provider: %s", configClient.DNSProvider.Type))
|
panic(fmt.Errorf("unknown DNS provider: %s", configClient.DNSProvider.Type))
|
||||||
}
|
}
|
||||||
|
|
||||||
var notificationProvider notificationProvider.NotificationProvider
|
var notificationProvider notificationProvider.NotificationProvider
|
||||||
switch configClient.NotificationProvider.Type {
|
switch configClient.NotificationProvider.Type {
|
||||||
case "gotify":
|
case "gotify":
|
||||||
logger.Info("Using Gotify notification provider", slog.String("notification_provider", "gotify"))
|
|
||||||
|
|
||||||
var gotifyConfig gotify.NotificationProviderImplGotifyConfig
|
var gotifyConfig gotify.NotificationProviderImplGotifyConfig
|
||||||
err := configClient.NotificationProvider.ProviderConfig.Decode(&gotifyConfig)
|
err := configClient.NotificationProvider.ProviderConfig.Decode(&gotifyConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -104,39 +80,29 @@ func main() {
|
|||||||
|
|
||||||
notificationProvider, err = gotify.New(gotifyConfig)
|
notificationProvider, err = gotify.New(gotifyConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create Gotify notification provider",
|
|
||||||
slog.String("notification_provider", "gotify"),
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
logger.Info("Using console notification provider", slog.String("notification_provider", "console"))
|
|
||||||
|
|
||||||
notificationProvider = notificationProviderConsole.New()
|
notificationProvider = notificationProviderConsole.New()
|
||||||
}
|
}
|
||||||
|
|
||||||
rdd := realDynDns.New(externalIpProvider, dnsProvider, notificationProvider, configClient.Domains, logger.With(slog.String("service", "realDynDns")))
|
rdd := realDynDns.New(externalIpProvider, dnsProvider, notificationProvider, configClient.Domains)
|
||||||
|
|
||||||
switch configClient.Mode {
|
switch configClient.Mode {
|
||||||
case config.ScheduledMode:
|
case config.ScheduledMode:
|
||||||
logger.Info("Running in scheduled mode", slog.String("interval", configClient.CheckInterval))
|
|
||||||
|
|
||||||
schedule, job, err := rdd.RunWithSchedule(configClient.CheckInterval)
|
schedule, job, err := rdd.RunWithSchedule(configClient.CheckInterval)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to create scheduler", slog.String("error", err.Error()))
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Info("Next run:", slog.String("time", job.NextRun().String()))
|
fmt.Println("Starting scheduler")
|
||||||
|
fmt.Println("Next run:", job.NextRun())
|
||||||
schedule.StartBlocking()
|
schedule.StartBlocking()
|
||||||
case config.RunOnceMode:
|
case config.RunOnceMode:
|
||||||
logger.Info("Running in run once mode")
|
numberOfChanges, err := rdd.RunOnce()
|
||||||
|
|
||||||
_, err := rdd.RunOnce()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to run once", slog.String("error", err.Error()))
|
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
fmt.Println("Number of changes:", numberOfChanges)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,29 +3,18 @@ package ionosDnsProvider
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
"realdnydns/model/common"
|
"realdnydns/model/common"
|
||||||
"realdnydns/pkg/dnsProvider"
|
"realdnydns/pkg/dnsProvider"
|
||||||
ionosAPI "realdnydns/pkg/dnsProvider/ionos/api"
|
ionosAPI "realdnydns/pkg/dnsProvider/ionos/api"
|
||||||
|
|
||||||
redis "github.com/redis/go-redis/v9"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type IONOS struct {
|
type IONOS struct {
|
||||||
API ionosAPI.IonosAPI
|
API ionosAPI.IonosAPI
|
||||||
Redis *redis.Client
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type IONOSConfig struct {
|
type IONOSConfig struct {
|
||||||
APIKey string `yaml:"api_key"`
|
APIKey string `yaml:"api_key"`
|
||||||
BaseURL string `yaml:"base_url"`
|
BaseURL string `yaml:"base_url"`
|
||||||
Redis *IONOSRedisConfig
|
|
||||||
}
|
|
||||||
|
|
||||||
type IONOSRedisConfig struct {
|
|
||||||
Address string `yaml:"host"`
|
|
||||||
Password string `yaml:"password"`
|
|
||||||
DB int `yaml:"db"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIonos(config *IONOSConfig) (dnsProvider.DNSProvider, error) {
|
func NewIonos(config *IONOSConfig) (dnsProvider.DNSProvider, error) {
|
||||||
@@ -37,18 +26,8 @@ func NewIonos(config *IONOSConfig) (dnsProvider.DNSProvider, error) {
|
|||||||
return nil, errors.New("base_url is required")
|
return nil, errors.New("base_url is required")
|
||||||
}
|
}
|
||||||
|
|
||||||
var redisClient *redis.Client = nil
|
|
||||||
if config.Redis != nil {
|
|
||||||
redisClient = redis.NewClient(&redis.Options{
|
|
||||||
Addr: config.Redis.Address,
|
|
||||||
Password: config.Redis.Password,
|
|
||||||
DB: config.Redis.DB,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return &IONOS{
|
return &IONOS{
|
||||||
ionosAPI.New(config.APIKey, config.BaseURL),
|
ionosAPI.New(config.APIKey, config.BaseURL),
|
||||||
redisClient,
|
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,8 +2,6 @@ package realDynDns
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"log/slog"
|
|
||||||
"sync"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"realdnydns/pkg/config"
|
"realdnydns/pkg/config"
|
||||||
@@ -19,7 +17,6 @@ type ChangeDetector struct {
|
|||||||
dnsProvider dnsProvider.DNSProvider
|
dnsProvider dnsProvider.DNSProvider
|
||||||
notificationProvider notificationProvider.NotificationProvider
|
notificationProvider notificationProvider.NotificationProvider
|
||||||
domains []config.DomainConfig
|
domains []config.DomainConfig
|
||||||
logger *slog.Logger
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(
|
func New(
|
||||||
@@ -27,14 +24,12 @@ func New(
|
|||||||
dnsProvider dnsProvider.DNSProvider,
|
dnsProvider dnsProvider.DNSProvider,
|
||||||
notificationProvider notificationProvider.NotificationProvider,
|
notificationProvider notificationProvider.NotificationProvider,
|
||||||
domains []config.DomainConfig,
|
domains []config.DomainConfig,
|
||||||
logger *slog.Logger,
|
|
||||||
) ChangeDetector {
|
) ChangeDetector {
|
||||||
return ChangeDetector{
|
return ChangeDetector{
|
||||||
externalIpProvider: externalIpProvider,
|
externalIpProvider: externalIpProvider,
|
||||||
dnsProvider: dnsProvider,
|
dnsProvider: dnsProvider,
|
||||||
notificationProvider: notificationProvider,
|
notificationProvider: notificationProvider,
|
||||||
domains: domains,
|
domains: domains,
|
||||||
logger: logger,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,90 +58,37 @@ func (c *ChangeDetector) RunOnce() (int, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *ChangeDetector) detectAndApplyChanges() (int, error) {
|
func (c *ChangeDetector) detectAndApplyChanges() (int, error) {
|
||||||
c.logger.Info("Detecting and applying changes")
|
|
||||||
|
|
||||||
externalIp, err := c.externalIpProvider.GetExternalIp()
|
externalIp, err := c.externalIpProvider.GetExternalIp()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.logger.Error("Failed to retrieve external IP", slog.String("error", err.Error()))
|
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var wg sync.WaitGroup
|
var numberUpdated int
|
||||||
|
|
||||||
numberUpdatedChannel := make(chan int)
|
|
||||||
|
|
||||||
for _, domain := range c.domains {
|
for _, domain := range c.domains {
|
||||||
for _, subdomain := range domain.Subdomains {
|
for _, subdomain := range domain.Subdomains {
|
||||||
wg.Add(1)
|
currentRecord, err := c.dnsProvider.GetRecord(domain.TLD, subdomain)
|
||||||
|
if err != nil {
|
||||||
|
return numberUpdated, err
|
||||||
|
}
|
||||||
|
|
||||||
go func(domain config.DomainConfig, subdomain string) {
|
if currentRecord.IP != externalIp.String() {
|
||||||
defer wg.Done()
|
err = c.notificationProvider.SendNotification(
|
||||||
|
fmt.Sprintf("Update %s.%s", subdomain, domain.TLD),
|
||||||
c.logger.Info("Checking record",
|
fmt.Sprintf("The IP of %s has changed from %s to %s", domain.TLD, currentRecord.IP, externalIp.String()),
|
||||||
slog.String("tld", domain.TLD),
|
|
||||||
slog.String("subdomain", subdomain),
|
|
||||||
)
|
)
|
||||||
currentRecord, err := c.dnsProvider.GetRecord(domain.TLD, subdomain)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.logger.Error("Failed to retrieve record",
|
return numberUpdated, err
|
||||||
slog.String("error", err.Error()),
|
|
||||||
slog.String("tld", domain.TLD),
|
|
||||||
slog.String("subdomain", subdomain),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if currentRecord.IP != externalIp.String() {
|
_, err = c.dnsProvider.UpdateRecord(domain.TLD, subdomain, externalIp, currentRecord.TTL, currentRecord.Prio, currentRecord.Disabled)
|
||||||
c.logger.Info("Record has changed",
|
numberUpdated++
|
||||||
slog.String("tld", domain.TLD),
|
if err != nil {
|
||||||
slog.String("subdomain", subdomain),
|
return numberUpdated, err
|
||||||
slog.String("current_ip", currentRecord.IP),
|
|
||||||
slog.String("external_ip", externalIp.String()),
|
|
||||||
)
|
|
||||||
|
|
||||||
err = c.notificationProvider.SendNotification(
|
|
||||||
fmt.Sprintf("Update %s.%s", subdomain, domain.TLD),
|
|
||||||
fmt.Sprintf("The IP of %s has changed from %s to %s", domain.TLD, currentRecord.IP, externalIp.String()),
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
c.logger.Warn("Failed to send notification",
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
c.logger.Info("Updating record",
|
|
||||||
slog.String("tld", domain.TLD),
|
|
||||||
slog.String("subdomain", subdomain),
|
|
||||||
slog.String("current_ip", currentRecord.IP),
|
|
||||||
slog.String("external_ip", externalIp.String()),
|
|
||||||
)
|
|
||||||
_, err = c.dnsProvider.UpdateRecord(domain.TLD, subdomain, externalIp, currentRecord.TTL, currentRecord.Prio, currentRecord.Disabled)
|
|
||||||
if err != nil {
|
|
||||||
c.logger.Error("Failed to update record",
|
|
||||||
slog.String("error", err.Error()),
|
|
||||||
slog.String("tld", domain.TLD),
|
|
||||||
slog.String("subdomain", subdomain),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
numberUpdatedChannel <- 1
|
|
||||||
}
|
}
|
||||||
}(domain, subdomain)
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
go func() {
|
|
||||||
wg.Wait()
|
|
||||||
close(numberUpdatedChannel)
|
|
||||||
}()
|
|
||||||
|
|
||||||
numberUpdated := 0
|
|
||||||
for v := range numberUpdatedChannel {
|
|
||||||
numberUpdated += v
|
|
||||||
}
|
|
||||||
|
|
||||||
c.logger.Info("Run completed", slog.Int("number_of_changes", numberUpdated))
|
|
||||||
return numberUpdated, nil
|
return numberUpdated, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
package realDynDns
|
package realDynDns
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"log/slog"
|
|
||||||
"net"
|
"net"
|
||||||
"realdnydns/model/common"
|
"realdnydns/model/common"
|
||||||
"realdnydns/pkg/config"
|
"realdnydns/pkg/config"
|
||||||
@@ -73,9 +72,7 @@ func testDetectAndApplyChangesWithChanges() func(t *testing.T) {
|
|||||||
"@",
|
"@",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
})
|
||||||
slog.Default(),
|
|
||||||
)
|
|
||||||
|
|
||||||
numberUpdated, err := changeDetector.RunOnce()
|
numberUpdated, err := changeDetector.RunOnce()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -104,9 +101,7 @@ func testDetectAndApplyChangesWithoutChanges() func(t *testing.T) {
|
|||||||
"@",
|
"@",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
})
|
||||||
slog.Default(),
|
|
||||||
)
|
|
||||||
|
|
||||||
numberUpdated, err := changeDetector.RunOnce()
|
numberUpdated, err := changeDetector.RunOnce()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
|
|
||||||
"packageRules": [
|
|
||||||
{
|
|
||||||
"matchPackageNames": [
|
|
||||||
"golang",
|
|
||||||
"go"
|
|
||||||
],
|
|
||||||
"groupName": "go version"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user