-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathdaemon.py
executable file
·103 lines (83 loc) · 3.31 KB
/
daemon.py
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
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#!/usr/bin/env python
import RPi.GPIO as GPIO
from time import sleep
import sys
import logging as log
from access import db
from access.models import User
class RFidReader(object):
GPIO_PIN_D0 = 17
GPIO_PIN_D1 = 22
GPIO_PIN_DOOR_RELEASE = 21
GPIO_PIN_SOLENOID = 23
def __init__(self):
# Use the Broadcom numbering scheme
GPIO.setmode(GPIO.BCM)
GPIO.setup(self.GPIO_PIN_D0, GPIO.IN, GPIO.PUD_OFF)
GPIO.setup(self.GPIO_PIN_D1, GPIO.IN, GPIO.PUD_OFF)
GPIO.setup(self.GPIO_PIN_DOOR_RELEASE, GPIO.IN, GPIO.PUD_UP)
GPIO.setup(self.GPIO_PIN_SOLENOID, GPIO.OUT)
# Set high so we lock on powerup
GPIO.output(self.GPIO_PIN_SOLENOID, True)
self.number_string = ""
self.awaiting_bit = True
self.open_door = False
def run(self, testfn):
# Now, here we go. The trick here is to detect the pulses and the gaps
# as we're polling the whole damned time.
while True:
d0, d1 = [GPIO.input(p) for p in
[self.GPIO_PIN_D0, self.GPIO_PIN_D1]]
# Check if the door release has been pressed...
door_release_pressed = not GPIO.input(self.GPIO_PIN_DOOR_RELEASE)
if door_release_pressed:
log.debug("Door release pressed")
self.open_door = True
# If we're not waiting for a bit (i.e. waiting for both lines to
# go low) and both lines go low, then mark us as ready to read a
# bit
if not self.awaiting_bit:
if not d0 and not d1:
self.awaiting_bit = True
continue
# If we get to here, it's assumed we're expecting a bit.
if d0 != d1: # Sure sign we have a bit...
if d1:
self.number_string = self.number_string + "1"
else:
self.number_string = self.number_string + "0"
self.awaiting_bit = False
if len(self.number_string) == 26:
# First and last bits are checksum bits, ignoring for now.
# TODO: use them to check that the number is valid
key_number = int(self.number_string[1:-1], 2)
log.info("Read tag: %d" % key_number)
if testfn(key_number):
log.info("Key accepted")
self.open_door = True
else:
log.info("Key not accepted")
self.number_string = ""
if self.open_door:
log.debug("Maglock open")
GPIO.output(self.GPIO_PIN_SOLENOID, False)
sleep(2)
log.debug("Maglock closed")
GPIO.output(self.GPIO_PIN_SOLENOID, True)
self.open_door = False
def validate_key(key_id):
user = User.query.filter_by(key_id=key_id).first()
if user:
return True
return False
if __name__ == "__main__":
log.basicConfig(filename='/var/log/door.log', level=log.DEBUG,
format='%(asctime)s %(message)s')
log.debug("Startup")
reader = RFidReader()
while True:
try:
reader.run(validate_key)
except Exception as e:
log.exception(e)
pass