/* Copyright (c) 2016 Christian Mollekopf This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "mailtransport.h" #include #include #include #include extern "C" { #include #include #include struct upload_status { int offset; const char *data; }; static size_t payload_source(void *ptr, size_t size, size_t nmemb, void *userp) { struct upload_status *upload_ctx = (struct upload_status *)userp; const char *data; if ((size == 0) || (nmemb == 0) || ((size*nmemb) < 1)) { return 0; } data = &upload_ctx->data[upload_ctx->offset]; if (data) { size_t len = strlen(data); if (len > size * nmemb) { len = size * nmemb; } fprintf(stdout, "read n bytes: %d\n", int(len)); memcpy(ptr, data, len); upload_ctx->offset += len; return len; } return 0; } static int progress_callback(void *clientp, curl_off_t dltotal, curl_off_t dlnow, curl_off_t ultotal, curl_off_t ulnow) { if (ultotal > 0) { auto progress = ulnow * 100.0/ ultotal; fprintf(stdout, "Upload progress: %d \n", int(progress)); } return 0; } static int debug_callback(CURL *handle, curl_infotype type, char *data, size_t size, void *userptr) { fprintf(stdout, "CURL_DEBUG: %s", data); return 0; } bool sendMessageCurl(const char *to[], int numTos, const char *cc[], int numCcs, const char *msg, bool useTls, const char* from, const char *username, const char *password, const char *server, bool verifyPeer, const QByteArray &cacert, QByteArray &errorMessage) { CURL *curl; CURLcode res = CURLE_OK; struct curl_slist *recipients = NULL; struct upload_status upload_ctx; upload_ctx.offset = 0; upload_ctx.data = msg; curl = curl_easy_init(); if(curl) { curl_easy_setopt(curl, CURLOPT_USERNAME, username); curl_easy_setopt(curl, CURLOPT_PASSWORD, password); curl_easy_setopt(curl, CURLOPT_URL, server); if (useTls) { curl_easy_setopt(curl, CURLOPT_USE_SSL, (long)CURLUSESSL_ALL); } if (!verifyPeer) { curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); } if (!cacert.isEmpty()) { curl_easy_setopt(curl, CURLOPT_CAINFO, cacert.constData()); } if (from) { curl_easy_setopt(curl, CURLOPT_MAIL_FROM, from); } for (int i = 0; i < numTos; i++) { recipients = curl_slist_append(recipients, to[i]); } for (int i = 0; i < numCcs; i++) { recipients = curl_slist_append(recipients, cc[i]); } curl_easy_setopt(curl, CURLOPT_MAIL_RCPT, recipients); /* We're using a callback function to specify the payload (the headers and * body of the message). You could just use the CURLOPT_READDATA option to * specify a FILE pointer to read from. */ curl_easy_setopt(curl, CURLOPT_READFUNCTION, payload_source); curl_easy_setopt(curl, CURLOPT_READDATA, &upload_ctx); curl_easy_setopt(curl, CURLOPT_UPLOAD, 1L); /* Since the traffic will be encrypted, it is very useful to turn on debug * information within libcurl to see what is happening during the transfer. */ // curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L); curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, debug_callback); // curl_easy_setopt(curl, CURLOPT_TIMEOUT, 1L); //Connection timeout of 40s curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 40L); curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0); curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, progress_callback); char errorBuffer[CURL_ERROR_SIZE]; curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorBuffer); res = curl_easy_perform(curl); if(res != CURLE_OK) { errorMessage += curl_easy_strerror(res); errorMessage += "; "; } long http_code = 0; curl_easy_getinfo (curl, CURLINFO_RESPONSE_CODE, &http_code); if (http_code == 200 && res != CURLE_ABORTED_BY_CALLBACK) { //Succeeded } else { errorMessage += errorBuffer; } curl_slist_free_all(recipients); curl_easy_cleanup(curl); return res == CURLE_OK; } return false; } }; bool MailTransport::sendMessage(const KMime::Message::Ptr &message, const QByteArray &server, const QByteArray &username, const QByteArray &password, const QByteArray &cacert, MailTransport::Options options) { QByteArray msg = message->encodedContent(); QByteArray from(message->from(true)->mailboxes().isEmpty() ? QByteArray() : message->from(true)->mailboxes().first().address()); QList toList; for (const auto &mb : message->to(true)->mailboxes()) { toList << mb.address(); } QList ccList; for (const auto &mb : message->cc(true)->mailboxes()) { ccList << mb.address(); } const bool verifyPeer = options.testFlag(VerifyPeers); const bool useTls = options.testFlag(UseTls); SinkLog() << "Sending message " << server << username << password << "CaCert: " << cacert << "Use tls: " << useTls << " Verify peer: " << verifyPeer; SinkTrace() << "Sending message " << msg; const int numTos = toList.size(); const char* to[numTos]; for (int i = 0; i < numTos; i++) { to[i] = toList.at(i); } const int numCcs = ccList.size(); const char* cc[numCcs]; for (int i = 0; i < numCcs; i++) { cc[i] = ccList.at(i); } //Because curl will fail with smtps, but it won't tell you why. auto serverAddress = server; serverAddress.replace("smtps://", "smtp://"); QByteArray errorMessage; auto ret = sendMessageCurl(to, numTos, cc, numCcs, msg, useTls, from.isEmpty() ? nullptr : from, username, password, serverAddress, verifyPeer, cacert, errorMessage); if (!ret) { SinkWarning() << "Failed to send message: " << errorMessage; } return ret; }