Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| f576dc6f1f | |||
| e7dcdba8aa | |||
| bd6c6cf4b2 | |||
|
1a8b065283
|
|||
| 7dacdccc01 | |||
|
8a34988dd4
|
|||
| 6e5004aa0e | |||
| 57df037137 | |||
|
315fd630e3
|
|||
|
1bc10bcb6e
|
|||
| 745075f38c |
@@ -46,7 +46,7 @@ steps:
|
||||
- bao kv get -mount secret -field GITHUB_COM_TOKEN renovate > /woodpecker/github_com_token
|
||||
- name: renovate
|
||||
# Renovate's built-in "woodpecker" manager tracks this image automatically.
|
||||
image: renovate/renovate:43.205.3
|
||||
image: renovate/renovate:43.207.4
|
||||
environment:
|
||||
# --- platform / target ---
|
||||
RENOVATE_PLATFORM: gitea
|
||||
|
||||
+30
-3
@@ -21,8 +21,8 @@
|
||||
# =============================================================================
|
||||
FROM --platform=$BUILDPLATFORM golang:1.26.3-alpine@sha256:91eda9776261207ea25fd06b5b7fed8d397dd2c0a283e77f2ab6e91bfa71079d AS builder
|
||||
|
||||
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale
|
||||
ARG TAILSCALE_VERSION=v1.98.3
|
||||
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale versioning=semver
|
||||
ARG TAILSCALE_VERSION=v1.98.5
|
||||
|
||||
# Provided automatically by buildx for the target platform.
|
||||
ARG TARGETOS
|
||||
@@ -69,6 +69,12 @@ WORKDIR /src/tailscale
|
||||
# trusted unix socket, so PermitRead/PermitWrite are
|
||||
# always false and EVERY CLI call (status, up, set, ...)
|
||||
# returns "access denied" (tailscale/tailscale#17873).
|
||||
# ipnbus — IPN bus watch. Without it, 'tailscale up' cannot wait
|
||||
# for completion: it fires config at the daemon and
|
||||
# returns immediately ("built with ts_omit_ipnbus; not
|
||||
# waiting for completion") WITHOUT printing the auth URL
|
||||
# or confirming success. Including it makes interactive
|
||||
# 'up' behave normally (blocks, prints login URL).
|
||||
#
|
||||
# Everything else remains omitted, including (rationale):
|
||||
# clientupdate — DELIBERATELY removed. The built-in updater would download
|
||||
@@ -111,6 +117,7 @@ RUN mkdir -p /out && \
|
||||
-e 's/ts_omit_health,\{0,1\}//g' \
|
||||
-e 's/ts_omit_iptables,\{0,1\}//g' \
|
||||
-e 's/ts_omit_unixsocketidentity,\{0,1\}//g' \
|
||||
-e 's/ts_omit_ipnbus,\{0,1\}//g' \
|
||||
-e 's/,$//' \
|
||||
) && \
|
||||
echo "Build tags: ${TAGS}" && \
|
||||
@@ -150,6 +157,24 @@ RUN mkdir -p /out/usrlocalbin && \
|
||||
ln -s /usr/local/bin/tailscale.combined /out/usrlocalbin/tailscale && \
|
||||
ln -s /usr/local/bin/tailscale.combined /out/usrlocalbin/tailscaled
|
||||
|
||||
# Entrypoint wrapper: enable IP forwarding inside the container's network
|
||||
# namespace, then exec tailscaled. tailscaled does NOT reliably enable IPv6
|
||||
# forwarding itself in a container netns ("IPv6 forwarding is disabled" warning),
|
||||
# which silently breaks advertised IPv6 subnet routes. The sysctls ARE writable
|
||||
# from inside a RouterOS container, so we set both here. Written in the builder
|
||||
# stage so it ships in the same single /usr/local/bin COPY layer (preserves the
|
||||
# overlayfs single-copy property). `exec` keeps tailscaled as PID 1.
|
||||
RUN printf '%s\n' \
|
||||
'#!/bin/sh' \
|
||||
'# Enable IPv4/IPv6 forwarding. Required for advertised subnet routes and' \
|
||||
'# exit-node functionality.' \
|
||||
'for f in /proc/sys/net/ipv4/ip_forward /proc/sys/net/ipv6/conf/all/forwarding; do' \
|
||||
' if [ -w "$f" ]; then echo 1 > "$f" 2>/dev/null || echo "warn: could not write $f"; fi' \
|
||||
'done' \
|
||||
'exec /usr/local/bin/tailscaled "$@"' \
|
||||
> /out/usrlocalbin/entrypoint.sh && \
|
||||
chmod +x /out/usrlocalbin/entrypoint.sh
|
||||
|
||||
# =============================================================================
|
||||
# Stage 2: Custom minimal busybox
|
||||
# =============================================================================
|
||||
@@ -267,7 +292,9 @@ ENV PATH=/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
|
||||
# -----------------------------------------------------------------------------
|
||||
VOLUME ["/var/lib/tailscale"]
|
||||
|
||||
ENTRYPOINT ["/usr/local/bin/tailscaled"]
|
||||
# entrypoint.sh enables IP forwarding (incl. IPv6) in the container netns, then
|
||||
# exec's tailscaled with the CMD flags below as its arguments.
|
||||
ENTRYPOINT ["/usr/local/bin/entrypoint.sh"]
|
||||
|
||||
# Default flags:
|
||||
# --no-logs-no-support disables logtail uploads (logtail binary code is
|
||||
|
||||
+16
-2
@@ -70,8 +70,21 @@ in a future release stays omitted until deliberately added to the Dockerfile.
|
||||
saves a real ~195 kB of flash (424 kB → 229 kB), not just transfer size.
|
||||
|
||||
The final image is built `FROM scratch` — there is no base distro layer.
|
||||
It contains only the busybox binary + applet symlinks, the CA bundle, and
|
||||
the Tailscale binary.
|
||||
It contains only the busybox binary + applet symlinks, the CA bundle, the
|
||||
Tailscale binary, and a tiny `entrypoint.sh`.
|
||||
|
||||
### Entrypoint: IP forwarding
|
||||
|
||||
`ENTRYPOINT` is a small `entrypoint.sh` that enables IPv4 and IPv6 forwarding
|
||||
(`net.ipv4.ip_forward`, `net.ipv6.conf.all.forwarding`) in the container's
|
||||
network namespace, then `exec`s `tailscaled` (so the daemon stays PID 1). This
|
||||
is necessary because `tailscaled` does **not** reliably enable IPv6 forwarding
|
||||
itself inside a container netns — it logs "IPv6 forwarding is disabled" and
|
||||
advertised IPv6 subnet routes silently fail. The sysctls are writable from
|
||||
inside a RouterOS container, so the entrypoint sets them directly; no
|
||||
host-side or `/container` configuration is required. The script is created in
|
||||
the builder stage so it ships in the same single `/usr/local/bin` `COPY` layer
|
||||
(preserving the [single-copy property](#avoiding-overlayfs-layer-duplication)).
|
||||
|
||||
### Avoiding overlayfs layer duplication
|
||||
|
||||
@@ -143,6 +156,7 @@ that's a separate build, not just a `--platform` change.
|
||||
| iptables | Linux iptables support for routing rules |
|
||||
| osrouter | Configure kernel network stack and routing tables |
|
||||
| unixsocketidentity | **Required** — without it the localapi denies every CLI call with "access denied" ([tailscale#17873](https://github.com/tailscale/tailscale/issues/17873)) |
|
||||
| ipnbus | Lets `tailscale up` wait for completion and print the login URL; without it `up` returns immediately without confirming success |
|
||||
|
||||
## Features intentionally omitted
|
||||
|
||||
|
||||
+2
-1
@@ -95,7 +95,8 @@ The daemon is now running but **not yet authenticated**.
|
||||
|
||||
### 5. Authenticate
|
||||
|
||||
> This image runs `tailscaled` directly and does **not** bundle Tailscale's
|
||||
> This image runs `tailscaled` via a tiny entrypoint (which enables IP
|
||||
forwarding, then `exec`s the daemon) and does **not** bundle Tailscale's
|
||||
`containerboot` wrapper, so the `TS_AUTHKEY` environment variable is **not**
|
||||
read automatically. You authenticate with `tailscale up --authkey=...` after the
|
||||
container starts.
|
||||
|
||||
+22
-5
@@ -7,6 +7,17 @@
|
||||
],
|
||||
"labels": ["dependencies"],
|
||||
"rebaseWhen": "behind-base-branch",
|
||||
"customManagers": [
|
||||
{
|
||||
"customType": "regex",
|
||||
"description": "Update version ARGs annotated with a `# renovate:` comment (the dockerfile manager only handles FROM/image lines, not ARG values).",
|
||||
"managerFilePatterns": ["/(^|/)Dockerfile$/"],
|
||||
"matchStrings": [
|
||||
"#\\s*renovate:\\s*datasource=(?<datasource>\\S+)\\s+depName=(?<depName>\\S+)(?:\\s+packageName=(?<packageName>\\S+))?(?:\\s+versioning=(?<versioning>\\S+))?\\s+ARG \\w+=(?<currentValue>\\S+)"
|
||||
],
|
||||
"matchStringsStrategy": "any"
|
||||
}
|
||||
],
|
||||
"packageRules": [
|
||||
{
|
||||
"matchManagers": ["dockerfile"],
|
||||
@@ -16,9 +27,8 @@
|
||||
{
|
||||
"matchDatasources": ["github-releases"],
|
||||
"matchPackageNames": ["tailscale/tailscale"],
|
||||
"description": "TAILSCALE_VERSION ARG: only stable releases. Tailscale uses EVEN minor versions for stable (v1.98.x); ODD minors (v1.99.x) are unstable, so filter to even minors and ignore pre-releases.",
|
||||
"extractVersion": "^v(?<version>\\d+\\.\\d+\\.\\d+)$",
|
||||
"allowedVersions": "/^\\d+\\.\\d*[02468]\\.\\d+$/",
|
||||
"description": "TAILSCALE_VERSION ARG: only stable releases. Tailscale uses EVEN minor versions for stable (v1.98.x); ODD minors (v1.99.x) are unstable, so filter to even minors and ignore pre-releases. The `v` prefix is kept (no extractVersion) so the ARG value stays v-prefixed to match the git tags cloned in the Dockerfile.",
|
||||
"allowedVersions": "/^v\\d+\\.\\d*[02468]\\.\\d+$/",
|
||||
"ignoreUnstable": true
|
||||
},
|
||||
{
|
||||
@@ -30,8 +40,15 @@
|
||||
},
|
||||
{
|
||||
"matchManagers": ["dockerfile"],
|
||||
"matchPackageNames": ["golang", "alpine", "busybox"],
|
||||
"description": "Automerge PATCH-only bumps of build components (Go/Alpine/busybox) once the PR build passes; review minor/major manually.",
|
||||
"matchPackageNames": ["golang", "alpine"],
|
||||
"description": "Automerge PATCH-only bumps of build components (Go/Alpine) once the PR build passes; review minor/major manually.",
|
||||
"matchUpdateTypes": ["patch"],
|
||||
"automerge": true
|
||||
},
|
||||
{
|
||||
"matchDatasources": ["docker"],
|
||||
"matchPackageNames": ["busybox"],
|
||||
"description": "busybox ARG (custom manager): automerge PATCH bumps once the PR build passes; review minor/major manually.",
|
||||
"matchUpdateTypes": ["patch"],
|
||||
"automerge": true
|
||||
},
|
||||
|
||||
Reference in New Issue
Block a user