💾 Archived View for gem.arisamiga.rocks › post › maildetect.gmi captured on 2024-08-31 at 12:10:15. Gemini links have been rewritten to link to archived content

View Raw

More Information

⬅️ Previous capture (2023-04-26)

-=-=-=-=-=-=-

Making a mail detection system

MailBox [IMG]

So a couple of weeks ago I saw myself checking the mailbox frequently and I thought to myself: "Is there a more ~~lazy~~ easier way to do this?".

A couple of weeks ago I made a project called

Spying on the Sun using a Photoresistor

and I thought that I could use the same principle to detect if there is mail in the mailbox.

Think about it for a second.

We can constantly shine a LED on the photoresistor if there is mail in the mailbox, the photoresistor will be covered by the mail and the voltage will drop. If there is no mail, the photoresistor will be exposed to the LED and the voltage will rise.

So I am gonna use the same Raspberry PI Zero W and the same photoresistor that I used in the previous project.

So first we will need to also connect a LED to the Raspberry PI Zero W.

I Followed this amazing guide by

Raspberry Pi Projects

to connect the LED to the Raspberry PI Zero W.

LEDResistorAndWires [IMG]

This uses 2 wires for positive and negative and a 330-ohm resistor connected to the anode (Longer leg) side of the LED.

We can also use the same connections from the last project for the photoresistor.

Photoresistor Wiring [IMG]

For more information about the wiring and how the photoresistor works, check out the previous project

Spying on the Sun using a Photoresistor

So now we have the LED and the photoresistor connected to the Raspberry PI Zero W.

We can now write the code to detect if there is mail in the mailbox we will take the code from the previous project and modify it.

from flask import Flask
import RPi.GPIO as GPIO
import time
import threading
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
light_values = []
light_times = []
app = Flask(__name__)
@app.route('/')
def index():
    html = """
    <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
    <canvas id="myChart" width="400" height="400"></canvas>
    <script>
        const labels = """ + str(light_times) + """;
        const ctx = document.getElementById('myChart');
        new Chart(ctx, {
            type: 'line',
            data: {
            labels: labels,
            datasets: [{
                label: 'Light Values',
                data: """ + str(light_values) + """,
                fill: false,
                borderColor: 'rgb(75, 192, 192)',
                tension: 0.1
            }]
            },
            options: {
            scales: {
                y: {
                    beginAtZero: true,
                    reverse: true
                }
            }
            }
        });
        </script>
    """
    return html
def read_photoresistor():
    while True:
        GPIO.setup(resistorPin, GPIO.OUT) 
        GPIO.output(resistorPin, GPIO.LOW)
        time.sleep(0.1)
        GPIO.setup(resistorPin, GPIO.IN)
        currentTime = time.time()
        diff = 0
        while(GPIO.input(resistorPin) == GPIO.LOW):
            diff  = time.time() - currentTime
        light_values.append(diff * 1000) # Add the time it took to charge the capacitor to the list 
        light_times.append(time.strftime("%H:%M:%S")) # Add the current time to the list
        print(diff * 1000)
        time.sleep(1)
if __name__ == '__main__':
    thread = threading.Thread(target=read_photoresistor)
    thread.start()
    app.run()

We will not be using any graphs or threading in this project, so we can remove the code that is not needed.

from flask import Flask
import RPi.GPIO as GPIO
import time
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
light_values = []
light_times = []
app = Flask(__name__)
@app.route('/')
def index():
    html = """
        <head> 
            <title> Mail Detection </title>
        <head>
        <body>
            <h1> Mail Detection </h1>
        </body>
    """
    return html
def read_photoresistor():
    while True:
        GPIO.setup(resistorPin, GPIO.OUT) 
        GPIO.output(resistorPin, GPIO.LOW)
        time.sleep(0.1)
        GPIO.setup(resistorPin, GPIO.IN)
        currentTime = time.time()
        diff = 0
        while(GPIO.input(resistorPin) == GPIO.LOW):
            diff  = time.time() - currentTime
        light_values.append(diff * 1000) # Add the time it took to charge the capacitor to the list 
        light_times.append(time.strftime("%H:%M:%S")) # Add the current time to the list
        print(diff * 1000)
        time.sleep(1)
if __name__ == '__main__':
    app.run(host="", debug=True)

