2023-12-22 20:29:15 +01:00
|
|
|
/*************************************************************************/
|
|
|
|
/* smtp_client.h */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* PANDEMONIUM ENGINE */
|
|
|
|
/* https://github.com/Relintai/pandemonium_engine */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Copyright (c) 2022-present Péter Magyar. */
|
|
|
|
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
|
|
|
|
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* */
|
|
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
|
|
/* a copy of this software and associated documentation files (the */
|
|
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
|
|
/* the following conditions: */
|
|
|
|
/* */
|
|
|
|
/* The above copyright notice and this permission notice shall be */
|
|
|
|
/* included in all copies or substantial portions of the Software. */
|
|
|
|
/* */
|
|
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
#include "smtp_client.h"
|
|
|
|
|
2023-12-23 00:50:09 +01:00
|
|
|
#include "core/bind/core_bind.h"
|
|
|
|
#include "core/io/ip.h"
|
|
|
|
#include "core/os/os.h"
|
|
|
|
|
|
|
|
#define SMTP_DEBUG_ENABLED 0
|
|
|
|
|
|
|
|
SMTPClient::AuthType SMTPClient::get_auth_type() const {
|
|
|
|
return _auth_type;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_auth_type(const AuthType value) {
|
|
|
|
_auth_type = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_server_address() const {
|
|
|
|
return _smtp_server_address;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_address(const String &value) {
|
|
|
|
_smtp_server_address = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SMTPClient::get_smtp_server_port() const {
|
|
|
|
return _smtp_server_port;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_port(const int value) {
|
|
|
|
_smtp_server_port = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_server_user_name() const {
|
|
|
|
return _smtp_server_user_name;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_user_name(const String &value) {
|
|
|
|
_smtp_server_user_name = value;
|
|
|
|
|
|
|
|
_smtp_server_user_name_base64 = _Marshalls::get_singleton()->utf8_to_base64(_smtp_server_user_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_server_password() const {
|
|
|
|
return _smtp_server_password;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_password(const String &value) {
|
|
|
|
_smtp_server_password = value;
|
|
|
|
|
|
|
|
_smtp_server_password_base64 = _Marshalls::get_singleton()->utf8_to_base64(_smtp_server_password);
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_server_sender_email_address() const {
|
|
|
|
return _smtp_server_sender_email_address;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_sender_email_address(const String &value) {
|
|
|
|
_smtp_server_sender_email_address = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_server_sender_name() const {
|
|
|
|
return _smtp_server_sender_name;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_server_sender_name(const String &value) {
|
|
|
|
_smtp_server_sender_name = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
String SMTPClient::get_smtp_client_id() const {
|
|
|
|
return _smtp_client_id;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_smtp_client_id(const String &value) {
|
|
|
|
_smtp_client_id = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SMTPClient::get_retries() const {
|
|
|
|
return _retries;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_retries(const int value) {
|
|
|
|
_retries = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SMTPClient::get_delay() const {
|
|
|
|
return _delay;
|
|
|
|
}
|
|
|
|
void SMTPClient::set_delay(const int value) {
|
|
|
|
_delay = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SMTPClient::get_use_threads() const {
|
|
|
|
#ifdef NO_THREADS
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
return _use_threads;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
void SMTPClient::set_use_threads(const bool value) {
|
|
|
|
_use_threads = value;
|
|
|
|
|
|
|
|
_apply_thread_setting();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTPClient::send_mail(const String &address, const String &subject, const String &message) {
|
|
|
|
MailData md;
|
|
|
|
md.address = address;
|
|
|
|
md.subject = subject;
|
|
|
|
md.message = message;
|
|
|
|
|
|
|
|
if (get_use_threads()) {
|
|
|
|
_mail_queue_lock.lock();
|
|
|
|
_mail_queue.push_back(md);
|
|
|
|
_mail_queue_lock.unlock();
|
|
|
|
|
|
|
|
_thread_semaphore.post();
|
|
|
|
} else {
|
|
|
|
_send_mail(md);
|
|
|
|
}
|
2023-12-22 20:29:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SMTPClient::SMTPClient() {
|
2023-12-23 00:50:09 +01:00
|
|
|
_auth_type = AUTH_TYPE_SMTPS;
|
|
|
|
|
|
|
|
_smtp_server_port = 465;
|
|
|
|
|
|
|
|
_retries = 3;
|
|
|
|
_delay = 200;
|
|
|
|
_use_threads = true;
|
|
|
|
|
|
|
|
_worker_thread = NULL;
|
|
|
|
_thread_running = false;
|
2023-12-22 20:29:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SMTPClient::~SMTPClient() {
|
|
|
|
}
|
|
|
|
|
2023-12-23 00:50:09 +01:00
|
|
|
void SMTPClient::_thread_main_loop(void *user_data) {
|
|
|
|
SMTPClient *self = (SMTPClient *)user_data;
|
|
|
|
|
|
|
|
while (self->_thread_running) {
|
|
|
|
self->_mail_queue_lock.lock();
|
|
|
|
|
|
|
|
if (self->_mail_queue.size() > 0) {
|
|
|
|
int index = self->_mail_queue.size() - 1;
|
|
|
|
MailData d = self->_mail_queue[index];
|
|
|
|
self->_mail_queue.remove(index);
|
|
|
|
|
|
|
|
self->_mail_queue_lock.unlock();
|
|
|
|
|
|
|
|
self->_send_mail(d);
|
|
|
|
} else {
|
|
|
|
self->_mail_queue_lock.lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->_mail_queue.size() == 0) {
|
|
|
|
self->_thread_semaphore.wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Error SMTPClient::_send_mail(const MailData &data) {
|
|
|
|
Error err = OK;
|
|
|
|
|
|
|
|
err = _connect();
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _wait_for_answer("");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_hello();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK && _auth_type == AUTH_TYPE_STARTTLS) {
|
|
|
|
err = _smtp_starttls();
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_hello();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_auth();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_mail_from();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_mail_to(data.address);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_mail_data(data.subject, data.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _smtp_quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
_disconnect();
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
if (err == OK) {
|
|
|
|
ERR_PRINT("Mail Sent");
|
|
|
|
} else {
|
|
|
|
ERR_PRINT("ERROR sending mail: " + itos(err));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
Error SMTPClient::_connect() {
|
|
|
|
Error error = OK;
|
|
|
|
|
|
|
|
if (_socket_tcp.is_valid()) {
|
|
|
|
return ERR_ALREADY_IN_USE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_socket_tcp.instance();
|
|
|
|
error = _socket_tcp->connect_to_host(_smtp_server_address, _smtp_server_port);
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("Connecting server...");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (error != OK) {
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("Connection failed.");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
IP_Address ip = IP::get_singleton()->resolve_hostname(_smtp_server_address);
|
|
|
|
error = _socket_tcp->connect_to_host(ip, _smtp_server_port);
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("Trying with IP");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < _retries; ++i) {
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("Try " + Streing::num(i) + " / " + String::num(_retries));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
StreamPeerTCP::Status status = _socket_tcp->get_status();
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("StreamPeerSSL::STATUS: " + itos(status));
|
|
|
|
#endif
|
|
|
|
if (status == StreamPeerTCP::STATUS_CONNECTED) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
_wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_socket_tcp->get_status() == StreamPeerTCP::STATUS_ERROR) {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_auth_type == AUTH_TYPE_SMTPS) {
|
|
|
|
_socket_ssl = Ref<StreamPeerSSL>(StreamPeerSSL::create());
|
|
|
|
_socket_ssl->connect_to_stream(_socket_tcp, true, _smtp_server_address);
|
|
|
|
|
|
|
|
for (int i = 0; i < _retries; ++i) {
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("SSL Try " + Streing::num(i) + " / " + String::num(_retries));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
StreamPeerSSL::Status status = _socket_ssl->get_status();
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("StreamPeerSSL::STATUS: " + itos(status));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (status == StreamPeerSSL::STATUS_CONNECTED) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
_wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_socket_ssl->get_status() == StreamPeerSSL::STATUS_ERROR) {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
void SMTPClient::_disconnect() {
|
|
|
|
if (_socket_ssl.is_valid()) {
|
|
|
|
_socket_ssl->disconnect_from_stream();
|
|
|
|
}
|
|
|
|
|
|
|
|
_socket_tcp->disconnect_from_host();
|
|
|
|
|
|
|
|
_socket_ssl.unref();
|
|
|
|
_socket_tcp.unref();
|
|
|
|
}
|
|
|
|
|
|
|
|
Error SMTPClient::_smtp_hello() {
|
|
|
|
Error err = _send("HELO " + _smtp_client_id);
|
|
|
|
_wait_for_answer("");
|
|
|
|
err = _send("EHLO " + _smtp_client_id);
|
|
|
|
err = _wait_for_answer("250");
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_starttls() {
|
|
|
|
Error err = _send("STARTTLS");
|
|
|
|
err = _wait_for_answer("220"); // 220 TLS go ahead
|
|
|
|
|
|
|
|
if (err != OK) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
_socket_ssl = Ref<StreamPeerSSL>(StreamPeerSSL::create());
|
|
|
|
|
|
|
|
_socket_ssl->connect_to_stream(_socket_tcp, true, _smtp_server_address);
|
|
|
|
|
|
|
|
for (int i = 0; i < _retries; ++i) {
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("SSL Try " + String::num(i) + " / " + String::num(_retries));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
StreamPeerSSL::Status status = _socket_ssl->get_status();
|
|
|
|
|
|
|
|
if (status == StreamPeerSSL::STATUS_ERROR) {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("StreamPeerSSL::STATUS: " + itos(status));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (status == StreamPeerSSL::STATUS_CONNECTED) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
_wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_socket_ssl->get_status() == StreamPeerSSL::STATUS_ERROR) {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_auth() {
|
|
|
|
Error err = _send("AUTH LOGIN");
|
|
|
|
err = _wait_for_answer("334");
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("_smtp_auth 1");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _send(_smtp_server_user_name_base64);
|
|
|
|
}
|
|
|
|
|
|
|
|
err = _wait_for_answer("334");
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("_smtp_auth 2");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _send(_smtp_server_password_base64);
|
|
|
|
}
|
|
|
|
|
|
|
|
err = _wait_for_answer("235");
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("_smtp_auth 3");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_mail_from() {
|
|
|
|
Error err = _send("MAIL FROM: <" + _smtp_server_sender_email_address + ">");
|
|
|
|
err = _wait_for_answer("250");
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_mail_to(const String &recipient) {
|
|
|
|
Error err = _send("RCPT TO: <" + recipient + ">");
|
|
|
|
err = _wait_for_answer("250");
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_mail_data(const String &subject, const String &message) {
|
|
|
|
Error err = _send("DATA");
|
|
|
|
|
|
|
|
err = _wait_for_answer("354");
|
|
|
|
|
|
|
|
if (err == OK) {
|
|
|
|
err = _send("FROM: " + _smtp_server_sender_name + " <" + _smtp_server_sender_email_address + ">");
|
|
|
|
//err = _wait_for_answer("250");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK && !subject.empty()) {
|
|
|
|
err = _send("SUBJECT: " + subject);
|
|
|
|
//err = _wait_for_answer("250");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == OK && !message.empty()) {
|
|
|
|
err = _send(message + "\r\n.\r\n");
|
|
|
|
//err = _wait_for_answer("250");
|
|
|
|
}
|
|
|
|
|
|
|
|
err = _wait_for_answer("250");
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_smtp_quit() {
|
|
|
|
return _send("QUIT");
|
|
|
|
}
|
|
|
|
|
|
|
|
Error SMTPClient::_send(const String &data) {
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("_send: " + data);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
String s = data + "\n";
|
|
|
|
Vector<uint8_t> v = s.to_utf8_buffer();
|
|
|
|
|
|
|
|
if (_socket_ssl.is_valid()) {
|
|
|
|
return _socket_ssl->put_data(v.ptr(), v.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_socket_tcp.is_valid()) {
|
|
|
|
return _socket_tcp->put_data(v.ptr(), v.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Error SMTPClient::_wait_for_answer(const String &expected_answer) {
|
|
|
|
String _read_buffer;
|
|
|
|
|
|
|
|
_wait();
|
|
|
|
|
|
|
|
for (int i = 0; i < _retries; ++i) {
|
|
|
|
if (_socket_ssl.is_valid()) {
|
|
|
|
_socket_ssl->poll();
|
|
|
|
}
|
|
|
|
|
|
|
|
int available_bytes = 0;
|
|
|
|
|
|
|
|
if (_socket_ssl.is_valid()) {
|
|
|
|
available_bytes = _socket_ssl->get_available_bytes();
|
|
|
|
} else if (_socket_tcp.is_valid()) {
|
|
|
|
available_bytes = _socket_tcp->get_available_bytes();
|
|
|
|
} else {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (available_bytes > 0) {
|
|
|
|
if (_socket_ssl.is_valid()) {
|
|
|
|
_read_buffer += _socket_ssl->get_utf8_string();
|
|
|
|
} else if (_socket_tcp.is_valid()) {
|
|
|
|
_read_buffer += _socket_tcp->get_available_bytes();
|
|
|
|
} else {
|
|
|
|
return ERR_CANT_CONNECT;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SMTP_DEBUG_ENABLED
|
|
|
|
ERR_PRINT("_wait_for_answer received: " + _read_buffer);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
_wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_read_buffer.empty()) {
|
|
|
|
return ERR_CANT_ACQUIRE_RESOURCE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_parse_answer_packet(_read_buffer, expected_answer) != OK) {
|
|
|
|
return FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
Error SMTPClient::_parse_answer_packet(const String &data, const String &expected_answer) {
|
|
|
|
if (expected_answer.empty()) {
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int slicecount = data.get_slice_count("\r\n");
|
|
|
|
|
|
|
|
if (slicecount <= 1) {
|
|
|
|
if (data.left(expected_answer.length()) == expected_answer) {
|
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
return FAILED;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
String ll = data.get_slice("\r\n", slicecount - 2);
|
|
|
|
if (ll.left(expected_answer.length()) == expected_answer) {
|
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
return FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTPClient::_wait() {
|
|
|
|
OS::get_singleton()->delay_usec(int64_t(_delay) * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTPClient::_apply_thread_setting() {
|
|
|
|
if (!is_inside_tree()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get_use_threads() || !_worker_thread) {
|
|
|
|
_thread_running = true;
|
|
|
|
_worker_thread = memnew(Thread);
|
|
|
|
_worker_thread->start(_thread_main_loop, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!get_use_threads() && _worker_thread) {
|
|
|
|
_thread_running = false;
|
|
|
|
_thread_semaphore.post();
|
|
|
|
_worker_thread->wait_to_finish();
|
|
|
|
memdelete(_worker_thread);
|
|
|
|
_worker_thread = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTPClient::_notification(const int what) {
|
|
|
|
switch (what) {
|
|
|
|
case NOTIFICATION_ENTER_TREE: {
|
|
|
|
if (get_use_threads()) {
|
|
|
|
_thread_running = true;
|
|
|
|
_worker_thread = memnew(Thread);
|
|
|
|
_worker_thread->start(_thread_main_loop, this);
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
case NOTIFICATION_EXIT_TREE: {
|
|
|
|
if (_worker_thread) {
|
|
|
|
_thread_running = false;
|
|
|
|
_thread_semaphore.post();
|
|
|
|
_worker_thread->wait_to_finish();
|
|
|
|
memdelete(_worker_thread);
|
|
|
|
_worker_thread = NULL;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-22 20:29:15 +01:00
|
|
|
void SMTPClient::_bind_methods() {
|
2023-12-23 00:50:09 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("get_auth_type"), &SMTPClient::get_auth_type);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_auth_type", "val"), &SMTPClient::set_auth_type);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "auth_type", PROPERTY_HINT_ENUM, "Plaintext,STARTTLS,SMTPS"), "set_auth_type", "get_auth_type");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_address"), &SMTPClient::get_smtp_server_address);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_address", "val"), &SMTPClient::set_smtp_server_address);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_server_address"), "set_smtp_server_address", "get_smtp_server_address");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_port"), &SMTPClient::get_smtp_server_port);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_port", "val"), &SMTPClient::set_smtp_server_port);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "smtp_server_port"), "set_smtp_server_port", "get_smtp_server_port");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_user_name"), &SMTPClient::get_smtp_server_user_name);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_user_name", "val"), &SMTPClient::set_smtp_server_user_name);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_server_user_name"), "set_smtp_server_user_name", "get_smtp_server_user_name");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_password"), &SMTPClient::get_smtp_server_password);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_password", "val"), &SMTPClient::set_smtp_server_password);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_server_password"), "set_smtp_server_password", "get_smtp_server_password");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_sender_email_address"), &SMTPClient::get_smtp_server_sender_email_address);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_sender_email_address", "val"), &SMTPClient::set_smtp_server_sender_email_address);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_server_sender_email_address"), "set_smtp_server_sender_email_address", "get_smtp_server_sender_email_address");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_server_sender_name"), &SMTPClient::get_smtp_server_sender_name);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_server_sender_name", "val"), &SMTPClient::set_smtp_server_sender_name);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_server_sender_name"), "set_smtp_server_sender_name", "get_smtp_server_sender_name");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_smtp_client_id"), &SMTPClient::get_smtp_client_id);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_smtp_client_id", "val"), &SMTPClient::set_smtp_client_id);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::STRING, "smtp_client_id"), "set_smtp_client_id", "get_smtp_client_id");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_retries"), &SMTPClient::get_retries);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_retries", "val"), &SMTPClient::set_retries);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "retries"), "set_retries", "get_retries");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_delay"), &SMTPClient::get_delay);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_delay", "val"), &SMTPClient::set_delay);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "get_delay"), "set_delay", "get_delay");
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_use_threads"), &SMTPClient::get_use_threads);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_use_threads", "val"), &SMTPClient::set_use_threads);
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "use_threads"), "set_use_threads", "get_use_threads");
|
|
|
|
|
2023-12-23 00:53:42 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("send_mail", "address", "subject", "message"), &SMTPClient::send_mail);
|
|
|
|
|
2023-12-23 00:50:09 +01:00
|
|
|
BIND_ENUM_CONSTANT(AUTH_TYPE_PLAINTEXT);
|
|
|
|
BIND_ENUM_CONSTANT(AUTH_TYPE_STARTTLS);
|
|
|
|
BIND_ENUM_CONSTANT(AUTH_TYPE_SMTPS);
|
2023-12-22 20:29:15 +01:00
|
|
|
}
|