#!/bin/sh

# Copyright 2015-2024 Rivoreo

# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to
# deal in the Software without restriction, including without limitation the
# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
# sell copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:

# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.

# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
# WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
# IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.


# May specify custom name servers to override resolv.conf(5)
NAME_SERVERS=
[ -f /etc/default/wireguard-peer-resolver ] && . /etc/default/wireguard-peer-resolver

is_inet4_address() {
	[ "${1#.}" = "$1" ] && [ "${1%.}" = "$1" ] || return
	local ORIG_IFS="$IFS"
	IFS=.
	set -- $1
	IFS="$ORIG_IFS"
	[ $# = 4 ] || return
	local i=1 v
	while [ $i -lt 5 ]; do
		eval "v=\"\${$i}\""
		[ "$v" -le 255 ] && [ "$v" -ge 0 ] || return
		[ "$v" -gt 0 ] && [ "${v#0}" != "$v" ] && return 1
		i=$((i+1))
	done
	[ $1 != 0 ] && return
	[ $2 = 0 ] || return
	[ $3 = 0 ] || return
	[ $4 = 0 ] || return
	true
}

is_inet6_address() {
	printf %s\\n "$1" | grep -Eq '^\[(([0-9a-f]{1,4}:){7}[0-9a-f]{1,4}|([0-9a-f]{1,4}:){1,6}:[0-9a-f]{1,4}|([0-9a-f]{1,4})?::([0-9a-f]{1,4})?)\]$'
}

parse_key_value() {
	case "$1" in
		*=*)
			key="${1%%=*}"
			key="${key%%[ 	]*}"
			value="${1#*=}"
			local trim_value
			while trim_value="${value#[ 	]}" && [ "$trim_value" != "$value" ]; do value="$trim_value"; done
			;;
		*\ *|*\	*)
			key="${1%%[ 	]*}"
			value="${1#*[ 	]}"
			;;
		*)
			false
			;;
	esac
}

unset LC_ALL LC_MESSAGES
unset LANGUAGE

set -f

while true; do
	sleep 60
	#interfaces="`wg show interfaces`" || continue
	#for i in $interfaces; do
	echo
	wg show all latest-handshakes
done | while read -r line; do
	if [ -z "$line" ]; then
		t="`date +%s`"
		continue
	fi
	set -- $line
	[ $# != 3 ] && continue
	interface="$1"
	publickey="$2"
	handshake="$3"
	[ $((t-handshake)) -lt 240 ] && continue
	[ -f "/etc/wireguard/$interface.conf" ] || continue
	section=
	found_peer=
	while read -r line; do case "$line" in
		\[[Pp][Ee][Ee][Rr]\]*)
			[ -n "$found_peer" ] && break
			section=peer
			name=
			port=
			;;
		\[*\]*)
			[ -n "$found_peer" ] && break
			section=
			;;
		?*=*)
			[ "$section" != peer ] && continue
			parse_key_value "$line" || continue
			case "$key" in
				[Pp][Uu][Bb][Ll][Ii][Cc][Kk][Ee][Yy])
					[ "$value" != "$publickey" ] && continue
					found_peer=1
					[ -n "$name" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_domain" ] && break
					;;
				[Ee][Nn][Dd][Pp][Oo][Ii][Nn][Tt])
					name="${value%:*}"
					port="${value##*:}"
					[ -n "$found_peer" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_domain" ] && [ -n "$loop_gre_key" ] && break
					;;
				\#[Gg][Rr][Ee][Tt][Uu][Nn][Nn][Ee][Ll][Ii][Nn][Tt][Ee][Rr][Ff][Aa][Cc][Ee])
					loop_gre_interface=1
					gre_tunnel_interface="$value"
					[ -n "$name" ] && [ -n "$found_peer" ] && [ -n "$loop_gre_domain" ] && [ -n "$loop_gre_key" ] && break
					;;
				\#[Gg][Rr][Ee][Tt][Uu][Nn][Nn][Ee][Ll][Dd][Oo][Mm][Aa][Ii][Nn])
					loop_gre_domain=1
					gre_tunnel_domain="$value"
					[ -n "$found_peer" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_key" ] && break
					;;
				\#[Gg][Rr][Ee][Tt][Uu][Nn][Nn][Ee][Ll][Kk][Ee][Yy])
					loop_gre_key=1
					gre_tunnel_key="$value"
					[ -n "$found_peer" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_interface" ] && [ -n "$loop_gre_domain" ] && break
					;;
			esac
			;;
	esac done < "/etc/wireguard/$interface.conf"
	if [ -n "$found_peer" ] && [ -n "$name" ] && ! is_inet4_address "$name" && ! is_inet6_address "$name" &&
		if [ -z "${name#\`*\`}" ]; then
			command="${name%\`}"
			command="${command#\`}"
			address="`sh -c \"$command\"`"
		elif address="`LANG=C host -4 -t A -- \"$name\" $NAME_SERVERS`" && [ "${address%%* has no A record}" = "$address" ]; then
			address="${address%%
*}"
			address="${address##* has address }"
		elif address="`LANG=C host -4 -t AAAA -- \"$name\" $NAME_SERVERS`" && [ "${address%%* has no AAAA record}" = "$address" ]; then
			address="${address%%
*}"
			address="${address##* has IPv6 address }"
		else
			false
		fi
	then
		wg set "$interface" peer "$publickey" endpoint "$address:$port"
	fi

	if [ -n "$gre_tunnel_interface" ] && [ -n "$gre_tunnel_domain" ] && ! is_inet4_address "$name" && ! is_inet6_address "$name" &&
		if address="`LANG=C host -4 -t A -- \"$gre_tunnel_domain\" $NAME_SERVERS`" && [ "${address%%* has no A record}" = "$address" ]; then
			address="${address%%
*}"
			address="${address##* has address }"
		elif address="`LANG=C host -4 -t AAAA -- \"$gre_tunnel_domain\" $NAME_SERVERS`" && [ "${address%%* has no AAAA record}" = "$address" ]; then
			address="${address%%
*}"
			address="${address##* has IPv6 address }"
		else
			false
		fi
	then
		if [ -n "$gre_tunnel_key" ]; then
			ip tunnel change $gre_tunnel_interface mode gre remote $address ttl inherit key $gre_tunnel_key
		else
			ip tunnel change $gre_tunnel_interface mode gre remote $address ttl inherit
		fi
	fi
done
