Обнаружение присутствия Raspberry Pi с использованием Wi-Fi


12

Как я могу заставить мой Raspberry Pi постоянно сканировать MAC-адрес в своей сети, а затем выполнять задачу при обнаружении определенного MAC-адреса?

Я использую свой Pi для запуска различных задач домашней автоматизации. Я бы хотел, чтобы он выполнил задание, когда обнаружит, что мой телефон подключен к Wi-Fi (то есть я только что вернулся домой). Каков наилучший способ непрерывного сканирования Pi на MAC-адрес моего телефона? Я знаю, что могу использовать arp-scanдля сканирования MAC-адресов, но мне нужно, чтобы это происходило постоянно. Я уверен, что есть лучший метод, чем arp-scanзапуск в цикле или использование cron для запуска каждую минуту.


2
Если вам не нужен ответ в режиме реального времени, лучше использовать cron. Сначала вы пишете скрипт, который возвращает что-то вроде true или false в sh (код состояния - лучший вариант для ответа), затем скрипт, который сохранит предыдущее значение где-нибудь в / tmp и проверит изменения. Если изменения произойдут, это зарегистрирует это.
Мизаз

Ответы:


3

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

3

Почему бы не добавить резервирование к своему IP-адресу телефона и просто выполнить пинг через задание cron каждые несколько минут.

    #!/bin/bash
    HOSTS="x.x.x.1 x.x.x.2"
    COUNT=10
    for myHost in $HOSTS
    do
      count=$(ping -c $COUNT $myHost | grep 'received' | awk -F',' '{ print $2 }' | awk '{ print $1 }')
      if [ $count -eq 10 ]; then
        # 100% response
        # let the cat out
      fi
    done

Я выполняю это на моем rpi, используя fhem , он имеет встроенный модуль обнаружения присутствия, который позволяет мне устанавливать резидентные статусы, которые можно запрашивать, чтобы влиять на задачи домашней автоматизации.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.