Now we will also not continuously check if there is mail in the mailbox, we will only check if there is mail when we visit the website. We can do that by calling the function **read_photoresistor** and returning the value of the photoresistor when asked.

from flask import Flask
import RPi.GPIO as GPIO
import time
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
light_values = []
light_times = []
app = Flask(__name__)
@app.route('/')
def index():
    html = """
        <head> 
            <title> Mail Detection </title>
        <head>
        <body>
            <h1> Mail Detection </h1>
        </body>
    """
    return html
def read_photoresistor():
    time.sleep(1)
    GPIO.setup(resistorPin, GPIO.OUT) 
    GPIO.output(resistorPin, GPIO.LOW)
    time.sleep(0.1)
    GPIO.setup(resistorPin, GPIO.IN)
    currentTime = time.time()
    diff = 0
    while(GPIO.input(resistorPin) == GPIO.LOW):
        diff  = time.time() - currentTime
    light_values.append(diff * 1000) # Add the time it took to charge the capacitor to the list 
    light_times.append(time.strftime("%H:%M:%S")) # Add the current time to the list
    return (diff * 1000)
if __name__ == '__main__':
    app.run(host="", debug=True)

We will also not be storing any values or times as they are not needed.

from flask import Flask
import RPi.GPIO as GPIO
import time
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
app = Flask(__name__)
@app.route('/')
def index():
    html = """
        <head> 
            <title> Mail Detection </title>
        <head>
        <body>
            <h1> Mail Detection </h1>
        </body>
    """
    return html
def read_photoresistor():
    time.sleep(1)
    GPIO.setup(resistorPin, GPIO.OUT) 
    GPIO.output(resistorPin, GPIO.LOW)
    time.sleep(0.1)
    GPIO.setup(resistorPin, GPIO.IN)
    currentTime = time.time()
    diff = 0
    while(GPIO.input(resistorPin) == GPIO.LOW):
        diff  = time.time() - currentTime
    light_values.append(diff * 1000) # Add the time it took to charge the capacitor to the list 
    light_times.append(time.strftime("%H:%M:%S")) # Add the current time to the list
    return (diff * 1000)
if __name__ == '__main__':
    app.run(host="", debug=True)

Now another trick I want to show before we do the code to check if there is mail is that if we want our website to be able to be seen by anything other than our computer we need to add the host IP to the `host=""` in `app.run`.

Instead of manually finding the ip and adding it to the code we can use the following code to find the IP of the raspberry pi.

import socket
hostname=socket.gethostname()
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.connect(("8.8.8.8", 80))
IPAddr = s.getsockname()[0]

This code will find the ip of the raspberry pi and store it in the variable IPAddr. It will also store the hostname of the raspberry pi in the variable hostname.

Now we can use this code to check if there is mail in the mailbox.

from flask import Flask
import RPi.GPIO as GPIO
import time
import socket
hostname=socket.gethostname()
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.connect(("8.8.8.8", 80))
IPAddr = s.getsockname()[0]
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
app = Flask(__name__)
@app.route('/')
def index():
    light = read_photoresistor()
    html = """
    <head>
        <title>Mail Detect</title>
        <link href='https://fonts.googleapis.com/css?family=Lexend' rel='stylesheet'>
    <style>
        body {
            font-family: Arial, Helvetica, sans-serif;
            background-color: #f1f1f1;
        }
        .box {
            width: 400px;
            height: 50px;
            background-color: #fff;
            border: 1px solid #000;
            border-radius: 10px;
            margin: 0 auto;
            margin-top: 100px;
            text-align: center;
            font-size: 20px;
            font-weight: bold;
            padding-top: 30px;
        }
        .title {
            font-size: 30px; 
            font-family: 'Lexend';  
        }
    </style>
    </head>
    <p>
    &nbsp;
    </p>
    <center>
        <div class="title">Mail Detecting System <br><br> Running on: <b>"""+ hostname + """</b></div>
        """
    if light > 100: # We check if the light value is greater than 100 which means something is blocking the light.
        html += '<div class="box" style="border: 10px solid #98e5e5">Mail Detected ✉️'
    else:
        html += '<div class="box" style="border: 10px solid gray"> No Mail ❌'
    html += """
    <pre> """ + str(light) + """ </pre>
        </div>
    </center>
    """
    return html
