/****************************************************** * (C) 2020 Dirk Jahnke (dirk.jahnke@mailbox.org) * Uses Arduino with Ethernet Shield. * Thus the GPIO usage is limited, because the shield * needs the SPI pns: 10 (SS), 11 (MOSI), 12 (MISO), 13 (SCK). * * If the SD card is to be used, then pin 4 is used as SS * for the SD card. * * We use 2 outputs to control Relays for * - switching LED at the mailbox (light on/off) and * - trigger the internal bell (gong). * Following pins can be used as I/O: * - 1 * - 2 * - 3 * - 5 * - 6 * - 7 * - 8 * - 9 * - 14 (A0) * - 15 (A1) * - 16 (A2) * - 17 (A3) * - 18 (A4) * - 19 (A5) ******************************************************/ #include #include #include #include #include // Update these with values suitable for your network. byte mac[] = { 0xDE, 0xED, 0xBA, 0xFE, 0xFE, 0xEA }; IPAddress ip(192, 168, 89, 205); IPAddress mqttServerIP(192, 168, 89, 95); const char *mqttUser = "default"; const char *mqttPassword = "12345678"; const char *doorBellTopic = "doorBellGW/doorbell"; const char *postboxFlapTopic = "doorBellGW/mailboxFlap"; const char *postboxDoorTopic = "doorBellGW/mailboxDoor"; const char *doorBellTriggerSoundTopic = "doorBellGW/doorbellTrigger"; const char *postboxLightTopic = "doorBellGW/light"; const char *doorLightRequestTopic = "doorBellGW/lightRequest"; EthernetClient ethClient; EthernetServer server(80); PubSubClient mqttClient(ethClient); #define DOOR_BELL_BUTTON_PIN 9 #define LIGHT_REQUEST_BUTTON_PIN 8 #define POSTBOX_FLAP_PIN 7 #define POSTBOX_DOOR_PIN 6 // Output pins #define LED_PIN 13 #define DOOR_BELL_BUZZER_PIN 14 #define POSTBOX_LIGHT_PIN 15 #define OUTPUT_NONE 0xff struct { const uint8_t pin; const unsigned long quietAfterTriggerFor_ms; unsigned long lastTrigger_ts; bool isTriggered; const char *topic; const uint8_t triggerOutputPin; } myInputs[] = { { DOOR_BELL_BUTTON_PIN, 500, 0 , false, doorBellTopic, DOOR_BELL_BUZZER_PIN }, { LIGHT_REQUEST_BUTTON_PIN, 500, 0 , false, doorLightRequestTopic, POSTBOX_LIGHT_PIN }, { POSTBOX_FLAP_PIN, 1000, 0, false, postboxFlapTopic, OUTPUT_NONE }, { POSTBOX_DOOR_PIN, 1000, 0, false, postboxDoorTopic, OUTPUT_NONE } }; #define NUM_INPUTS (sizeof(myInputs) / sizeof(myInputs[0])) int ledState = LOW; struct { const uint8_t pin; const unsigned long triggerTime_ms; bool isTriggered; unsigned long lastTrigger_ts; const char *listenTopic; // can be triggered by mqtt message as well } myOutputs[] = { { DOOR_BELL_BUZZER_PIN, 500, false, 0, doorBellTriggerSoundTopic}, { POSTBOX_LIGHT_PIN, 30000, false, 0, postboxLightTopic} }; #define NUM_OUTPUTS (sizeof(myOutputs) / sizeof(myOutputs[0])) uint8_t getOutputIndexByPin(uint8_t pin) { for (unsigned int i=0; i myInputs[i].quietAfterTriggerFor_ms) { if (!digitalRead(myInputs[i].pin)) { // input is triggered, compare with existing state if (myInputs[i].isTriggered) { // was triggered as well, could send a retrigger now Serial.print(F("Re-Trigger pin ")); Serial.println(myInputs[i].pin); sendMqttMessage(myInputs[i].topic, "re-trigger"); needToToggleLed = true; } else { // handle trigger now myInputs[i].isTriggered = true; // send Trigger Serial.print(F("Trigger pin ")); Serial.println(myInputs[i].pin); sendMqttMessage(myInputs[i].topic, "trigger"); needToToggleLed = true; if (myInputs[i].triggerOutputPin != OUTPUT_NONE) { switchOutputOn(getOutputIndexByPin(myInputs[i].triggerOutputPin)); } } myInputs[i].lastTrigger_ts = millis(); } else { // no trigger seen, if it was before, send a "released" message if (myInputs[i].isTriggered) { myInputs[i].isTriggered = false; myInputs[i].lastTrigger_ts = millis(); Serial.print(F("Released pin ")); Serial.println(myInputs[i].pin); sendMqttMessage(myInputs[i].topic, "release"); needToToggleLed = true; } } } } // if a LED toggle has been flagged : if ( needToToggleLed ) { // Toggle LED state : ledState = !ledState; digitalWrite(LED_PIN, ledState); } } void checkWebRequest() { // Wait for an incomming connection EthernetClient client = server.available(); // Do we have a client? if (!client) return; Serial.println(F("New client on web ingress")); // Read the request (we ignore the content in this example) while (client.available()) client.read(); // Allocate a temporary JsonDocument // Use arduinojson.org/v6/assistant to compute the capacity. StaticJsonDocument<500> doc; for (unsigned int pin = 0; pin < NUM_INPUTS; pin++) { doc[myInputs[pin].topic] = myInputs[pin].isTriggered ? "triggered" : "released"; } // Create the "digital" array JsonArray digitalValues = doc.createNestedArray("digitalPin"); for (unsigned int pin = 0; pin < 14; pin++) { // Read the digital input int value = digitalRead(pin); // Add the value at the end of the array digitalValues.add(value); } Serial.print(F("Sending: ")); serializeJson(doc, Serial); Serial.println(); // Write response headers client.println(F("HTTP/1.0 200 OK")); client.println(F("Content-Type: application/json")); client.println(F("Connection: close")); client.print(F("Content-Length: ")); // client.println(measureJsonPretty(doc)); client.println(measureJson(doc)); client.println(); // Write JSON document //serializeJsonPretty(doc, client); serializeJson(doc, client); // Disconnect client.stop(); } void loop() { // Client if (!mqttClient.connected()) { reconnect(); } mqttClient.loop(); checkInputSignals(); checkWebRequest(); checkOutputAutoOff(); }