Garduino

From Randomdata wiki
Jump to navigation Jump to search

The gArduino is a Automated Garden Facility (AGF) and it's build on the Arduino platform. It is used to automate your garden. Our primary goal is to "water" the garden and monitor the garden environment. Currently we have build the gArduino to measure the water status of a plant. The information is collected by an NSLU2 mini linux server. The information will be published soon on cacti graphs.


Why

Why the Garduino? Well, we are very lazy and we want to be sure our garden is doing fine in the hot summer days

What

We have several sensors. Currently we have build these ones:

  • Moist sensor
  • Solar sensor
  • Water level sensor (same as moist)

Some sensors which can be attached in the future:

  • Rain sensor (will be adjusted moist sensor)
  • Wind sensor
  • Temperature sensor (ordered and installed soon)

Statistics

Our graph page: [1] We build our graphs in cacti.


Actions

  • start the pump
  • water circuit selector
  • Call for help:
    • Out of water
    • Ground is not getting wet
  • Monitor
    • How much water is needed and how often
    • Weather/water conditions
    • Graphs in cacti
    • etc.

Details

Overview

This is a small overview, and technical overview will be posted soon. The relais and pump or missing in the code for now.

File:GArduino picture.jpg

Moist sensor

The current 0.2 moist sensor is build of:

  • film canister (one with a flat cap!)
  • two INOX screws (about >8 cm)
  • Wire (we used CAT5)
  • Epoxy

HOW to build:

  • Open the film canister and screw the screws thru the cap
  • Drill a hole in the bottom of the canister and put the wire thru
  • Strip the wire and connect two of them to the screws in the cap
  • Mix the epoxy and will the canister (Be sure to ventilate!)
  • Add the cap to the canister and put it upside-down so the epoxy will protect the connections of the screws
  • Connect the other end of the wires to you gArduino configuration

Overview

Nice to have:

  • Build the circuit into the moist sensor, this will increase the quality of the measurements
  • Stacking option of multi sensors

Solar sensor

The solar sensor is just a solar panel, we need to adjust it a bit because it's working to well. For this we will use a filter.
File:GArduino solarsensor.jpg

Code

My programming skill's aren't the best.... but it's functioning :-)
As you can see, I added an ethernet shield to monitor the data. You can do this monitoring also by using the USB serial connection. We are working on live cacti graphs... soon they will be available.

Some of the code is re-used code from several example's of the Arduino toolkit. You will probably recognize some.

Current dev version of the gArduino is v0.5 alpha

/*
 * AnalogInput
 * by DojoDave <http://www.0j0.org>
 *
 * Turns on and off a light emitting diode(LED) connected to digital  
 * pin 13. The amount of time the LED will be on and off depends on
 * the value obtained by analogRead(). In the easiest case we connect
 * a potentiometer to analog pin 2.
 *
 * http://www.arduino.cc/en/Tutorial/AnalogInput
 */

int potPin = 0;         // select the input pin for the moist sensor
int solarPin = 5;       // select the input pin for the solar sensor
int tempPin = 4;        // select the input pin for the temp sensor
int ledprobe = 9;       // select the input pin for the sensor
int ledPin1 = 8;        // select the pin for the LED
int val = 0;            // variable to store the value coming from the sensor
int valtot = 0;         // variable to store the total probe details
int valprobe = 0;       // variable to store the probe details
int valcnt = 0;         // variable to store the probe counter
int valtmp = 0;         // variable to temperary store the total probe details
int interval = 32;      // the total ammount of probes before advertising the average, caution!!! the max count is 32 if measurring a max 1024 vallue!!! WILL BE CORRECTED WITH long int soon
int probedelay = 100;   // the delay between the probes in milliseconds
int wet = 350;          // the probevalue to decide what wet is
int dry = 200;          // the probevalue to decide what too dry is

#include "Ethernet.h"
#include "WebServer.h"

// network configuration.  gateway and subnet are optional.
byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED };
byte ip[] = { 192, 168, 1, 177 };
byte gateway[] = { 192, 168, 1, 1 };
byte subnet[] = { 255, 255, 255, 0 };

