1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
|
import logging
import threading
from typing import Optional
from time import sleep
from ..util import stringify, send_datagram, Addr
from pyA20.gpio import gpio
from pyA20.gpio import port as gpioport
logger = logging.getLogger(__name__)
class SoundSensorNode:
def __init__(self,
name: str,
pinname: str,
server_addr: Optional[Addr],
delay=0.005):
if not hasattr(gpioport, pinname):
raise ValueError(f'invalid pin {pinname}')
self.pin = getattr(gpioport, pinname)
self.name = name
self.delay = delay
self.server_addr = server_addr
self.hits = 0
self.hitlock = threading.Lock()
self.interrupted = False
def run(self):
try:
t = threading.Thread(target=self.sensor_reader)
t.daemon = True
t.start()
while True:
with self.hitlock:
hits = self.hits
self.hits = 0
if hits > 0:
try:
if self.server_addr is not None:
send_datagram(stringify([self.name, hits]), self.server_addr)
else:
logger.debug(f'server reporting disabled, skipping reporting {hits} hits')
except OSError as exc:
logger.exception(exc)
sleep(1)
except (KeyboardInterrupt, SystemExit) as e:
self.interrupted = True
logger.info(str(e))
def sensor_reader(self):
gpio.init()
gpio.setcfg(self.pin, gpio.INPUT)
gpio.pullup(self.pin, gpio.PULLUP)
while not self.interrupted:
state = gpio.input(self.pin)
sleep(self.delay)
if not state:
with self.hitlock:
logger.debug('got a hit')
self.hits += 1
|