2022-06-12 17:53:02 +02:00
|
|
|
#!/usr/bin/env sh
|
|
|
|
set -Eeo pipefail
|
|
|
|
|
|
|
|
if [ "${1:0:1}" = '-' ]; then
|
|
|
|
set -- zerotier-one "$@"
|
|
|
|
fi
|
|
|
|
|
2022-07-19 10:41:38 +02:00
|
|
|
# useful paths
|
|
|
|
CONFIG_DIR="/var/lib/zerotier-one"
|
2022-07-22 05:50:28 +02:00
|
|
|
NETWORKS_DIR="${CONFIG_DIR}/networks.d"
|
2022-07-19 10:41:38 +02:00
|
|
|
|
|
|
|
# set up network auto-join if (a) the networks directory does not exist
|
2022-07-22 05:50:28 +02:00
|
|
|
# and (b) the ZEROTIER_ONE_NETWORK_IDS environment variable is non-null.
|
|
|
|
if [ ! -d "${NETWORKS_DIR}" -a -n "${ZEROTIER_ONE_NETWORK_IDS}" ] ; then
|
|
|
|
echo "Assuming container first run."
|
|
|
|
mkdir -p "${NETWORKS_DIR}"
|
|
|
|
for NETWORK_ID in ${ZEROTIER_ONE_NETWORK_IDS} ; do
|
|
|
|
echo "Configuring auto-join of network ID: ${NETWORK_ID}"
|
|
|
|
touch "${NETWORKS_DIR}/${NETWORK_ID}.conf"
|
|
|
|
echo "You will need to authorize this host at:"
|
|
|
|
echo " https://my.zerotier.com/network/${NETWORK_ID}"
|
|
|
|
done
|
2022-07-19 10:41:38 +02:00
|
|
|
fi
|
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# make sure permissions are always as expected (self-repair)
|
2022-07-19 10:41:38 +02:00
|
|
|
PUID="${PUID:-"999"}"
|
|
|
|
PGID="${PGID:-"994"}"
|
2022-07-22 05:50:28 +02:00
|
|
|
if [ "$(id -u)" = '0' -a -d "${CONFIG_DIR}" ]; then
|
|
|
|
chown -Rc "${PUID}:${PGID}" "${CONFIG_DIR}"
|
2022-07-19 10:41:38 +02:00
|
|
|
fi
|
|
|
|
|
2022-07-27 13:58:39 +02:00
|
|
|
# is routing enabled?
|
|
|
|
if [ $(sysctl -n net.ipv4.ip_forward) -ne 1 ] ; then
|
|
|
|
|
|
|
|
# no! there is no point in setting up rules or termination handler
|
|
|
|
echo "$(date) - IPv4 forwarding not enabled - launching ZeroTier-One in non-routing mode"
|
|
|
|
|
|
|
|
# just exec the client (this script ends here)
|
|
|
|
exec "$@"
|
|
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
echo "$(date) - launching ZeroTier-One in routing mode"
|
|
|
|
|
2022-07-19 10:41:38 +02:00
|
|
|
# use an appropriate default for a local physical interface
|
2022-07-25 03:09:06 +02:00
|
|
|
# (using eth0 maintains backwards compatibility)
|
2022-07-19 10:41:38 +02:00
|
|
|
PHY_IFACES="${ZEROTIER_ONE_LOCAL_PHYS:-"eth0"}"
|
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# default to iptables (maintains backwards compatibility)
|
2022-07-19 10:41:38 +02:00
|
|
|
IPTABLES_CMD=iptables
|
2022-07-25 03:09:06 +02:00
|
|
|
# but support an override to use iptables-nft
|
2022-07-22 05:50:28 +02:00
|
|
|
[ "${ZEROTIER_ONE_USE_IPTABLES_NFT}" = "true" ] && IPTABLES_CMD=iptables-nft
|
2022-07-19 10:41:38 +02:00
|
|
|
|
2022-07-27 13:58:39 +02:00
|
|
|
# the default forwarding mode is inbound (backwards compatible)
|
|
|
|
GATEWAY_MODE="${ZEROTIER_ONE_GATEWAY_MODE:-"inbound"}"
|
|
|
|
|
2022-07-19 10:41:38 +02:00
|
|
|
# the wildcard for the local zerotier interface is
|
2022-06-12 17:53:02 +02:00
|
|
|
ZT_IFACE="zt+"
|
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# function to add and remove the requisite rules
|
|
|
|
# - $1 is either "A" (add) or "D" (delete)
|
2022-07-27 13:58:39 +02:00
|
|
|
# - $2 is comment
|
2022-07-22 05:50:28 +02:00
|
|
|
update_iptables() {
|
2022-07-27 13:58:39 +02:00
|
|
|
case "${GATEWAY_MODE}" in
|
|
|
|
"inbound" )
|
|
|
|
echo "$2 ${IPTABLES_CMD} rules for inbound traffic (ZeroTier to local interfaces ${PHY_IFACES})"
|
|
|
|
for PHY_IFACE in ${PHY_IFACES} ; do
|
|
|
|
${IPTABLES_CMD} -t nat -${1} POSTROUTING -o ${PHY_IFACE} -j MASQUERADE
|
2022-07-29 10:07:34 +02:00
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${PHY_IFACE} -o ${ZT_IFACE} -j DROP
|
2022-07-27 13:58:39 +02:00
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${PHY_IFACE} -o ${ZT_IFACE} -m state --state RELATED,ESTABLISHED -j ACCEPT
|
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${ZT_IFACE} -o ${PHY_IFACE} -j ACCEPT
|
|
|
|
done
|
|
|
|
;;
|
|
|
|
"outbound" )
|
|
|
|
echo "$2 ${IPTABLES_CMD} rules for outbound traffic (local interfaces ${PHY_IFACES} to ZeroTier)"
|
|
|
|
${IPTABLES_CMD} -t nat -${1} POSTROUTING -o ${ZT_IFACE} -j MASQUERADE
|
|
|
|
for PHY_IFACE in ${PHY_IFACES} ; do
|
2022-07-29 10:07:34 +02:00
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${ZT_IFACE} -o ${PHY_IFACE} -j DROP
|
2022-07-27 13:58:39 +02:00
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${ZT_IFACE} -o ${PHY_IFACE} -m state --state RELATED,ESTABLISHED -j ACCEPT
|
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${PHY_IFACE} -o ${ZT_IFACE} -j ACCEPT
|
|
|
|
done
|
|
|
|
;;
|
|
|
|
"both" )
|
|
|
|
echo "$2 ${IPTABLES_CMD} rules for bi-directional traffic (local interfaces ${PHY_IFACES} to/from ZeroTier)"
|
|
|
|
${IPTABLES_CMD} -t nat -${1} POSTROUTING -o ${ZT_IFACE} -j MASQUERADE
|
|
|
|
for PHY_IFACE in ${PHY_IFACES} ; do
|
|
|
|
${IPTABLES_CMD} -t nat -${1} POSTROUTING -o ${PHY_IFACE} -j MASQUERADE
|
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${ZT_IFACE} -o ${PHY_IFACE} -j ACCEPT
|
|
|
|
${IPTABLES_CMD} -${1} FORWARD -i ${PHY_IFACE} -o ${ZT_IFACE} -j ACCEPT
|
|
|
|
done
|
|
|
|
;;
|
|
|
|
* )
|
|
|
|
echo "Warning: ZEROTIER_ONE_GATEWAY_MODE=${GATEWAY_MODE} is not supported - ignored"
|
|
|
|
;;
|
|
|
|
esac
|
2022-07-22 05:50:28 +02:00
|
|
|
}
|
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# add rules to set up NAT-routing
|
2022-07-27 13:58:39 +02:00
|
|
|
update_iptables "A" "adding"
|
2022-07-22 05:50:28 +02:00
|
|
|
|
|
|
|
# define where the ZeroTier daemon will write its output (if any)
|
2022-07-25 03:09:06 +02:00
|
|
|
TAIL_PIPE=$(mktemp /tmp/zerotier-ipc-XXXXXX)
|
2022-07-22 05:50:28 +02:00
|
|
|
|
|
|
|
# start listening and echoing anything that appears there into this process
|
2022-07-25 03:09:06 +02:00
|
|
|
tail -f "${TAIL_PIPE}" &
|
2022-07-22 05:50:28 +02:00
|
|
|
|
|
|
|
# make a note of the process ID for tail
|
|
|
|
TAIL_PIPE_PID=${!}
|
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# start the ZeroTier daemon in detached state
|
|
|
|
nohup "$@" </dev/null >"${TAIL_PIPE}" 2>&1 &
|
2022-07-22 05:50:28 +02:00
|
|
|
|
|
|
|
# make a note of the process ID
|
|
|
|
ZEROTIER_DAEMON_PID=${!}
|
|
|
|
|
|
|
|
# report
|
2022-07-25 03:09:06 +02:00
|
|
|
echo "$(date) - ZeroTier daemon is running as process ${ZEROTIER_DAEMON_PID}"
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# function to handle cleanup
|
|
|
|
termination_handler() {
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
echo "$(date) - terminating ZeroTier-One"
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# remove rules
|
2022-07-27 13:58:39 +02:00
|
|
|
update_iptables "D" "removing"
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# relay the termination message to the daemon
|
|
|
|
if [ -d "/proc/${ZEROTIER_DAEMON_PID}" ] ; then
|
|
|
|
kill -TERM ${ZEROTIER_DAEMON_PID}
|
|
|
|
wait ${ZEROTIER_DAEMON_PID}
|
|
|
|
fi
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# tell the pipe listener to go away too
|
|
|
|
if [ -d "/proc/${TAIL_PIPE_PID}" ] ; then
|
|
|
|
kill -TERM ${TAIL_PIPE_PID}
|
|
|
|
wait ${TAIL_PIPE_PID}
|
|
|
|
fi
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# clean up the pipe file
|
|
|
|
rm "${TAIL_PIPE}"
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
# set up termination handler (usually catches TERM)
|
|
|
|
trap termination_handler INT TERM HUP
|
|
|
|
|
|
|
|
# suspend this script while the zerotier daemon is running
|
|
|
|
wait ${ZEROTIER_DAEMON_PID}
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# would not usually expect to arrive here inside a Docker container but
|
|
|
|
# it can happen if the user does a "sudo killall zerotier-one" rather
|
|
|
|
# that use Docker commands
|
|
|
|
echo "$(date) - the ZeroTier daemon has quit unexpectedly - cleaning up"
|
2022-07-22 05:50:28 +02:00
|
|
|
|
2022-07-25 03:09:06 +02:00
|
|
|
# run the termination handler
|
|
|
|
termination_handler
|