A Simple Pibell# May 5, 2020
With all this free time I finally got around to installing a doorbell at home. I had no interest in Ring or the like: what I really wanted was a simple push doorbell that fit the (Victorian) house but would also somehow notify me if I was downstairs…
There are several documented projects on splicing in a Raspberry Pi into existing powered doorbell systems, but that wasn’t what I wanted either.
Instead, the doorbell is a simple contact switch feeding into the Pi’s GPIO pins. It’s effectively extremely simple but I didn’t find a step by step, so this is what I could have done with reading.
I bought the Pi, a case, a power supply, an SD card, and a USB speaker:
And the doorbell itself plus wiring:
I bought a pre-installed Raspbian SD card as I don’t have an SD card caddy. After some basic configuration (which required HDMI over to a monitor) I started playing with how to set up the Pi.
Of course the PI is absurdly over-powered for this purpose, but I wanted something simple to play with. And anyway, it’s running Pihole too.
The wiring itself is simple: bell wire over through a hole in the door frame to the back of the doorbell (which is a simple contact push). The other end of the wires are connected to the PI’s GPIO pin 18, and ground. The pin is pulled up and we trigger the event when we see a falling edge.
Actually connecting the wires was a bit fiddly: the bell wire is too thin for the 0.1” connector, and lacking a proper crimping tool I had to bodge it with needle-nose pliers. But once in the pins the housing connection is solid enough.
At first I tried to connect it to Alexa but soon gave up on that idea. There’s no way to “announce” via any API, and it kept disconnecting when used as a Bluetooth speaker. And Alexa has that infuriating “Now playing from…” thing you can’t turn off as well.
During fiddling with this I removed PulseAudio from the Pi as a dead loss.
Nor could I use an Anker Soundcore as a Bluetooth speaker: the stupid thing has some sleep mode that means it misses off the first 3 seconds or so of whatever’s playing.
Instead I have the USB speaker above. It’s not great but is enough to be heard from outside and inside.
Aside from playing whatever through the speaker, the bell notifies my desktop as well as sending an email. Here’s the somewhat crappy script it’s running:
#!/usr/bin/python3 -u # # Not going to win any awards this one, is it? # # The Pi is wired up such that pin 18 goes through the switch to ground. # The on-pin pull-up resistor is enabled (so .input() is normally True). # When the circuit completes, it goes to ground and hence we get a # falling edge and .input() becomes False. # # I get the occasional phantom still so we wait for settle_time before # thinking it's real. # from email.mime.text import MIMEText from subprocess import Popen, PIPE from datetime import datetime import RPi.GPIO as GPIO import subprocess import alsaaudio import threading import signal import wave import time import sys import os samplefile = sys.argv device='plughw:1,0' # in seconds settle_time = 0.1 bounce_time = 1 active = False def notify(): subprocess.run(['/home/pi/notify-sent']) msg = MIMEText('At %s' % datetime.now().strftime('%Y-%m-%d %H:%M:%S')) msg['From'] = 'doorbell <[email protected]ntarian.org>' msg['To'] = 'John Levon <[email protected]>' msg['Subject'] = 'Someone is ringing the doorbell' p = Popen(['/usr/sbin/sendmail', '-f', '[email protected]', '-t', '-oi'], stdin=PIPE) p.stdin.write(msg.as_string().encode()) p.stdin.close() def play(): global samplefile global active active = True count = 0 with wave.open(samplefile) as f: format = None # 8bit is unsigned in wav files if f.getsampwidth() == 1: format = alsaaudio.PCM_FORMAT_U8 # Otherwise we assume signed data, little endian elif f.getsampwidth() == 2: format = alsaaudio.PCM_FORMAT_S16_LE elif f.getsampwidth() == 3: format = alsaaudio.PCM_FORMAT_S24_3LE elif f.getsampwidth() == 4: format = alsaaudio.PCM_FORMAT_S32_LE else: raise ValueError('Unsupported format') rate = f.getframerate() periodsize = rate // 8 out = alsaaudio.PCM(alsaaudio.PCM_PLAYBACK, device=device) out.setchannels(f.getnchannels()) out.setrate(rate) out.setformat(format) out.setperiodsize(periodsize) # We always play at least one time round... while active or count < 1: data = f.readframes(periodsize) if data: out.write(data) else: print('looping after %d plays, active %s' % (count, active)) count += 1 f.rewind() print('pausing audio') out.pause() print('stopped after %d plays' % count) def wait(): global active while True: input_state = GPIO.input(18) if input_state: print('got input_state %s, active -> False' % input_state) active = False break time.sleep(0.2) def trigger(): print('triggering at %s' % time.time()) tn = threading.Thread(target=notify) tn.start() tp = threading.Thread(target=play) tp.start() tw = threading.Thread(target=wait) tw.start() tw.join() tp.join() tn.join() def settle(): global settle_time time.sleep(settle_time) input_state = GPIO.input(18) print('input state now %s' % input_state) return not input_state def falling_edge(channel): input_state = GPIO.input(18) print('got falling edge, input_state %s' % input_state) if settle(): trigger() with wave.open(samplefile) as f: # things go horrible if the rate isn't 48000 for some reason if f.getframerate() != 48000: raise ValueError('file must be 48000 rate') if f.getsampwidth() not in [ 1, 2, 3, 4]: raise ValueError('Unsupported format') GPIO.setmode(GPIO.BCM) GPIO.setup(18, GPIO.IN, pull_up_down=GPIO.PUD_UP) GPIO.add_event_detect(18, GPIO.FALLING, callback=falling_edge, bouncetime=(bounce_time * 1000)) print('started') signal.pause()