mirror of
https://github.com/arendst/Tasmota.git
synced 2025-05-13 07:58:39 +00:00

* `Sendmail` upgraded to ESP-Mail-Client v3.4.9 from v1.2.0, using BearSSL instead of MbedTLS * Fix compilation on ESP8266 * Fix compilation * fix compilation
220 lines
6.5 KiB
C++
220 lines
6.5 KiB
C++
/**
|
|
* Created by K. Suwatchai (Mobizt)
|
|
*
|
|
* Email: suwatchai@outlook.com
|
|
*
|
|
* Github: https://github.com/mobizt/ESP-Mail-Client
|
|
*
|
|
* Copyright (c) 2023 mobizt
|
|
*/
|
|
|
|
// This example shows how to receive mailbox updates by sending the custom IMAP command IDLE.
|
|
|
|
/** Note for library update from v2.x.x to v3.x.x.
|
|
*
|
|
* Struct data names changed
|
|
*
|
|
* "ESP_Mail_Session" changes to "Session_Config"
|
|
* "IMAP_Config" changes to "IMAP_Data"
|
|
*
|
|
* Changes in the examples
|
|
*
|
|
* ESP_Mail_Session session;
|
|
* to
|
|
* Session_Config config;
|
|
*
|
|
* IMAP_Config config;
|
|
* to
|
|
* IMAP_Data imap_data;
|
|
*/
|
|
|
|
/** For ESP8266, with BearSSL WiFi Client
|
|
* The memory reserved for completed valid SSL response from IMAP is 16 kbytes which
|
|
* may cause your device out of memory reset in case the memory
|
|
* allocation error.
|
|
*/
|
|
|
|
#include <Arduino.h>
|
|
#if defined(ESP32) || defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
#include <WiFi.h>
|
|
#elif defined(ESP8266)
|
|
#include <ESP8266WiFi.h>
|
|
#elif __has_include(<WiFiNINA.h>)
|
|
#include <WiFiNINA.h>
|
|
#elif __has_include(<WiFi101.h>)
|
|
#include <WiFi101.h>
|
|
#elif __has_include(<WiFiS3.h>)
|
|
#include <WiFiS3.h>
|
|
#endif
|
|
|
|
#include <ESP_Mail_Client.h>
|
|
|
|
// To use only IMAP functions, you can exclude the SMTP from compilation, see ESP_Mail_FS.h.
|
|
|
|
#define WIFI_SSID "<ssid>"
|
|
#define WIFI_PASSWORD "<password>"
|
|
|
|
/** For Gmail, IMAP option should be enabled. https://support.google.com/mail/answer/7126229?hl=en
|
|
* and also https://accounts.google.com/b/0/DisplayUnlockCaptcha
|
|
*
|
|
* Some Gmail user still not able to sign in using account password even above options were set up,
|
|
* for this case, use "App Password" to sign in instead.
|
|
* About Gmail "App Password", go to https://support.google.com/accounts/answer/185833?hl=en
|
|
*
|
|
* For Yahoo mail, log in to your yahoo mail in web browser and generate app password by go to
|
|
* https://login.yahoo.com/account/security/app-passwords/add/confirm?src=noSrc
|
|
*
|
|
* To use Gmai and Yahoo's App Password to sign in, define the AUTHOR_PASSWORD with your App Password
|
|
* and AUTHOR_EMAIL with your account email.
|
|
*/
|
|
|
|
/* The imap host name e.g. imap.gmail.com for GMail or outlook.office365.com for Outlook */
|
|
#define IMAP_HOST "<host>"
|
|
|
|
/** The imap port e.g.
|
|
* 143 or esp_mail_imap_port_143
|
|
* 993 or esp_mail_imap_port_993
|
|
*/
|
|
#define IMAP_PORT 993
|
|
|
|
/* The log in credentials */
|
|
#define AUTHOR_EMAIL "<email>"
|
|
#define AUTHOR_PASSWORD "<password>"
|
|
|
|
/* Declare the global used IMAPSession object for IMAP transport */
|
|
IMAPSession imap;
|
|
|
|
unsigned long lastIdleTerminatedMillis = 0;
|
|
|
|
#if defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
WiFiMulti multi;
|
|
#endif
|
|
|
|
void customCommandCallback(IMAP_Response res)
|
|
{
|
|
// The server responses will included tagged and/or untagged data.
|
|
|
|
// Tagged data is the status which begins with command identifier (tag) i.e. "A01" in this case.
|
|
// Tagged status responses included OK, NO, BAD, PREAUTH and BYE.
|
|
|
|
// Untagged data is the information or result of the request which begins with *
|
|
|
|
// When you send multiple commands with different tag simultaneously,
|
|
// tag will be used as command identifier.
|
|
|
|
MailClient.printf("> C: TAG %s\n", res.tag.c_str());
|
|
MailClient.printf("< S: %s\n", res.text.c_str());
|
|
|
|
if (res.completed)
|
|
{
|
|
MailClient.printf("> C: Response finished with status %s\n\n", res.status.c_str());
|
|
}
|
|
}
|
|
|
|
void setup()
|
|
{
|
|
|
|
Serial.begin(115200);
|
|
|
|
#if defined(ARDUINO_ARCH_SAMD)
|
|
while (!Serial)
|
|
;
|
|
#endif
|
|
|
|
Serial.println();
|
|
|
|
#if defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
multi.addAP(WIFI_SSID, WIFI_PASSWORD);
|
|
multi.run();
|
|
#else
|
|
WiFi.begin(WIFI_SSID, WIFI_PASSWORD);
|
|
#endif
|
|
|
|
Serial.print("Connecting to Wi-Fi");
|
|
|
|
#if defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
unsigned long ms = millis();
|
|
#endif
|
|
|
|
while (WiFi.status() != WL_CONNECTED)
|
|
{
|
|
Serial.print(".");
|
|
delay(300);
|
|
#if defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
if (millis() - ms > 10000)
|
|
break;
|
|
#endif
|
|
}
|
|
Serial.println();
|
|
Serial.print("Connected with IP: ");
|
|
Serial.println(WiFi.localIP());
|
|
Serial.println();
|
|
|
|
/* Set the network reconnection option */
|
|
MailClient.networkReconnect(true);
|
|
|
|
// The WiFi credentials are required for Pico W
|
|
// due to it does not have reconnect feature.
|
|
#if defined(ARDUINO_RASPBERRY_PI_PICO_W)
|
|
MailClient.clearAP();
|
|
MailClient.addAP(WIFI_SSID, WIFI_PASSWORD);
|
|
#endif
|
|
|
|
/* Declare the Session_Config for user defined session credentials */
|
|
Session_Config config;
|
|
|
|
/* Set the session config */
|
|
config.server.host_name = IMAP_HOST;
|
|
config.server.port = IMAP_PORT;
|
|
|
|
/* Connect to the server */
|
|
if (!imap.customConnect(&config, customCommandCallback, F("A01") /* tag */))
|
|
return;
|
|
|
|
if (imap.isAuthenticated())
|
|
Serial.println("\nSuccessfully logged in.\n");
|
|
else
|
|
Serial.println("\nConnected with no Auth.\n");
|
|
|
|
String cmd = F("LOGIN ");
|
|
cmd += AUTHOR_EMAIL;
|
|
cmd += F(" ");
|
|
cmd += AUTHOR_PASSWORD;
|
|
|
|
// You can also assign tag to the begining of the command e.g. "A01 FETCH 1 UID"
|
|
// Do not assign tag to command when you assign tag to the last parameter of function.
|
|
|
|
imap.sendCustomCommand(cmd, customCommandCallback, F("A02") /* tag */);
|
|
|
|
imap.sendCustomCommand(F("SELECT \"INBOX\""), customCommandCallback, F("A03") /* tag */);
|
|
|
|
imap.sendCustomCommand(F("LIST \"\" *"), customCommandCallback, F("A04") /* tag */);
|
|
}
|
|
|
|
void loop()
|
|
{
|
|
|
|
if (imap.connected())
|
|
{
|
|
if (!imap.sendCustomCommand(F("IDLE"), customCommandCallback, F("A05") /* tag */))
|
|
{
|
|
// If error, need to re-connect if imap.connected() returns false and re-log in again
|
|
Serial.println("\nTCP connection closed!");
|
|
return;
|
|
}
|
|
|
|
if (millis() - lastIdleTerminatedMillis > 20 * 60 * 1000) // terminate the IDLE every 20 min.
|
|
{
|
|
lastIdleTerminatedMillis = millis();
|
|
imap.sendCustomCommand(F("DONE"), customCommandCallback, "" /* don't include tag for DONE command */);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* The server MAY consider a client inactive if it has an IDLE command
|
|
* running, and if such a server has an inactivity timeout it MAY log
|
|
* the client off implicitly at the end of its timeout period. Because
|
|
* of that, clients using IDLE are advised to terminate the IDLE and
|
|
* re-issue it at least every 29 minutes to avoid being logged off.
|
|
*/ |