PhoneHome
Мой телефон работает по-разному, так как все телефоны имеют некоторые различия. Ответ на пинг происходит только в том случае, если телефон не спит. Если Pi перезагружен и телефон находится в спящем режиме, несколько пингов поместят его ip и mac адреса в таблицу arp со 100% потерей пакетов. Я только что узнал, что arp
команда устарела, ip neighbor
сейчас используется.
pi@RPi0:~ $ ip neighbor
169.254.65.43 dev eth0 lladdr 64:31:00:00:00:00 REACHABLE
192.168.0.1 dev wlan0 lladdr ac:b3:00:00:00:00 STALE
fe80::aeb3:13ff:fe00:000 dev wlan0 lladdr ac:b3:00:00:00:00 router STALE
pi@RPi0:~ $ ping 192.168.0.22
PING 192.168.0.22 (192.168.0.22) 56(84) bytes of data.
From 192.168.0.10 icmp_seq=1 Destination Host Unreachable
From 192.168.0.10 icmp_seq=2 Destination Host Unreachable
From 192.168.0.10 icmp_seq=3 Destination Host Unreachable
--- 192.168.0.22 ping statistics ---
34 packets transmitted, 0 received, +3 errors, 100% packet loss, time 34303ms
pi@RPi0:~ $ ip neighbor
192.168.0.1 dev wlan0 lladdr ac:b3:00:00:00:00 REACHABLE
169.254.65.43 dev eth0 lladdr 64:31:00:00:00:00 REACHABLE
192.168.0.22 dev wlan0 lladdr ac:37:00:00:00:00 REACHABLE
fe80::aeb3:13ff:fe00:000 dev wlan0 lladdr ac:b3:00:00:00:00 router STALE
После тестирования мое решение будет иметь две петли внутри петли навсегда. первый внутренний цикл должен был бы выполнить многократный пинг по диапазону IP-адресов, что было бы возможно для моего телефона. Мой маршрутизатор зарезервировал первые 19 IP-адресов, и у меня может быть около полудюжины адресов, которые DHCP назначит, включая мой телефон, начиная с адреса 192.168.0.20. Я буду пинговать дюжину IP-адресов один раз в фоновом режиме, подождать одну секунду ответа и выбросить результаты как ненужные. Я подожду восемь секунд в таблице arp и выполню ip neighbor
команду, grep mac-адрес для IP-адреса. Маршрутизатор и телефон сохранят один и тот же IP-адрес, если не произойдет что-то необычное. Таблица агр будет оставаться в Pi, но изменит состояния от REACHABLE
, STALE
и FAILED
от пингов и времени.
Второй внутренний цикл будет пинговать и проверять таблицу arp каждые пять минут, чтобы определить, находится ли телефон дома. С тремя пингами «СБОЙ» подряд телефона нет дома. Один «REACHABLE», когда телефона нет дома, заставит телефон вернуться домой (что-то сделать). Существуют проверки для проверки IP-адреса и возврата к первому внутреннему циклу, если требуются исправления.
#!/bin/bash
# A script to do something when Phone returns Home.
mac="ac:37:00:00:00:00" # Your phone mac address
ip_addr="" # Leave blank or ip for test
network="192.168.0.0" # Your network (Class C only)
range="20 32" # ip address possible range
pgm='echo "do something"' # program to exec when Phone returns Home
start=$(echo "$range" | cut -d " " -f1)
stop=$(echo "$range" | cut -d " " -f2)
network=$(echo "$network" | cut -d. -f1-3)
echo "Start $(date)"
while [ 1 ]; do
cnt=0
fail=0
[ "$ip_addr" ] || while [ ! "$ip_addr" ]; do
for x in $(seq "$start" "$stop"); do
(junk=$(ping -c1 -W1 "$network"."$x") & )
wait
done
sleep 8
ip_addr=$(ip neighbor | grep "$mac" | cut -d " " -f1)
((cnt++))
if (( $cnt > 15 )); then
cnt=0
echo "--- Phone not Home $(date)"
sleep 300 # 5 minutes
fi
if [ "$ip_addr" ]; then
echo "--- Phone is Home, Count = $cnt, Date = $(date)"
echo "Phone ip = $ip_addr mac = $mac"
fi
done
while [ "$ip_addr" ]; do
junk="$(ping -c1 -W1 $ip_addr)"
sleep 8
home_nw="$(ip neighbor | grep $ip_addr | cut -d ' ' -f 1,5,6)"
echo "$home_nw - $(date)"
is_home=$(echo "$home_nw" | cut -d " " -f3)
if [ "$is_home" == "REACHABLE" ] && (( "$fail" >= 3 )); then
echo "--- Phone returned Home - $(date)"
$pgm
fi
[ "$is_home" == "REACHABLE" ] && fail=0
mac_stat=$(echo "$home_nw" | cut -d " " -f2)
if [ "$mac_stat" == "FAILED" ]; then
(( "$fail" < 10 )) && ((fail++))
ip_test="$(ip neighbor | grep $mac | cut -d ' ' -f1)"
if [ "$ip_test" ]; then
[ "$ip_test" == "$ip_addr" ] || ip_addr=""
fi
if (( "$fail" == 3 )); then
echo "--- Phone not at Home $(date)"
fi
else
if [ "$mac_stat" != "$mac" ]; then
ip_addr=""
fi
fi
sleep 300 # 5 minutes
done
done