Help on code for Mysensors 2.0



  • Hi everone.

    Couple of months i created a code for my dallas an ph sensor. however i reinstalled my pc and installed the mysensors 2.0 resulting in a error with No forward link or gateway feature activated.

    /*
      Original code created by DFRobot for their probes, 
      adapted by haze5@icloud.com Mysensors for the project.
    */
    
    
    #include <SPI.h>
    #include <MySensors.h>
    #include <DallasTemperature.h>
    #include <OneWire.h>
    #include "Vcc.h"        // https://github.com/Yveaux/Arduino_Vcc
    
    static const float VccMin = 0.0;        // Minimum expected Vcc level, in Volts. (0.6V for 1 AA Alkaline)
    static const float VccMax = 3.3;        // Maximum expected Vcc level, in Volts. (1.5V for 1 AA Alkaline)
    static const float VccCorrection = 3.29 / 3.31;  // Measured Vcc by multimeter divided by reported Vcc
    Vcc vcc(VccCorrection);
    
    #define CHILD_ID_PH 0
    #define ArrayLenth 10               // times of collection
    #define PH_SENSOR_ANALOG_PIN A0         // pH meter Analog output to Arduino Analog Input 0
    #define LED_DIGITAL_PIN 13
    #define Offset 0.25           //deviation compensate
    
    #define CHILD_ID_TEMP 1
    #define COMPARE_TEMP 1 // Send temperature only if changed? 1 = Yes 0 = No
    #define ONE_WIRE_BUS 3 // Pin where dallase sensor is connected 
    #define MAX_ATTACHED_DS18B20 16
    
    unsigned long SLEEP_TIME = 30000; // Sleep time between reads (in milliseconds)
    OneWire oneWire(ONE_WIRE_BUS); // Setup a oneWire instance to communicate with any OneWire devices (not just Maxim/Dallas temperature ICs)
    DallasTemperature sensors(&oneWire); // Pass the oneWire reference to Dallas Temperature.
    
    unsigned long lastSend = 0;
    static const unsigned long SEND_FREQUENCY = 30000;  // Minimum time between send (in milliseconds)
    
    MySensor gw;
    
    float lastPhHValue;
    static const float deltaPhValue = 0.5;
    
    float lastTemperature[MAX_ATTACHED_DS18B20];
    int numSensors=0;
    boolean receivedConfig = false;
    boolean metric = true; 
    
    MyMessage msgPH(CHILD_ID_PH, V_VAR1);
    MyMessage msg(CHILD_ID_TEMP,V_TEMP);
    
    void setup()
    {
        // Startup up the OneWire library
        sensors.begin();
        // requestTemperatures() will not block current thread
        sensors.setWaitForConversion(false);
        
        //gw.begin(NULL, 100, false); //deze gebruiken, 100 is de node_id, die weer gebruiken in pimatic
        gw.begin();
        //Serial.print("0;0;3;0;2;");Serial.print(LIBRARY_VERSION);
        
        pinMode(LED_DIGITAL_PIN, OUTPUT);
        numSensors = sensors.getDeviceCount();
        
        // Send the Sketch Version Information to the Gateway
        gw.sendSketchInfo("pHmeter", "1.0");
        
        // Register all sensors to gw (they will be created as child devices)
        gw.present(CHILD_ID_PH, S_WATER);
    
        // Present all sensors to controller
        for (int i=0; i<numSensors && i<MAX_ATTACHED_DS18B20; i++) {   
        gw.present(i, S_TEMP);
        }
    }
    
    void loop()
    {
        // By calling process() you route messages in the background
        gw.process();
        read_PH();
        read_TEMP();
    
        gw.sleep(SLEEP_TIME);
    }
    
    void read_PH(){
       unsigned long now = millis();
        bool sendTime = now - lastSend > SEND_FREQUENCY;
        if (sendTime)
        {
            lastSend = now;
            
            //    float v = vcc.Read_Volts();
            //    Serial.print("VCC = " );
            //    Serial.print(v);
            //    Serial.println(" Volts" );
            
            int batteryPcnt = (int)vcc.Read_Perc(VccMin, VccMax);
            //    Serial.print("VCC = " );
            //    Serial.print(batteryPcnt);
            //    Serial.println(" %" );
            
            gw.sendBatteryLevel(batteryPcnt);
        }
        
        // Read PH_SENSOR_ANALOG_PIN in phValue
        float voltage = analogReadAverage(PH_SENSOR_ANALOG_PIN, 10) * 5.0 / 1024;
        
        // convert the millivolt into pH value
        float PhValue = 3.5 * voltage+Offset;
        
        
        
        
        if (sendTime || abs(PhValue - lastPhHValue) > deltaPhValue)
        {
            Serial.print("    pH:");
            Serial.print(PhValue, 2);
            Serial.println(" ");
            
            gw.send(msgPH.set(PhValue, 2)); // envoi au reseau avec deux decimales
            
            digitalWrite(LED_DIGITAL_PIN, digitalRead(LED_DIGITAL_PIN) ^ 1);
            
            lastPhHValue = PhValue;
        }
    }
    
    
    void read_TEMP(){
      sensors.requestTemperatures();
    
      // query conversion time and sleep until conversion completed
      int16_t conversionTime = sensors.millisToWaitForConversion(sensors.getResolution());
      // sleep() call can be replaced by wait() call if node need to process incoming messages (or if node is repeater)
      gw.sleep(conversionTime);
    
      // Read temperatures and send them to controller 
      for (int i=0; i<numSensors && i<MAX_ATTACHED_DS18B20; i++) {
     
        // Fetch and round temperature to one decimal
        float temperature = static_cast<float>(static_cast<int>((gw.getConfig().isMetric?sensors.getTempCByIndex(i):sensors.getTempFByIndex(i)) * 10.)) / 10.;
     
        // Only send data if temperature has changed and no error
        #if COMPARE_TEMP == 1
        if (lastTemperature[i] != temperature && temperature != -127.00 && temperature != 85.00) {
        #else
        if (temperature != -127.00 && temperature != 85.00) {
        #endif
     
          // Send in the new temperature
          gw.send(msg.setSensor(i).set(temperature,1));
          // Save new temperatures for next compare
          lastTemperature[i]=temperature;
        }
      }   
    }
    
    
    
    
    
    
    
    double analogReadAverage(uint8_t pin, unsigned long ms)
    {
        double average = 0;
        int buffer[ArrayLenth];
        
        for (int i = 0; i < ArrayLenth; i++)
        {
            buffer[i] = analogRead(PH_SENSOR_ANALOG_PIN);
            delay(ms);
        }
        
        if (ArrayLenth < 5)
        {
            // less than 5, calculated directly statistics
            for (int i = 0; i < ArrayLenth; i++)
            {
                average += buffer[i];
            }
            average = average / ArrayLenth;
        }
        else
        {
            // Sort the values from small to large
            for (int i = 0; i < ArrayLenth; i++)
            {
                for (int j = i + 1; j < 10; j++)
                {
                    if (buffer[i] > buffer[j])
                    {
                        int temp = buffer[i];
                        buffer[i] = buffer[j];
                        buffer[j] = temp;
                    }
                }
            }
            
            // take the average value of center sample
            for (int i = 2; i < ArrayLenth - 2; i++)
            {
                average += buffer[i];
            }
            
            average = average / (ArrayLenth - 4);
        }
        
        return average;
    }
    
    

  • Hardware Contributor

    Hello.

    It seems that you have not converted your old sketch to 2.0 In the new release there is no need of declaring a Mysensor instance like before "Mysensor gw" and some other stuff ..

    So, what you can do:

    If still issues, please post logs so we can help you. without not easy..



  • @scalz said:

    Hello.

    It seems that you have not converted your old sketch to 2.0 In the new release there is no need of declaring a Mysensor instance like before "Mysensor gw" and some other stuff ..

    So, what you can do:

    If still issues, please post logs so we can help you. without not easy..

    I will try to change the code. for now i installed the 1.5 again however compiling is fine but it says at the end error uploading sketch thats all no other messages.. i already tried 3 board other usb cable. com ports settings are fine and i can use the serial monitor with the existing code but somehow it wont upload the new.


  • Hardware Contributor

    @stingone - if you get an upload error, its not about the code. Its the connection between computer, usb, uploader and arduino.

    Do you have changed drivers for the programmer recently? got the right programmer set in arduino ide? Can you paste the log from arudino ide in here?


  • Hardware Contributor

    @stingone oki. when it fails to upload (for the error you're describing), try again to upload, that should work. sometimes it can need multiple attempts. it's not mysensors related.


  • Hardware Contributor

    After recompiling to 2.0.0 and uploading to your node, you will also need to bring the gateway to version 2.0.0

    A 2.0.0 gateway should work with older nodes. But new 2.0.0 nodes require a 2.0.0 gateway.



  • Got it working with another laptop..wil recompile to 2.0 in the weekend.

    Just a quick question. I have 2 sensors on 1 arduino. Works fine however sometimes it shows the value of the temp sensor on the value of the ph sensor.

    #include <SPI.h>
    #include <MySensor.h>
    #include <DallasTemperature.h>
    #include <OneWire.h>
    #include "Vcc.h"        // https://github.com/Yveaux/Arduino_Vcc
    
    static const float VccMin = 0.0;        // Minimum expected Vcc level, in Volts. (0.6V for 1 AA Alkaline)
    static const float VccMax = 3.3;        // Maximum expected Vcc level, in Volts. (1.5V for 1 AA Alkaline)
    static const float VccCorrection = 3.29 / 3.31;  // Measured Vcc by multimeter divided by reported Vcc
    Vcc vcc(VccCorrection);
    
    #define CHILD_ID_PH 0
    #define ArrayLenth 10               // times of collection
    #define PH_SENSOR_ANALOG_PIN A0         // pH meter Analog output to Arduino Analog Input 0
    #define LED_DIGITAL_PIN 13
    #define Offset 0.25           //deviation compensate
    
    #define CHILD_ID_TEMP 1
    #define COMPARE_TEMP 0 // Send temperature only if changed? 1 = Yes 0 = No
    #define ONE_WIRE_BUS 3 // Pin where dallase sensor is connected 
    #define MAX_ATTACHED_DS18B20 16
    
    unsigned long SLEEP_TIME = 30000; // Sleep time between reads (in milliseconds)
    OneWire oneWire(ONE_WIRE_BUS); // Setup a oneWire instance to communicate with any OneWire devices (not just Maxim/Dallas temperature ICs)
    DallasTemperature sensors(&oneWire); // Pass the oneWire reference to Dallas Temperature.
    
    unsigned long lastSend = 0;
    static const unsigned long SEND_FREQUENCY = 30000;  // Minimum time between send (in milliseconds)
    
    MySensor gw;
    
    float lastPhHValue;
    
    float lastTemperature[MAX_ATTACHED_DS18B20];
    int numSensors=0;
    boolean receivedConfig = false;
    boolean metric = true; 
    
    MyMessage msgPH(CHILD_ID_PH, V_VAR1);
    MyMessage msg(CHILD_ID_TEMP,V_TEMP);
    
    void setup()
    {
        // Startup up the OneWire library
        sensors.begin();
        // requestTemperatures() will not block current thread
        sensors.setWaitForConversion(false);
        
        //gw.begin(NULL, 100, false); //deze gebruiken, 100 is de node_id, die weer gebruiken in pimatic
        gw.begin();
        //Serial.print("0;0;3;0;2;");Serial.print(LIBRARY_VERSION);
        
        pinMode(LED_DIGITAL_PIN, OUTPUT);
        numSensors = sensors.getDeviceCount();
        
        // Send the Sketch Version Information to the Gateway
        gw.sendSketchInfo("pHmeter", "1.1");
        
        // Register all sensors to gw (they will be created as child devices)
        gw.present(CHILD_ID_PH, S_WATER);
    
        // Present all sensors to controller
        for (int i=0; i<numSensors && i<MAX_ATTACHED_DS18B20; i++) {   
        gw.present(i, S_TEMP);
        }
    }
    
    void loop()
    {
        // By calling process() you route messages in the background
        gw.process();
        read_PH();
        read_TEMP();
    
        gw.sleep(SLEEP_TIME);
    }
    
    void read_PH(){
       unsigned long now = millis();
        bool sendTime = now - lastSend > SEND_FREQUENCY;
        if (sendTime)
        {
            lastSend = now;
            
            //    float v = vcc.Read_Volts();
            //    Serial.print("VCC = " );
            //    Serial.print(v);
            //    Serial.println(" Volts" );
            
            int batteryPcnt = (int)vcc.Read_Perc(VccMin, VccMax);
            //    Serial.print("VCC = " );
            //    Serial.print(batteryPcnt);
            //    Serial.println(" %" );
            
            gw.sendBatteryLevel(batteryPcnt);
        }
        
        // Read PH_SENSOR_ANALOG_PIN in phValue
        float voltage = analogReadAverage(PH_SENSOR_ANALOG_PIN, 10) * 5.0 / 1024;
        
        // convert the millivolt into pH value
        float PhValue = 3.5 * voltage+Offset;
        
        
        
            Serial.print("    pH:");
            Serial.print(PhValue, 2);
            Serial.println(" ");
            
            gw.send(msgPH.set(PhValue, 2)); // envoi au reseau avec deux decimales
            
            digitalWrite(LED_DIGITAL_PIN, digitalRead(LED_DIGITAL_PIN) ^ 1);
            
            lastPhHValue = PhValue;
        
    }
    
    
    void read_TEMP(){
      sensors.requestTemperatures();
    
      // query conversion time and sleep until conversion completed
      int16_t conversionTime = sensors.millisToWaitForConversion(sensors.getResolution());
      // sleep() call can be replaced by wait() call if node need to process incoming messages (or if node is repeater)
      gw.sleep(conversionTime);
    
      // Read temperatures and send them to controller 
      for (int i=0; i<numSensors && i<MAX_ATTACHED_DS18B20; i++) {
     
        // Fetch and round temperature to one decimal
        float temperature = static_cast<float>(static_cast<int>((gw.getConfig().isMetric?sensors.getTempCByIndex(i):sensors.getTempFByIndex(i)) * 10.)) / 10.;
     
        // Only send data if temperature has changed and no error
        #if COMPARE_TEMP == 1
        if (lastTemperature[i] != temperature && temperature != -127.00 && temperature != 85.00) {
        #else
        if (temperature != -127.00 && temperature != 85.00) {
        #endif
     
          // Send in the new temperature
          gw.send(msg.setSensor(i).set(temperature,1));
          // Save new temperatures for next compare
          lastTemperature[i]=temperature;
        }
      }   
    }
    
    
    
    
    
    
    
    double analogReadAverage(uint8_t pin, unsigned long ms)
    {
        double average = 0;
        int buffer[ArrayLenth];
        
        for (int i = 0; i < ArrayLenth; i++)
        {
            buffer[i] = analogRead(PH_SENSOR_ANALOG_PIN);
            delay(ms);
        }
        
        if (ArrayLenth < 5)
        {
            // less than 5, calculated directly statistics
            for (int i = 0; i < ArrayLenth; i++)
            {
                average += buffer[i];
            }
            average = average / ArrayLenth;
        }
        else
        {
            // Sort the values from small to large
            for (int i = 0; i < ArrayLenth; i++)
            {
                for (int j = i + 1; j < 10; j++)
                {
                    if (buffer[i] > buffer[j])
                    {
                        int temp = buffer[i];
                        buffer[i] = buffer[j];
                        buffer[j] = temp;
                    }
                }
            }
            
            // take the average value of center sample
            for (int i = 2; i < ArrayLenth - 2; i++)
            {
                average += buffer[i];
            }
            
            average = average / (ArrayLenth - 4);
        }
        
        return average;
    }
    
    
    

    And in Pimatic

    {
          "id": "temperatuur-aquarium",
          "name": "Temperatuur Aquarium",
          "class": "MySensorsDST",
          "nodeid": 2,
          "sensorid": 0
        },
        {
          "id": "phmeter-aquarium",
          "name": "pH Aquarium",
          "class": "MySensorsPH",
          "nodeid": 2,
          "sensorid": 0,
          "batterySensor": false
        },
    

    debug [pimatic-mysensors]: <- MySensorsPH { sender: 2, sensor: 0, type: 0, value: '25.4' }
    22:23:06debug [pimatic-mysensors]: <- MySensorDST { sender: 2, sensor: 0, type: 0, value: '25.4' }
    22:23:06debug [pimatic-mysensors]: <- I_LOG_MESSAGE 0;0;3;0;9;read: 2-2-0 s=0,c=1,t=0,pt=7,l=5,sg=0:25.4
    22:23:05debug [pimatic-mysensors]: <- MySensorsPH { sender: 2, sensor: 0, type: 24, value: '6.60' }
    22:23:05debug [pimatic-mysensors]: <- I_LOG_MESSAGE 0;0;3;0;9;read: 2-2-0 s=0,c=1,t=24,pt=7,l=5,sg=0:6.60

    If you look at the output in pimatic you can see that the mysensorsPH and mysensorsDST have the temp value. This happens 50% of the time.


  • Mod

    @stingone this line

     gw.send(msg.setSensor(i).set(temperature,1));
    

    will set the child id to 0 for the first temp sensor, to 1 for the second and so on. That's why you get temperature data on id 0, which is also used by the ph sensor.



  • @mfalkvidd said:

    @stingone this line

     gw.send(msg.setSensor(i).set(temperature,1));
    

    will set the child id to 0 for the first temp sensor, to 1 for the second and so on. That's why you get temperature data on id 0, which is also used by the ph sensor.

    What would be best to do?


  • Mod

    It depends on how much work you are prepared to do and how much flexibility you want in the code.

    If you only use one temperature sensor and don't have any plans to add more, change

    gw.send(msg.setSensor(i).set(temperature,1));
    

    to

    gw.send(msg.set(temperature,1));
    


  • @mfalkvidd said:

    It depends on how much work you are prepared to do and how much flexibility you want in the code.

    If you only use one temperature sensor and don't have any plans to add more, change

    gw.send(msg.setSensor(i).set(temperature,1));
    
    gw.send(msg.set(temperature,1));
    

    Only using the ph sensor and 1 temperatuur sensor no need for more temp sensors.


Log in to reply
 

Suggested Topics

47
Online

11.5k
Users

11.1k
Topics

112.7k
Posts