DEV Community

Cover image for Fish Cam : Live water temperature monitoring
philbasford for AWS Community Builders

Posted on • Edited on

Fish Cam : Live water temperature monitoring

In this blog I will be looking into how to implement live monitoring of water temperature inside the Fish tank where my daughters two goldfish, Goldie and Star, spend their days swimming in. The tank is located next to the desk that I am writing this blog from.

Further background:

My "Office" (aka my daughters bedroom) is located at the front of our house. Our house is north facing, therefore in the morning the sun is on the back of the house and in the afternoon it is on the front where my "Office" is. My "Office has a window that I look out from when I am working from home. In late afternoon the sun especially shines straight in this window (you may see this on some of the YouTube videos that I have done).

The main thing here is the safety of the Fish. They ** must not** be put in direct sunshine, this would heat the water too much and also increase the algae due to the sunlight. So an extra safeguard I typically close the curtains when the sun is very bright or strong. However the ambient temperature of the room still increases and this means the water temperature of the tank also increases. Therefore it would be good to know the water temperature as Goldfish require a range of between 20°C and 23°C.

Hardware:

To measure water temperature you need to invest in some additional components to extend your RaspberryPI. Both a submersible temperature Sensor like the DS18B20 (’https://www.amazon.co.uk/dp/B08HHWD1K9) and a breadboard is essential for this. The DS18B20 is ideal as it has an integrated digital interface where you can read straight from it and you do not need to fiddle with an analog converter.

To setup the hardware you need to do the following:

  • Shutdown your Pi and unplug it
  • Connect your breadboard to the GPIO on the Pi
  • Connect the sensor to the breadboard.
  • Start up your Pi

Here is a wiring diagram that helps, thanks to Circuit Basics:

Circuit

Some tips:

  • A pull-up resistor of at least 4.7 Ohm is required. This basically allows the sensor's reading to be between a value of 0 and 5 volts and that the digital interface contained within the housing can understand it. The PI then reads this as a temperature from the digital interface. I had to use a resistor from my breadboard kit and this website will help you workout what Ohm is your resistor is https://www.calculator.net/resistor-calculator.html. If you need use a couple of resistors then make sure you wire them in series.

  • Notice the sensor above comes with a connector at the end. For my first attempt to wire this up, I cut off the connector, then I stripped the wires and tried push them into my breadboard. This was a mistake and did not work. Luckily my breadboard came with a single row pin header. This allowed me to put the row pin header into the board and connector on to the row pin header. Much easier but make sure you place the single row pin header horizontally not vertically.

My final breadboard wiring looked like this:

Board

GPIO

Turning on and setting up the sensor:

You will need to turn on the digital(one-wire) interface on the PI (GPIO 4). This can be done using the raspi-config.

Select the Interfaces Menu:

Select Interfaces

Select one-wire:

Select one wire

Turn on one-wire:

turn on

Then restart your Pi

Testing

Using cat then we can see if we can read the temperature value straight from the digital interface(one wire). One-wire mounts devices at /sys/bus/w1/devices/. This folder should contain another folder with the name of the UUID of this sensor. This is in case you have more than one and each one would have its own folder. With in the sensor's folder then there is a file called 'temperature' and it contents is the °C value from the sensor.

Here is an example:

ssh pi                       
Linux raspberrypi 5.10.17-v7+ #1414 SMP Fri Apr 30 13:18:35 BST 2021 armv7l

The programs included with the Debian GNU/Linux system are free software;
the exact distribution terms for each program are described in the
individual files in /usr/share/doc/*/copyright.

Debian GNU/Linux comes with ABSOLUTELY NO WARRANTY, to the extent
permitted by applicable law.
Last login: Sat Apr  2 10:53:44 2022 from 192.168.x.x
pi@raspberrypi:~ $ ls /sys/bus/w1/devices/
28-00000001a7df  w1_bus_master1
pi@raspberrypi:~ $ cat /sys/bus/w1/devices/28-00000001a7df/temperature 
20625
pi@raspberrypi:~ $ 
Enter fullscreen mode Exit fullscreen mode

So that is 20.625c.

Lambda:

In order to get the temperature from the one wire interface into AWS we need to write some custom code. To do this I used a lambda written in Python. The lambda reads the temperature from the interface like any other normal file. The value is then parsed and turned into a JSON object. Then using the publish option in the Greengrass SDK I send the message via an MQTT topic into IoT core. Here is some very important concerns/notes when doing this:

  1. The lambda needs to be a long running lambda. This basically means it is not a lambda at all. It is in fact a Python app that is deployed via GreenGrass and started when Greengrass starts. It needs a run loop, poll the temperature and publish. Listening to sigterm and sigint
  2. It needs to run as a container with read access to /sys on the host.
  3. The Greengrass v2 and v1 way off sending MQTT messages and how a deployment is done is completely different. My code was originally v1 so I had to deploy the LegacySubscriptionRouter with the correct configuration to make it compatible
  4. I had to do a OTA upgrade of nucleus so that my other components were deployable

Here is the main function executed within the run loop:

def process(iot_client: Client):

    # conh
    sensor_id = os.getenv('TEMP_ID', '28-00000001e2d1')
    path = os.getenv('TEMP_PATH', '/sys/bus/w1/devices/')

    temperature = gettemp(sensor_id, path) / float(1000)
    msg = {
        'sensor_id': sensor_id,
        'temperature': temperature
    }

    send_message(iot_client, msg)

    return msg
Enter fullscreen mode Exit fullscreen mode

This is how to read the temperature:

def gettemp(sensor_id, path):
    try:
        mytemp = ''
        filename = 'w1_slave'
        f = open(path + sensor_id + '/' + filename, 'r')
        line = f.readline()  # read 1st line
        crc = line.rsplit(' ', 1)
        crc = crc[1].replace('\n', '')
        if crc == 'YES':
            line = f.readline()  # read 2nd line
            mytemp = line.rsplit('t=', 1)
        else:
            mytemp = 99999
        f.close()

        return int(mytemp[1])

    except Exception as e:
        LOGGING.exception(e)
        return 99999
Enter fullscreen mode Exit fullscreen mode

This is how to send the message via MQTT (note this is V1 style):

def send_message(iot_client: greengrasssdk.IoTDataPlane.Client, message: dict, topic=TOPIC):

    iot_client.publish(
        topic=topic,
        payload=json.dumps(message).encode("utf-8")
    )

Enter fullscreen mode Exit fullscreen mode

Now if you want to look at code in more detail it is located [https://github.com/philbasford/fishcam] and also note I deployed this using AWS SAM.

IOT Component:

First select the Fish Cam Lambda and use the lambda name and version as the initial component details. However you may need to update the version later if you need to reconfigure.

Select Lambda

As mention before one of the most important things is that the lambda is pinned/long running, so that it starts up and down with Greengrass. Also remove any event source:

Long Running

Make sure you use a container but with access to /sys:

Container

Create IOT Deployment:

Create a new deployment:

New Deployment

Add the Lambda component you created, then the legacy subscription router and nucleus public components.

Add Components

Now go into the legacy subscription router configuration:

Configure Legacy

Add the following(change the function name to match yours):

{
    "subscriptions": {
        "Greengrass_FishCam_to_cloud": {
            "id": "Greengrass_FishCam_to_cloud",
            "source": "component:cfFishCam-FishCamFunction-R7vEIfJai8vl",
            "subject": "fishcam/temperature",
            "target": "cloud"
        }
    }
}

Enter fullscreen mode Exit fullscreen mode

Then submit it:

Legacy Settings

Leave the Advance Settings as is:

Advance Settings

Then deploy:

Deploy

Now if it deployed ok, then within the test area of IOT Greengrass section of the AWS Console you should see the messages coming in:

MQTT

IOT Rule & CloudWatch

To get the messages into CloudWatch you need to create a new rule in the Act area of IOT Greengrass section of the AWS Console:

New Rule

Use a the following SQL to make sure you process all the messages sent to the topic:

Select SQL

The add action and select CloudWatch Metric:

CW Rule

Then use the following settings with substitutions for the values contain in messages:

CW Settings

Your need a IAM Role and here are the permissions needed:

{
    "Version": "2012-10-17",
    "Statement": {
        "Effect": "Allow",
        "Action": "cloudwatch:PutMetricData",
        "Resource": [
            "*"
        ]
    }
}
Enter fullscreen mode Exit fullscreen mode

Conclusion:

Measuring temperature of the water in the fish tank is really easy, I was able to do this in few hours once I had purchased the additional sensor. It is a good example of how to chain a lot AWS services together to make something useful at low cost.

Here is the final CloudWatch chart for the temperature for a a day, notice the as the sun comes up and goes down the temperature changes:

Chart

Thank you and I hope you found this interesting. Next time I will be looking at how to use machine learning to spot fish!

Top comments (0)