def read_photoresistor():
    time.sleep(1)
    GPIO.setup(resistorPin, GPIO.OUT) # Set the resistorPin to output mode so we can charge the capacitor
    GPIO.output(resistorPin, GPIO.LOW) # Set the resistorPin to low so we can charge the capacitor
    time.sleep(0.1)
    GPIO.setup(resistorPin, GPIO.IN) # Set the resistorPin to input mode so we can read the voltage
    currentTime = time.time() # Get the current time
    diff = 0
    while(GPIO.input(resistorPin) == GPIO.LOW): # While the voltage is low keep looping until the voltage is high
        diff  = time.time() - currentTime # Get the difference between the current time and the time when the voltage was low 
    return (diff * 1000) # Print the time it took to charge the capacitor in milliseconds(ms)
if __name__ == '__main__':
    app.run(host=IPAddr, debug=True)

Now we can go to the ip of the raspberry pi and see if there is mail in the mailbox.

mailbox website [IMG]

Now there is a problem with this code. It will not update if there is a mail or not unless you refresh the page.

To fix this we can make a sort of API using the `app.route()` where it will have the value of the photoresistor so we can continuously call that API without needing to refresh the page.

from flask import Flask
import RPi.GPIO as GPIO
import time
import socket
hostname=socket.gethostname()
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.connect(("8.8.8.8", 80))
IPAddr = s.getsockname()[0]
GPIO.setmode(GPIO.BOARD)
resistorPin = 7
app = Flask(__name__)
@app.route('/')
def index():
    light = read_photoresistor()
    html = """
    <head>
        <title>Mail Detect</title>
        <link href='https://fonts.googleapis.com/css?family=Lexend' rel='stylesheet'>
    <style>
        body {
            font-family: Arial, Helvetica, sans-serif;
            background-color: #f1f1f1;
        }
        .box {
            width: 400px;
            height: 50px;
            background-color: #fff;
            border: 1px solid #000;
            border-radius: 10px;
            margin: 0 auto;
            margin-top: 100px;
            text-align: center;
            font-size: 20px;
            font-weight: bold;
            padding-top: 30px;
        }
        .title {
            font-size: 30px; 
            font-family: 'Lexend';  
        }
    </style>
    </head>
    <p>
    &nbsp;
    </p>
    <center>
        <div class="title">Mail Detecting System <br><br> Running on: <b>"""+ hostname + """</b></div> 
        """
    if light > 100:
        html += '<div class="box" style="border: 10px solid #98e5e5">Mail Detected ✉️'
    else:
        html += '<div class="box" style="border: 10px solid gray"> No Mail ❌'
    html += """
        </div>
    </center>
    <script>
        setInterval(function() {
            fetch('/api/v1/light')
            .then(response => response.text())
            .then(data => {
                if (data > 100) {
                    document.querySelector('.box').style.border = '10px solid #98e5e5';
                    document.querySelector('.box').innerHTML = 'Mail Detected ✉️';
                } else {
                    document.querySelector('.box').style.border = '10px solid gray';
                    document.querySelector('.box').innerHTML = 'No Mail ❌';
                }
            });
        }, 2000);
    </script>
    """
    return html
@app.route('/api/v1/light') # We use the route /api/v1/light to get the value of the photoresistor
def light():
    return str(read_photoresistor())
def read_photoresistor():
    time.sleep(1)
    GPIO.setup(resistorPin, GPIO.OUT)
    GPIO.output(resistorPin, GPIO.LOW)
    time.sleep(0.1)
    GPIO.setup(resistorPin, GPIO.IN) # 
    currentTime = time.time() 
    diff = 0
    while(GPIO.input(resistorPin) == GPIO.LOW): 
        diff  = time.time() - currentTime 
    return (diff * 1000)
if __name__ == '__main__':
    app.run(host=IPAddr, debug=True)

And now it will continuously update the value of the photoresistor without needing to refresh the page.

And if I block the LED with my hand it will detect that there is mail in the mailbox.

mailbox website [IMG]

This was a fun project to make as I worked a lot with electronics and I learned a lot about the photoresistor and how it works.

This project is more practical as it can be used in real life. I hope you enjoyed this project and thanks for reading :D