2014-08-09 03:09:54 +08:00
#!/bin/sh
2017-04-04 22:28:18 +08:00
. /lib/functions.sh
2020-08-13 08:38:56 +08:00
. /lib/functions/network.sh
2017-11-09 16:57:58 +08:00
. /lib/mwan3/common.sh
2017-04-04 22:28:18 +08:00
2017-04-04 22:00:06 +08:00
INTERFACE=""
2017-04-06 21:41:05 +08:00
DEVICE=""
IFDOWN_EVENT=0
2020-07-27 05:21:50 +08:00
IFUP_EVENT=0
2020-08-13 08:38:56 +08:00
TRACK_OUTPUT=$MWAN3TRACK_STATUS_DIR/$INTERFACE/TRACK_OUTPUT
mwan3_init
stop_subprocs() {
[ -n "$SLEEP_PID" ] && kill "$SLEEP_PID" && unset SLEEP_PID
[ -n "$TRACK_PID" ] && kill "$TRACK_PID" && unset TRACK_PID
}
2017-04-04 22:00:06 +08:00
2020-10-12 06:37:25 +08:00
WRAP() {
# shellcheck disable=SC2048
FAMILY=$FAMILY DEVICE=$DEVICE SRCIP=$SRC_IP FWMARK=$MMX_DEFAULT LD_PRELOAD=/lib/mwan3/libwrap_mwan3_sockopt.so.1.0 $*
}
2017-04-04 22:00:06 +08:00
clean_up() {
2020-08-13 08:38:56 +08:00
LOG notice "Stopping mwan3track for interface \"${INTERFACE}\". Status was \"${STATUS}\""
stop_subprocs
2017-04-04 22:00:06 +08:00
exit 0
}
2017-04-04 21:33:37 +08:00
2017-04-06 21:41:05 +08:00
if_down() {
2020-07-27 05:21:50 +08:00
LOG info "Detect ifdown event on interface ${INTERFACE} (${DEVICE})"
2017-04-06 21:41:05 +08:00
IFDOWN_EVENT=1
2020-08-13 08:38:56 +08:00
stop_subprocs
2017-04-06 21:41:05 +08:00
}
2020-07-27 05:21:50 +08:00
if_up() {
LOG info "Detect ifup event on interface ${INTERFACE} (${DEVICE})"
2020-08-13 08:38:56 +08:00
IFDOWN_EVENT=0
2020-07-27 05:21:50 +08:00
IFUP_EVENT=1
2020-08-13 08:38:56 +08:00
STARTED=1
stop_subprocs
2020-07-27 05:21:50 +08:00
}
2017-07-26 01:34:40 +08:00
validate_track_method() {
case "$1" in
ping)
2020-10-12 06:43:56 +08:00
if [ -x "/usr/bin/ping" ] && [ "$(/usr/bin/ping -V | grep -o '[0-9]*$')" -gt 20150519 ]; then
# -4 option added in iputils c3e68ac6
PING="/usr/bin/ping -${FAMILY#ipv}"
elif [ "$FAMILY" = "ipv6" ] && [ -x "/usr/bin/ping6" ]; then
PING="/usr/bin/ping6"
elif [ "$FAMILY" = "ipv4" ] && [ -x "/usr/bin/ping" ]; then
PING="/usr/bin/ping"
elif [ -x "/bin/ping" ]; then
PING="/bin/ping -${FAMILY#ipv}"
else
LOG warn "Missing ping. Please enable BUSYBOX_DEFAULT_PING and recompile busybox or install iputils-ping package."
2017-11-02 09:44:33 +08:00
return 1
2020-10-12 06:43:56 +08:00
fi
2017-07-26 01:34:40 +08:00
;;
arping)
2020-08-10 07:19:25 +08:00
command -v arping 1>/dev/null 2>&1 || {
2020-07-27 05:21:50 +08:00
LOG warn "Missing arping. Please install iputils-arping package."
2017-07-26 01:34:40 +08:00
return 1
}
;;
httping)
2020-08-10 07:19:25 +08:00
command -v httping 1>/dev/null 2>&1 || {
2020-07-27 05:21:50 +08:00
LOG warn "Missing httping. Please install httping package."
2017-07-26 01:34:40 +08:00
return 1
}
;;
2018-12-04 21:47:26 +08:00
nping-*)
2020-08-10 07:19:25 +08:00
command -v nping 1>/dev/null 2>&1 || {
2020-07-27 05:21:50 +08:00
LOG warn "Missing nping. Please install nping package."
2018-12-04 21:47:26 +08:00
return 1
}
;;
2017-07-26 01:34:40 +08:00
*)
2020-07-27 05:21:50 +08:00
LOG warn "Unsupported tracking method: $track_method"
2017-07-26 01:34:40 +08:00
return 2
;;
esac
}
2020-10-12 06:37:25 +08:00
validate_wrap() {
[ -x /lib/mwan3/libwrap_mwan3_sockopt.so.1.0 ] && return
LOG error "Missing libwrap_mwan3_sockopt. Please reinstall mwan3." &&
exit 1
}
2020-07-27 05:21:50 +08:00
disconnected() {
2020-08-13 08:38:56 +08:00
STATUS='offline'
echo "offline" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/STATUS
get_uptime > $MWAN3TRACK_STATUS_DIR/$INTERFACE/OFFLINE
echo "0" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/ONLINE
2020-07-27 05:21:50 +08:00
score=0
2020-09-01 07:15:09 +08:00
[ "$1" = 1 ] && return
2020-07-27 05:21:50 +08:00
LOG notice "Interface $INTERFACE ($DEVICE) is offline"
env -i ACTION="disconnected" INTERFACE="$INTERFACE" DEVICE="$DEVICE" /sbin/hotplug-call iface
}
connected() {
2020-08-13 08:38:56 +08:00
STATUS='online'
2020-09-01 03:49:17 +08:00
echo "online" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/STATUS
echo "0" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/OFFLINE
get_uptime > $MWAN3TRACK_STATUS_DIR/$INTERFACE/ONLINE
2020-07-27 05:21:50 +08:00
host_up_count=0
lost=0
turn=0
loss=0
LOG notice "Interface $INTERFACE ($DEVICE) is online"
2020-08-13 08:38:56 +08:00
env -i FIRSTCONNECT=$1 ACTION="connected" INTERFACE="$INTERFACE" DEVICE="$DEVICE" /sbin/hotplug-call iface
}
disabled() {
STATUS='disabled'
echo "disabled" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/STATUS
STARTED=0
2020-07-27 05:21:50 +08:00
}
firstconnect() {
2020-08-13 08:38:56 +08:00
local true_iface
network_flush_cache
mwan3_get_true_iface true_iface $INTERFACE
network_get_device DEVICE $true_iface
if [ "$STATUS" != "online" ]; then
config_get STATUS $INTERFACE initial_state "online"
fi
if ! network_is_up $true_iface || [ -z "$DEVICE" ]; then
disabled
return
fi
2020-08-26 06:16:07 +08:00
mwan3_get_src_ip SRC_IP $INTERFACE
2020-08-13 08:38:56 +08:00
LOG debug "firstconnect: called on $INTERFACE/$true_iface ($DEVICE). Status is $STATUS. SRC_IP is $SRC_IP"
STARTED=1
2020-07-27 05:21:50 +08:00
if [ "$STATUS" = "offline" ]; then
disconnected 1
else
connected 1
fi
}
update_status() {
2020-09-01 07:15:09 +08:00
local track_ip=$1
2020-07-27 05:21:50 +08:00
2020-09-01 03:49:17 +08:00
echo "$2" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/TRACK_${track_ip}
2020-07-27 05:21:50 +08:00
[ -z "$3" ] && return
2020-09-01 03:49:17 +08:00
echo "$3" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/LATENCY_${track_ip}
echo "$4" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/LOSS_${track_ip}
2020-07-27 05:21:50 +08:00
}
2017-04-04 21:01:34 +08:00
main() {
2017-04-04 22:28:18 +08:00
local reliability count timeout interval failure_interval
local recovery_interval down up size
2018-05-11 15:00:57 +08:00
local keep_failure_interval check_quality failure_latency
local recovery_latency failure_loss recovery_loss
2020-10-12 06:37:25 +08:00
local max_ttl httping_ssl track_ips
2014-08-09 03:09:54 +08:00
2017-04-04 22:00:06 +08:00
INTERFACE=$1
2020-08-13 08:38:56 +08:00
STATUS=""
STARTED=0
mkdir -p $MWAN3TRACK_STATUS_DIR/$INTERFACE
2018-01-31 18:58:38 +08:00
trap clean_up TERM
trap if_down USR1
2020-07-27 05:21:50 +08:00
trap if_up USR2
2017-04-04 21:01:34 +08:00
2017-04-04 22:28:18 +08:00
config_load mwan3
2020-10-12 06:37:25 +08:00
config_get FAMILY $INTERFACE family ipv4
2020-07-27 05:21:50 +08:00
config_get track_method $INTERFACE track_method ping
config_get_bool httping_ssl $INTERFACE httping_ssl 0
2020-10-12 06:37:25 +08:00
validate_track_method $track_method || {
2017-07-26 01:34:40 +08:00
track_method=ping
2017-11-02 09:44:33 +08:00
if validate_track_method $track_method; then
2020-07-27 05:21:50 +08:00
LOG warn "Using ping to track interface $INTERFACE avaliability"
2017-11-02 09:44:33 +08:00
else
2020-07-27 05:21:50 +08:00
LOG err "No track method avaliable"
2017-11-02 09:44:33 +08:00
exit 1
fi
2017-07-26 01:34:40 +08:00
}
2020-07-27 05:21:50 +08:00
config_get reliability $INTERFACE reliability 1
config_get count $INTERFACE count 1
config_get timeout $INTERFACE timeout 4
config_get interval $INTERFACE interval 10
config_get down $INTERFACE down 5
config_get up $INTERFACE up 5
config_get size $INTERFACE size 56
config_get max_ttl $INTERFACE max_ttl 60
config_get failure_interval $INTERFACE failure_interval $interval
config_get_bool keep_failure_interval $INTERFACE keep_failure_interval 0
config_get recovery_interval $INTERFACE recovery_interval $interval
config_get_bool check_quality $INTERFACE check_quality 0
config_get failure_latency $INTERFACE failure_latency 1000
config_get recovery_latency $INTERFACE recovery_latency 500
config_get failure_loss $INTERFACE failure_loss 40
config_get recovery_loss $INTERFACE recovery_loss 10
2020-08-13 08:38:56 +08:00
local sleep_time result ping_status loss latency
mwan3_list_track_ips()
{
track_ips="$track_ips $1"
}
config_list_foreach "$1" track_ip mwan3_list_track_ips
2017-04-04 22:28:18 +08:00
2020-09-01 07:15:09 +08:00
local score=$((down+up))
2017-04-04 21:01:34 +08:00
local host_up_count=0
local lost=0
2017-04-06 21:17:17 +08:00
local turn=0
2017-04-04 21:01:34 +08:00
2020-07-27 05:21:50 +08:00
firstconnect
2017-04-04 21:01:34 +08:00
while true; do
2020-08-13 08:38:56 +08:00
[ $STARTED -eq 0 ] && { sleep $MAX_SLEEP & SLEEP_PID=$!; wait; }
unset SLEEP_PID
2017-04-04 22:28:18 +08:00
sleep_time=$interval
2017-04-04 21:01:34 +08:00
for track_ip in $track_ips; do
2017-08-25 17:42:32 +08:00
if [ $host_up_count -lt $reliability ]; then
case "$track_method" in
ping)
2018-05-11 14:30:54 +08:00
if [ $check_quality -eq 0 ]; then
2020-10-12 06:43:56 +08:00
WRAP $PING -c $count -W $timeout -s $size -t $max_ttl -q $track_ip &> /dev/null &
2020-08-13 08:38:56 +08:00
TRACK_PID=$!
wait $TRACK_PID
2018-05-11 14:30:54 +08:00
result=$?
else
2020-10-12 06:43:56 +08:00
WRAP $PING -c $count -W $timeout -s $size -t $max_ttl -q $track_ip 2>/dev/null > $TRACK_OUTPUT &
2020-08-13 08:38:56 +08:00
TRACK_PID=$!
wait $TRACK_PID
2020-05-20 17:34:16 +08:00
ping_status=$?
2020-08-13 08:38:56 +08:00
loss=$(grep $TRACK_OUTPUT "packet loss" | cut -d "," -f3 | awk '{print $1}' | sed -e 's/%//')
2020-05-20 17:34:16 +08:00
if [ "$ping_status" -ne 0 ] || [ "$loss" -eq 100 ]; then
2018-05-11 14:30:54 +08:00
latency=999999
2020-05-20 17:34:16 +08:00
loss=100
2018-04-22 15:19:46 +08:00
else
2020-08-13 08:38:56 +08:00
latency="$(grep $TRACK_OUTPUT -E 'rtt|round-trip' | cut -d "=" -f2 | cut -d "/" -f2 | cut -d "." -f1)"
2018-04-22 15:19:46 +08:00
fi
2018-05-11 14:30:54 +08:00
fi
2018-04-22 15:19:46 +08:00
;;
2017-08-25 17:42:32 +08:00
arping)
2020-10-12 06:37:25 +08:00
WRAP arping -I $DEVICE -c $count -w $timeout -q $track_ip &> /dev/null &
2020-08-13 08:38:56 +08:00
TRACK_PID=$!
wait $TRACK_PID
2018-04-22 15:19:46 +08:00
result=$?
;;
2017-08-25 17:42:32 +08:00
httping)
2019-11-11 16:05:49 +08:00
if [ "$httping_ssl" -eq 1 ]; then
2020-10-12 06:37:25 +08:00
WRAP httping -c $count -t $timeout -q "https://$track_ip" &> /dev/null &
2019-11-11 16:05:49 +08:00
else
2020-10-12 06:37:25 +08:00
WRAP httping -c $count -t $timeout -q "http://$track_ip" &> /dev/null &
2019-11-11 16:05:49 +08:00
fi
2020-08-13 08:38:56 +08:00
TRACK_PID=$!
wait $TRACK_PID
2018-04-22 15:19:46 +08:00
result=$?
;;
2020-08-13 08:38:56 +08:00
nping-*)
2020-10-12 06:37:25 +08:00
WRAP nping -c $count $track_ip --${FAMILY#nping-} > $TRACK_OUTPUT &
2020-08-13 08:38:56 +08:00
TRACK_PID=$!
wait $TRACK_PID
result=$(grep $TRACK_OUTPUT Lost | awk '{print $12}')
2018-12-04 21:47:26 +08:00
;;
2017-08-25 17:42:32 +08:00
esac
2018-04-22 15:19:46 +08:00
if [ $check_quality -eq 0 ]; then
if [ $result -eq 0 ]; then
let host_up_count++
2020-07-27 05:21:50 +08:00
update_status "$track_ip" "up"
2018-04-22 15:19:46 +08:00
if [ $score -le $up ]; then
2020-07-27 05:21:50 +08:00
LOG info "Check ($track_method) success for target \"$track_ip\" on interface $INTERFACE ($DEVICE). Current score: $score"
2018-04-22 15:19:46 +08:00
fi
else
let lost++
2020-07-27 05:21:50 +08:00
update_status "$track_ip" "down"
2018-04-22 15:19:46 +08:00
if [ $score -gt $up ]; then
2020-07-27 05:21:50 +08:00
LOG info "Check ($track_method) failed for target \"$track_ip\" on interface $INTERFACE ($DEVICE). Current score: $score"
2018-04-22 15:19:46 +08:00
fi
2018-03-21 21:13:57 +08:00
fi
2017-08-25 17:42:32 +08:00
else
2020-09-01 07:15:09 +08:00
if [ "$loss" -ge "$failure_loss" ] || [ "$latency" -ge "$failure_latency" ]; then
2018-04-22 15:19:46 +08:00
let lost++
2020-07-27 05:21:50 +08:00
update_status "$track_ip" "down" $latency $loss
2018-04-22 15:19:46 +08:00
if [ $score -gt $up ]; then
2020-07-27 05:21:50 +08:00
LOG info "Check (${track_method}: latency=${latency}ms loss=${loss}%) failed for target \"$track_ip\" on interface $INTERFACE ($DEVICE). Current score: $score"
2018-04-22 15:19:46 +08:00
fi
2020-09-01 07:15:09 +08:00
elif [ "$loss" -le "$recovery_loss" ] && [ "$latency" -le "$recovery_latency" ]; then
2018-04-22 15:19:46 +08:00
let host_up_count++
2020-07-27 05:21:50 +08:00
update_status "$track_ip" "up" $latency $loss
2018-04-22 15:19:46 +08:00
if [ $score -le $up ]; then
2020-07-27 05:21:50 +08:00
LOG info "Check (${track_method}: latency=${latency}ms loss=${loss}%) success for target \"$track_ip\" on interface $INTERFACE ($DEVICE). Current score: $score"
2018-04-22 15:19:46 +08:00
fi
else
2020-08-13 08:38:56 +08:00
echo "skipped" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/TRACK_${track_ip}
2018-03-21 21:13:57 +08:00
fi
2017-08-25 17:42:32 +08:00
fi
2017-04-04 21:01:34 +08:00
else
2020-08-13 08:38:56 +08:00
echo "skipped" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/TRACK_${track_ip}
2017-04-04 21:01:34 +08:00
fi
done
2017-04-04 22:28:18 +08:00
if [ $host_up_count -lt $reliability ]; then
2017-04-04 21:01:34 +08:00
let score--
2017-04-04 22:28:18 +08:00
if [ $score -lt $up ]; then
2017-04-04 21:01:34 +08:00
score=0
2017-07-20 14:55:55 +08:00
[ ${keep_failure_interval} -eq 1 ] && {
sleep_time=$failure_interval
}
2017-04-04 21:01:34 +08:00
else
2017-04-04 22:28:18 +08:00
sleep_time=$failure_interval
2017-04-04 21:01:34 +08:00
fi
2017-04-04 22:28:18 +08:00
if [ $score -eq $up ]; then
2020-07-27 05:21:50 +08:00
disconnected
2017-04-04 21:01:34 +08:00
score=0
fi
2017-03-24 22:21:23 +08:00
else
2020-09-01 07:15:09 +08:00
if [ $score -lt $((down+up)) ] && [ $lost -gt 0 ]; then
LOG info "Lost $((lost*count)) ping(s) on interface $INTERFACE ($DEVICE). Current score: $score"
2017-04-04 21:01:34 +08:00
fi
let score++
lost=0
2017-04-04 22:28:18 +08:00
if [ $score -gt $up ]; then
2020-09-01 03:49:17 +08:00
echo "online" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/STATUS
2020-09-01 07:15:09 +08:00
score=$((down+up))
2017-04-04 22:28:18 +08:00
elif [ $score -le $up ]; then
sleep_time=$recovery_interval
2017-04-04 21:01:34 +08:00
fi
2017-04-04 22:28:18 +08:00
if [ $score -eq $up ]; then
2020-08-13 08:38:56 +08:00
connected
2017-04-04 21:01:34 +08:00
fi
2014-08-09 03:09:54 +08:00
fi
2017-04-06 21:17:17 +08:00
let turn++
2020-09-01 03:49:17 +08:00
mkdir -p "$MWAN3TRACK_STATUS_DIR/${1}"
echo "${lost}" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/LOST
echo "${score}" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/SCORE
echo "${turn}" > $MWAN3TRACK_STATUS_DIR/$INTERFACE/TURN
get_uptime > $MWAN3TRACK_STATUS_DIR/$INTERFACE/TIME
2017-04-06 21:17:17 +08:00
2017-04-04 21:01:34 +08:00
host_up_count=0
2017-04-06 22:36:46 +08:00
sleep "${sleep_time}" &
wait
2017-04-06 21:41:05 +08:00
if [ "${IFDOWN_EVENT}" -eq 1 ]; then
2020-07-27 05:21:50 +08:00
LOG debug "Register ifdown event on interface ${INTERFACE} (${DEVICE})"
2020-08-13 08:38:56 +08:00
disabled
disconnected
2017-04-06 21:41:05 +08:00
IFDOWN_EVENT=0
fi
2020-07-27 05:21:50 +08:00
if [ "${IFUP_EVENT}" -eq 1 ]; then
LOG debug "Register ifup event on interface ${INTERFACE} (${DEVICE})"
firstconnect
IFUP_EVENT=0
fi
2017-04-04 21:01:34 +08:00
done
}
2014-08-09 03:09:54 +08:00
2017-04-04 21:01:34 +08:00
main "$@"