Compare commits
16 Commits
v1.98.5-mt.2
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 43f913cffc | |||
| 43698b733d | |||
|
ee5ca68fc3
|
|||
|
8a550f23d8
|
|||
| a34f30483b | |||
| 26debfaf30 | |||
| cae5aca3b3 | |||
| 16fd2170db | |||
| b7f3bdbbc6 | |||
| c2fee4d239 | |||
|
cb70afb345
|
|||
| 568f114c6e | |||
|
6ba07dd23b
|
|||
| 75b95fe4c4 | |||
| c8b5101416 | |||
| cba8447fa7 |
@@ -9,10 +9,24 @@
|
|||||||
# Reports pass/fail status back to Gitea, so it shows up as a required check on
|
# Reports pass/fail status back to Gitea, so it shows up as a required check on
|
||||||
# the PR.
|
# the PR.
|
||||||
|
|
||||||
|
# Changes that can't affect the image don't trigger the build: docs and the
|
||||||
|
# RouterOS-side script (routeros/**: lives on the router, not in the image).
|
||||||
|
# NOTE: if Gitea is ever configured to REQUIRE this check for merging, a
|
||||||
|
# PR touching only excluded files will have no check at all — exempt such PRs
|
||||||
|
# or merge manually. Renovate PRs always touch the Dockerfile or pipeline
|
||||||
|
# files, so the automerge gate is unaffected by these exclusions.
|
||||||
when:
|
when:
|
||||||
- event: pull_request
|
- event: pull_request
|
||||||
|
path:
|
||||||
|
exclude: &non_image_paths
|
||||||
|
- '**/*.md'
|
||||||
|
- 'docs/**'
|
||||||
|
- 'routeros/**'
|
||||||
|
- 'renovate.json'
|
||||||
- event: push
|
- event: push
|
||||||
branch: main
|
branch: main
|
||||||
|
path:
|
||||||
|
exclude: *non_image_paths
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Build all arches (no push)
|
- name: Build all arches (no push)
|
||||||
|
|||||||
@@ -13,9 +13,19 @@
|
|||||||
# unchanged, so no tag is created and nothing is released — they ride along
|
# unchanged, so no tag is created and nothing is released — they ride along
|
||||||
# with the next Tailscale bump or manual tag.
|
# with the next Tailscale bump or manual tag.
|
||||||
|
|
||||||
|
# Skipped for pushes that can't introduce a new Tailscale version:
|
||||||
|
# TAILSCALE_VERSION lives in the Dockerfile, so a push touching only docs or
|
||||||
|
# the RouterOS-side script can never produce a new version to tag (the job
|
||||||
|
# would just no-op after spinning up OpenBao + git containers).
|
||||||
when:
|
when:
|
||||||
- event: push
|
- event: push
|
||||||
branch: main
|
branch: main
|
||||||
|
path:
|
||||||
|
exclude:
|
||||||
|
- '**/*.md'
|
||||||
|
- 'docs/**'
|
||||||
|
- 'routeros/**'
|
||||||
|
- 'renovate.json'
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Get git token from OpenBao
|
- name: Get git token from OpenBao
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ steps:
|
|||||||
- bao kv get -mount secret -field GITHUB_COM_TOKEN renovate > /woodpecker/github_com_token
|
- bao kv get -mount secret -field GITHUB_COM_TOKEN renovate > /woodpecker/github_com_token
|
||||||
- name: renovate
|
- name: renovate
|
||||||
# Renovate's built-in "woodpecker" manager tracks this image automatically.
|
# Renovate's built-in "woodpecker" manager tracks this image automatically.
|
||||||
image: renovate/renovate:43.207.4
|
image: renovate/renovate:43.220.0
|
||||||
environment:
|
environment:
|
||||||
# --- platform / target ---
|
# --- platform / target ---
|
||||||
RENOVATE_PLATFORM: gitea
|
RENOVATE_PLATFORM: gitea
|
||||||
@@ -58,8 +58,12 @@ steps:
|
|||||||
# Use the committed renovate.json; don't open an onboarding PR.
|
# Use the committed renovate.json; don't open an onboarding PR.
|
||||||
RENOVATE_ONBOARDING: "false"
|
RENOVATE_ONBOARDING: "false"
|
||||||
RENOVATE_REQUIRE_CONFIG: "optional"
|
RENOVATE_REQUIRE_CONFIG: "optional"
|
||||||
# Git identity for the branches/commits Renovate creates.
|
# Git identity for the branches/commits Renovate creates. MUST match the
|
||||||
RENOVATE_GIT_AUTHOR: "Renovate Bot <renovate@localhost>"
|
# bot's Gitea account email: platform actions (automerge merge commits,
|
||||||
|
# "update branch") are attributed to the account email, and Renovate
|
||||||
|
# flags branches containing commits from unrecognized emails as
|
||||||
|
# "edited by someone else" and stops rebasing them.
|
||||||
|
RENOVATE_GIT_AUTHOR: "Renovate Bot <renovate@lumpiasty.xyz>"
|
||||||
# GitHub token (read-only, no repo access) lets Renovate fetch release
|
# GitHub token (read-only, no repo access) lets Renovate fetch release
|
||||||
# notes / changelogs and avoids GitHub API rate limits for the
|
# notes / changelogs and avoids GitHub API rate limits for the
|
||||||
# github-releases datasource (tailscale). Optional but recommended.
|
# github-releases datasource (tailscale). Optional but recommended.
|
||||||
|
|||||||
+3
-3
@@ -19,7 +19,7 @@
|
|||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Stage 1: Build Tailscale combined binary (cross-compiled, runs natively)
|
# Stage 1: Build Tailscale combined binary (cross-compiled, runs natively)
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
FROM --platform=$BUILDPLATFORM golang:1.26.4-alpine@sha256:a6a091eac01ceac4b97496fe2957a49b6cdd83365337d5f46f6f73710424e805 AS builder
|
FROM --platform=$BUILDPLATFORM golang:1.26.4-alpine@sha256:7a3e50096189ad57c9f9f865e7e4aa8585ed1585248513dc5cda498e2f41812c AS builder
|
||||||
|
|
||||||
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale versioning=semver
|
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale versioning=semver
|
||||||
ARG TAILSCALE_VERSION=v1.98.5
|
ARG TAILSCALE_VERSION=v1.98.5
|
||||||
@@ -213,10 +213,10 @@ RUN printf '%s\n' \
|
|||||||
# This stage runs on the TARGET platform (no --platform override): gcc then
|
# This stage runs on the TARGET platform (no --platform override): gcc then
|
||||||
# produces native target-arch binaries directly. Under buildx this is
|
# produces native target-arch binaries directly. Under buildx this is
|
||||||
# transparently emulated via binfmt/QEMU for non-native targets.
|
# transparently emulated via binfmt/QEMU for non-native targets.
|
||||||
FROM alpine:3.23.4@sha256:5b10f432ef3da1b8d4c7eb6c487f2f5a8f096bc91145e68878dd4a5019afde11 AS busybox
|
FROM alpine:3.24.0@sha256:a2d49ea686c2adfe3c992e47dc3b5e7fa6e6b5055609400dc2acaeb241c829f4 AS busybox
|
||||||
|
|
||||||
# renovate: datasource=docker depName=busybox versioning=docker
|
# renovate: datasource=docker depName=busybox versioning=docker
|
||||||
ARG BUSYBOX_VERSION=1.37.0
|
ARG BUSYBOX_VERSION=1.38.0
|
||||||
|
|
||||||
RUN apk add --no-cache build-base linux-headers wget bzip2 perl upx
|
RUN apk add --no-cache build-base linux-headers wget bzip2 perl upx
|
||||||
|
|
||||||
|
|||||||
@@ -294,6 +294,22 @@ Only the small, rarely-written state file touches flash; the socket dir is
|
|||||||
tmpfs. The netmap is held in memory only — see
|
tmpfs. The netmap is held in memory only — see
|
||||||
[Why netmap disk-caching is removed](#why-netmap-disk-caching-is-removed).
|
[Why netmap disk-caching is removed](#why-netmap-disk-caching-is-removed).
|
||||||
|
|
||||||
|
### What lives in the state dir
|
||||||
|
|
||||||
|
| File | Purpose | Write frequency |
|
||||||
|
|---|---|---|
|
||||||
|
| `tailscaled.state` | Node identity, auth keys, prefs | On auth / key rotation / prefs change |
|
||||||
|
| `derpmap.cached.json` | Cached DERP relay server list for **bootstrap DNS**: at cold start with broken/unavailable DNS, tailscaled asks DERP servers to resolve the control plane. The binary ships a static DERP list, but it goes stale; this cache keeps the current one. | Once at first auth, then **only when Tailscale's relay infrastructure changes** (a few times a year). `dnsfallback.UpdateCache` has a deep-equal guard and skips the write when the DERP map is unchanged — netmap churn never touches it. |
|
||||||
|
|
||||||
|
`derpmap.cached.json` is intentionally **kept** despite the flash-wear policy:
|
||||||
|
the policy targets *frequent* writes (netmap deltas, logs), not one-shot
|
||||||
|
caches. On a router this cache is genuinely useful — after a power outage the
|
||||||
|
device may boot with WAN up but upstream DNS broken, exactly the case where a
|
||||||
|
fresh DERP list lets the node reach the control plane anyway. With
|
||||||
|
`cachenetmap` omitted, this file and `tailscaled.state` are the only cold-start
|
||||||
|
resilience the node has. (There is no `ts_omit_*` tag for it; it is written
|
||||||
|
only because `--statedir` is set.)
|
||||||
|
|
||||||
## Flash wear protection
|
## Flash wear protection
|
||||||
|
|
||||||
Several measures are in place to avoid wearing out internal flash:
|
Several measures are in place to avoid wearing out internal flash:
|
||||||
|
|||||||
+6
-2
@@ -9,7 +9,7 @@ reasoning behind these choices, see [DESIGN.md](DESIGN.md).
|
|||||||
|
|
||||||
## Deploy on MikroTik (RouterOS)
|
## Deploy on MikroTik (RouterOS)
|
||||||
|
|
||||||
Verified on RouterOS 7.21.2 (arm64, CRS418). Commands are grouped into
|
Verified on RouterOS 7.23 (arm64, CRS418). Commands are grouped into
|
||||||
copy-paste blocks, defaults should fit most configurations.
|
copy-paste blocks, defaults should fit most configurations.
|
||||||
|
|
||||||
> Because the image has no built-in updater (the `clientupdate` feature is
|
> Because the image has no built-in updater (the `clientupdate` feature is
|
||||||
@@ -19,7 +19,9 @@ copy-paste blocks, defaults should fit most configurations.
|
|||||||
|
|
||||||
### 0. Prerequisites
|
### 0. Prerequisites
|
||||||
|
|
||||||
- RouterOS >7.13 with the **container** package installed.
|
- RouterOS >= 7.23 with the **container** package installed
|
||||||
|
(7.23 is needed for container `restart-policy`; the deploy itself works on
|
||||||
|
>= 7.13 if you drop the restart options).
|
||||||
- Container mode enabled ([documentation](https://manual.mikrotik.com/docs/System%20Information%20and%20Utilities/device-mode/#changing-mode-of-device-mode)):
|
- Container mode enabled ([documentation](https://manual.mikrotik.com/docs/System%20Information%20and%20Utilities/device-mode/#changing-mode-of-device-mode)):
|
||||||
|
|
||||||
```
|
```
|
||||||
@@ -80,6 +82,8 @@ just that directory:
|
|||||||
mountlists=tailscale_state \
|
mountlists=tailscale_state \
|
||||||
logging=yes \
|
logging=yes \
|
||||||
start-on-boot=yes \
|
start-on-boot=yes \
|
||||||
|
restart-policy=on-failure \
|
||||||
|
restart-interval=10s \
|
||||||
name=tailscale
|
name=tailscale
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -3,8 +3,9 @@
|
|||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Checks the Gitea registry for a new :stable image and, only if the published
|
# Checks the Gitea registry for a new :stable image and, only if the published
|
||||||
# image actually changed, recreates the container. Designed for RouterOS 7.x
|
# image actually changed, recreates the container. Designed for RouterOS 7.x
|
||||||
# (tested target: 7.21.2, arm64). Requires RouterOS >= 7.13 for the :deserialize
|
# (tested target: 7.23, arm64). Requires RouterOS >= 7.23 for the container
|
||||||
# command used to parse the registry token JSON.
|
# restart-policy properties (and >= 7.13 for the :deserialize command used to
|
||||||
|
# parse the registry token JSON).
|
||||||
#
|
#
|
||||||
# HOW IT DECIDES "something changed":
|
# HOW IT DECIDES "something changed":
|
||||||
# It fetches the manifest digest of the :stable tag from the registry and
|
# It fetches the manifest digest of the :stable tag from the registry and
|
||||||
@@ -59,6 +60,12 @@
|
|||||||
:local cInterface "veth-tailscale"
|
:local cInterface "veth-tailscale"
|
||||||
:local cLogging yes
|
:local cLogging yes
|
||||||
:local cStartOnBoot yes
|
:local cStartOnBoot yes
|
||||||
|
# Restart the container automatically if tailscaled crashes (tailscaled is
|
||||||
|
# PID 1; if it dies the container stops). on-failure restarts only on abnormal
|
||||||
|
# exit (a manual /container/stop stays stopped); 10s is a gentle backoff.
|
||||||
|
# Requires RouterOS >= 7.23.
|
||||||
|
:local cRestartPolicy "on-failure"
|
||||||
|
:local cRestartInterval "10s"
|
||||||
# ----------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------
|
||||||
|
|
||||||
:log info "$scriptName: checking for image updates"
|
:log info "$scriptName: checking for image updates"
|
||||||
@@ -174,6 +181,8 @@
|
|||||||
mountlists=$cMountList \
|
mountlists=$cMountList \
|
||||||
logging=$cLogging \
|
logging=$cLogging \
|
||||||
start-on-boot=$cStartOnBoot \
|
start-on-boot=$cStartOnBoot \
|
||||||
|
restart-policy=$cRestartPolicy \
|
||||||
|
restart-interval=$cRestartInterval \
|
||||||
name=$cName
|
name=$cName
|
||||||
} do={
|
} do={
|
||||||
:log error "$scriptName: container add failed: $e"
|
:log error "$scriptName: container add failed: $e"
|
||||||
|
|||||||
Reference in New Issue
Block a user