/* This creates an instance of the webserver.  By specifying a prefix
 * of "/", all pages will be at the root of the server. */
#define PREFIX "/garduino"
WebServer webserver(PREFIX, 80);

/* commands are functions that get called by the webserver framework
 * they can read any posted data from client, and they output to the
 * server to send data back to the web browser. */
void helloCmd(WebServer &server, WebServer::ConnectionType type)
{
  /* this line sends the standard "we're all OK" headers back to the
     browser */
  server.httpSuccess();

  /* if we're handling a GET or POST, we can output our data here.
     For a HEAD request, we just stop after outputting headers. */
  if (type != WebServer::HEAD)
  {
    /* this defines some HTML text in read-only memory aka PROGMEM.
     * This is needed to avoid having the string copied to our limited
     * amount of RAM. */
    P(helloMsg) = "<h1>Hello, World!</h1>";

    /* this is a special form of print that outputs from PROGMEM */
    server.printP(helloMsg);
  }
}

void moistCmd(WebServer &server, WebServer::ConnectionType type)
{
  /* this line sends the standard "we're all OK" headers back to the
     browser */
  server.httpSuccess();

  /* if we're handling a GET or POST, we can output our data here.
     For a HEAD request, we just stop after outputting headers. */
  if (type != WebServer::HEAD)
  {
    //Print Moist val on http
    server.print(val);
  }
}

void solarCmd(WebServer &server, WebServer::ConnectionType type)
{
  /* this line sends the standard "we're all OK" headers back to the
     browser */
  server.httpSuccess();

  /* if we're handling a GET or POST, we can output our data here.
     For a HEAD request, we just stop after outputting headers. */
  if (type != WebServer::HEAD)
  {
    //Print Solar on http
    int valsolar = analogRead(solarPin);
    server.print(valsolar);
  }
}

void tempCmd(WebServer &server, WebServer::ConnectionType type)
{
  /* this line sends the standard "we're all OK" headers back to the
     browser */
  server.httpSuccess();

  /* if we're handling a GET or POST, we can output our data here.
     For a HEAD request, we just stop after outputting headers. */
  if (type != WebServer::HEAD)
  {
    //Print Solar on http
    int valtemp = analogRead(tempPin);
    server.print(valtemp);
  }
}


void setup() {
  pinMode(ledPin1, OUTPUT);  // declare the ledPin as an OUTPUT
  pinMode(ledprobe, OUTPUT);  // declare the ledPin as an OUTPUT
  Serial.begin(9600);
  
    // initialize the ethernet device
  Ethernet.begin(mac, ip, gateway, subnet);
  
  /* setup our default command that will be run when the user accesses
   * the root page on the server */
  webserver.setDefaultCommand(&helloCmd);

  /* run the same command if you try to load /index.html, a common
   * default page name */
  webserver.addCommand("index.html", &helloCmd);
  webserver.addCommand("moist", &moistCmd);
  webserver.addCommand("solar", &solarCmd);
  webserver.addCommand("temp", &tempCmd);
  /* start the webserver */
  webserver.begin();

}

void loop() {
  
do
{
    digitalWrite(ledprobe, HIGH);  // turn the ledprobe on
    valprobe = analogRead(potPin); // read the value from the sensor
    delay(50);
    digitalWrite(ledprobe, LOW);  // turn the ledprobe off
    valtmp = valtot;
    valtot = valtmp + valprobe;
    valcnt = valcnt + 1;
    delay(probedelay - 50);
} while (valcnt != interval);
  

val = (valtot / valcnt);
if (val < 300) //TODOPARAMETER
  {
  digitalWrite(ledPin1, HIGH);  // turn the ledPin on
  }
  else 
  {
  digitalWrite(ledPin1, LOW);  // turn the ledPin off
  }
  
Serial.println(val);   

/* process incoming connections one at a time forever */
webserver.processConnection();

valcnt = 0;
valtot = 0;
}