mirror of
https://github.com/qbittorrent/qBittorrent.git
synced 2025-01-09 16:47:27 +03:00
545 lines
18 KiB
C++
545 lines
18 KiB
C++
/*
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
* Copyright (C) 2011 Christophe Dumez <chris@qbittorrent.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* In addition, as a special exception, the copyright holders give permission to
|
|
* link this program with the OpenSSL project's "OpenSSL" library (or with
|
|
* modified versions of it that use the same license as the "OpenSSL" library),
|
|
* and distribute the linked executables. You must obey the GNU General Public
|
|
* License in all respects for all of the code used other than "OpenSSL". If you
|
|
* modify file(s), you may extend this exception to your version of the file(s),
|
|
* but you are not obligated to do so. If you do not wish to do so, delete this
|
|
* exception statement from your version.
|
|
*/
|
|
|
|
/*
|
|
* This code is based on QxtSmtp from libqxt (http://libqxt.org)
|
|
*/
|
|
|
|
#include "smtp.h"
|
|
|
|
#include <QCryptographicHash>
|
|
#include <QDateTime>
|
|
#include <QDebug>
|
|
#include <QHostInfo>
|
|
#include <QStringList>
|
|
#include <QTextCodec>
|
|
|
|
#ifndef QT_NO_OPENSSL
|
|
#include <QSslSocket>
|
|
#else
|
|
#include <QTcpSocket>
|
|
#endif
|
|
|
|
#include "base/global.h"
|
|
#include "base/logger.h"
|
|
#include "base/preferences.h"
|
|
|
|
namespace
|
|
{
|
|
const short DEFAULT_PORT = 25;
|
|
#ifndef QT_NO_OPENSSL
|
|
const short DEFAULT_PORT_SSL = 465;
|
|
#endif
|
|
|
|
QByteArray hmacMD5(QByteArray key, const QByteArray &msg)
|
|
{
|
|
const int blockSize = 64; // HMAC-MD5 block size
|
|
if (key.length() > blockSize) // if key is longer than block size (64), reduce key length with MD5 compression
|
|
key = QCryptographicHash::hash(key, QCryptographicHash::Md5);
|
|
|
|
QByteArray innerPadding(blockSize, char(0x36)); // initialize inner padding with char "6"
|
|
QByteArray outerPadding(blockSize, char(0x5c)); // initialize outer padding with char "\"
|
|
// ascii characters 0x36 ("6") and 0x5c ("\") are selected because they have large
|
|
// Hamming distance (http://en.wikipedia.org/wiki/Hamming_distance)
|
|
|
|
for (int i = 0; i < key.length(); ++i) {
|
|
innerPadding[i] = innerPadding[i] ^ key.at(i); // XOR operation between every byte in key and innerpadding, of key length
|
|
outerPadding[i] = outerPadding[i] ^ key.at(i); // XOR operation between every byte in key and outerpadding, of key length
|
|
}
|
|
|
|
// result = hash ( outerPadding CONCAT hash ( innerPadding CONCAT baseString ) ).toBase64
|
|
QByteArray total = outerPadding;
|
|
QByteArray part = innerPadding;
|
|
part.append(msg);
|
|
total.append(QCryptographicHash::hash(part, QCryptographicHash::Md5));
|
|
return QCryptographicHash::hash(total, QCryptographicHash::Md5);
|
|
}
|
|
|
|
QByteArray determineFQDN()
|
|
{
|
|
QString hostname = QHostInfo::localHostName();
|
|
if (hostname.isEmpty())
|
|
hostname = "localhost";
|
|
|
|
return hostname.toLocal8Bit();
|
|
}
|
|
} // namespace
|
|
|
|
using namespace Net;
|
|
|
|
Smtp::Smtp(QObject *parent)
|
|
: QObject(parent)
|
|
, m_state(Init)
|
|
, m_useSsl(false)
|
|
, m_authType(AuthPlain)
|
|
{
|
|
static bool needToRegisterMetaType = true;
|
|
|
|
if (needToRegisterMetaType) {
|
|
qRegisterMetaType<QAbstractSocket::SocketError>();
|
|
needToRegisterMetaType = false;
|
|
}
|
|
|
|
#ifndef QT_NO_OPENSSL
|
|
m_socket = new QSslSocket(this);
|
|
#else
|
|
m_socket = new QTcpSocket(this);
|
|
#endif
|
|
|
|
connect(m_socket, &QIODevice::readyRead, this, &Smtp::readyRead);
|
|
connect(m_socket, &QAbstractSocket::disconnected, this, &QObject::deleteLater);
|
|
connect(m_socket, qOverload<QAbstractSocket::SocketError>(&QAbstractSocket::error)
|
|
, this, &Smtp::error);
|
|
|
|
// Test hmacMD5 function (http://www.faqs.org/rfcs/rfc2202.html)
|
|
Q_ASSERT(hmacMD5("Jefe", "what do ya want for nothing?").toHex()
|
|
== "750c783e6ab0b503eaa86e310a5db738");
|
|
Q_ASSERT(hmacMD5(QByteArray::fromHex("0b0b0b0b0b0b0b0b0b0b0b0b0b0b0b0b"), "Hi There").toHex()
|
|
== "9294727a3638bb1c13f48ef8158bfc9d");
|
|
}
|
|
|
|
Smtp::~Smtp()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
}
|
|
|
|
void Smtp::sendMail(const QString &from, const QString &to, const QString &subject, const QString &body)
|
|
{
|
|
const Preferences *const pref = Preferences::instance();
|
|
QTextCodec *latin1 = QTextCodec::codecForName("latin1");
|
|
m_message = "Date: " + getCurrentDateTime().toLatin1() + "\r\n"
|
|
+ encodeMimeHeader("From", from, latin1)
|
|
+ encodeMimeHeader("Subject", subject, latin1)
|
|
+ encodeMimeHeader("To", to, latin1)
|
|
+ "MIME-Version: 1.0\r\n"
|
|
+ "Content-Type: text/plain; charset=UTF-8\r\n"
|
|
+ "Content-Transfer-Encoding: base64\r\n"
|
|
+ "\r\n";
|
|
// Encode the body in base64
|
|
QString crlfBody = body;
|
|
const QByteArray b = crlfBody.replace("\n", "\r\n").toUtf8().toBase64();
|
|
const int ct = b.length();
|
|
for (int i = 0; i < ct; i += 78)
|
|
m_message += b.mid(i, 78);
|
|
m_from = from;
|
|
m_rcpt = to;
|
|
// Authentication
|
|
if (pref->getMailNotificationSMTPAuth()) {
|
|
m_username = pref->getMailNotificationSMTPUsername();
|
|
m_password = pref->getMailNotificationSMTPPassword();
|
|
}
|
|
|
|
// Connect to SMTP server
|
|
#ifndef QT_NO_OPENSSL
|
|
if (pref->getMailNotificationSMTPSSL()) {
|
|
m_socket->connectToHostEncrypted(pref->getMailNotificationSMTP(), DEFAULT_PORT_SSL);
|
|
m_useSsl = true;
|
|
}
|
|
else {
|
|
#endif
|
|
m_socket->connectToHost(pref->getMailNotificationSMTP(), DEFAULT_PORT);
|
|
m_useSsl = false;
|
|
#ifndef QT_NO_OPENSSL
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void Smtp::readyRead()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
// SMTP is line-oriented
|
|
m_buffer += m_socket->readAll();
|
|
while (true) {
|
|
const int pos = m_buffer.indexOf("\r\n");
|
|
if (pos < 0) return; // Loop exit condition
|
|
const QByteArray line = m_buffer.left(pos);
|
|
m_buffer = m_buffer.mid(pos + 2);
|
|
qDebug() << "Response line:" << line;
|
|
// Extract response code
|
|
const QByteArray code = line.left(3);
|
|
|
|
switch (m_state) {
|
|
case Init:
|
|
if (code[0] == '2') {
|
|
// The server may send a multiline greeting/INIT/220 response.
|
|
// We wait until it finishes.
|
|
if (line[3] != ' ')
|
|
break;
|
|
// Connection was successful
|
|
ehlo();
|
|
}
|
|
else {
|
|
logError(QLatin1String("Connection failed, unrecognized reply: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
case EhloSent:
|
|
case HeloSent:
|
|
case EhloGreetReceived:
|
|
parseEhloResponse(code, line[3] != ' ', line.mid(4));
|
|
break;
|
|
#ifndef QT_NO_OPENSSL
|
|
case StartTLSSent:
|
|
if (code == "220") {
|
|
m_socket->startClientEncryption();
|
|
ehlo();
|
|
}
|
|
else {
|
|
authenticate();
|
|
}
|
|
break;
|
|
#endif
|
|
case AuthRequestSent:
|
|
case AuthUsernameSent:
|
|
if (m_authType == AuthPlain) authPlain();
|
|
else if (m_authType == AuthLogin) authLogin();
|
|
else authCramMD5(line.mid(4));
|
|
break;
|
|
case AuthSent:
|
|
case Authenticated:
|
|
if (code[0] == '2') {
|
|
qDebug() << "Sending <mail from>...";
|
|
m_socket->write("mail from:<" + m_from.toLatin1() + ">\r\n");
|
|
m_socket->flush();
|
|
m_state = Rcpt;
|
|
}
|
|
else {
|
|
// Authentication failed!
|
|
logError(QLatin1String("Authentication failed, msg: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
case Rcpt:
|
|
if (code[0] == '2') {
|
|
m_socket->write("rcpt to:<" + m_rcpt.toLatin1() + ">\r\n");
|
|
m_socket->flush();
|
|
m_state = Data;
|
|
}
|
|
else {
|
|
logError(QLatin1String("<mail from> was rejected by server, msg: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
case Data:
|
|
if (code[0] == '2') {
|
|
m_socket->write("data\r\n");
|
|
m_socket->flush();
|
|
m_state = Body;
|
|
}
|
|
else {
|
|
logError(QLatin1String("<Rcpt to> was rejected by server, msg: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
case Body:
|
|
if (code[0] == '3') {
|
|
m_socket->write(m_message + "\r\n.\r\n");
|
|
m_socket->flush();
|
|
m_state = Quit;
|
|
}
|
|
else {
|
|
logError(QLatin1String("<data> was rejected by server, msg: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
case Quit:
|
|
if (code[0] == '2') {
|
|
m_socket->write("QUIT\r\n");
|
|
m_socket->flush();
|
|
// here, we just close.
|
|
m_state = Close;
|
|
}
|
|
else {
|
|
logError(QLatin1String("Message was rejected by the server, error: ") + line);
|
|
m_state = Close;
|
|
}
|
|
break;
|
|
default:
|
|
qDebug() << "Disconnecting from host";
|
|
m_socket->disconnectFromHost();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
QByteArray Smtp::encodeMimeHeader(const QString &key, const QString &value, const QTextCodec *latin1, const QByteArray &prefix)
|
|
{
|
|
QByteArray rv = "";
|
|
QByteArray line = key.toLatin1() + ": ";
|
|
if (!prefix.isEmpty()) line += prefix;
|
|
if (!value.contains("=?") && latin1->canEncode(value)) {
|
|
bool firstWord = true;
|
|
for (const QByteArray &word : asConst(value.toLatin1().split(' '))) {
|
|
if (line.size() > 78) {
|
|
rv = rv + line + "\r\n";
|
|
line.clear();
|
|
}
|
|
if (firstWord)
|
|
line += word;
|
|
else
|
|
line += ' ' + word;
|
|
firstWord = false;
|
|
}
|
|
}
|
|
else {
|
|
// The text cannot be losslessly encoded as Latin-1. Therefore, we
|
|
// must use base64 encoding.
|
|
const QByteArray utf8 = value.toUtf8();
|
|
// Use base64 encoding
|
|
const QByteArray base64 = utf8.toBase64();
|
|
const int ct = base64.length();
|
|
line += "=?utf-8?b?";
|
|
for (int i = 0; i < ct; i += 4) {
|
|
/*if (line.length() > 72) {
|
|
rv += line + "?\n\r";
|
|
line = " =?utf-8?b?";
|
|
}*/
|
|
line = line + base64.mid(i, 4);
|
|
}
|
|
line += "?="; // end encoded-word atom
|
|
}
|
|
return rv + line + "\r\n";
|
|
}
|
|
|
|
void Smtp::ehlo()
|
|
{
|
|
const QByteArray address = determineFQDN();
|
|
m_socket->write("ehlo " + address + "\r\n");
|
|
m_socket->flush();
|
|
m_state = EhloSent;
|
|
}
|
|
|
|
void Smtp::helo()
|
|
{
|
|
const QByteArray address = determineFQDN();
|
|
m_socket->write("helo " + address + "\r\n");
|
|
m_socket->flush();
|
|
m_state = HeloSent;
|
|
}
|
|
|
|
void Smtp::parseEhloResponse(const QByteArray &code, const bool continued, const QString &line)
|
|
{
|
|
if (code != "250") {
|
|
// Error
|
|
if (m_state == EhloSent) {
|
|
// try to send HELO instead of EHLO
|
|
qDebug() << "EHLO failed, trying HELO instead...";
|
|
helo();
|
|
}
|
|
else {
|
|
// Both EHLO and HELO failed, chances are this is NOT
|
|
// a SMTP server
|
|
logError("Both EHLO and HELO failed, msg: " + line);
|
|
m_state = Close;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (m_state != EhloGreetReceived) {
|
|
if (!continued) {
|
|
// greeting only, no extensions
|
|
qDebug() << "No extension";
|
|
m_state = EhloDone;
|
|
}
|
|
else {
|
|
// greeting followed by extensions
|
|
m_state = EhloGreetReceived;
|
|
qDebug() << "EHLO greet received";
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
qDebug() << Q_FUNC_INFO << "Supported extension: " << line.section(' ', 0, 0).toUpper()
|
|
<< line.section(' ', 1);
|
|
m_extensions[line.section(' ', 0, 0).toUpper()] = line.section(' ', 1);
|
|
if (!continued)
|
|
m_state = EhloDone;
|
|
}
|
|
|
|
if (m_state != EhloDone) return;
|
|
|
|
if (m_extensions.contains("STARTTLS") && m_useSsl) {
|
|
qDebug() << "STARTTLS";
|
|
startTLS();
|
|
}
|
|
else {
|
|
authenticate();
|
|
}
|
|
}
|
|
|
|
void Smtp::authenticate()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
if (!m_extensions.contains("AUTH") ||
|
|
m_username.isEmpty() || m_password.isEmpty()) {
|
|
// Skip authentication
|
|
qDebug() << "Skipping authentication...";
|
|
m_state = Authenticated;
|
|
// At this point the server will not send any response
|
|
// So fill the buffer with a fake one to pass the tests
|
|
// in readyRead()
|
|
m_buffer.push_front("250 QBT FAKE RESPONSE\r\n");
|
|
return;
|
|
}
|
|
// AUTH extension is supported, check which
|
|
// authentication modes are supported by
|
|
// the server
|
|
const QStringList auth = m_extensions["AUTH"].toUpper().split(' ', QString::SkipEmptyParts);
|
|
if (auth.contains("CRAM-MD5")) {
|
|
qDebug() << "Using CRAM-MD5 authentication...";
|
|
authCramMD5();
|
|
}
|
|
else if (auth.contains("PLAIN")) {
|
|
qDebug() << "Using PLAIN authentication...";
|
|
authPlain();
|
|
}
|
|
else if (auth.contains("LOGIN")) {
|
|
qDebug() << "Using LOGIN authentication...";
|
|
authLogin();
|
|
}
|
|
else {
|
|
// Skip authentication
|
|
logError("The SMTP server does not seem to support any of the authentications modes "
|
|
"we support [CRAM-MD5|PLAIN|LOGIN], skipping authentication, "
|
|
"knowing it is likely to fail... Server Auth Modes: " + auth.join('|'));
|
|
m_state = Authenticated;
|
|
// At this point the server will not send any response
|
|
// So fill the buffer with a fake one to pass the tests
|
|
// in readyRead()
|
|
m_buffer.push_front("250 QBT FAKE RESPONSE\r\n");
|
|
}
|
|
}
|
|
|
|
void Smtp::startTLS()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
#ifndef QT_NO_OPENSSL
|
|
m_socket->write("starttls\r\n");
|
|
m_socket->flush();
|
|
m_state = StartTLSSent;
|
|
#else
|
|
authenticate();
|
|
#endif
|
|
}
|
|
|
|
void Smtp::authCramMD5(const QByteArray &challenge)
|
|
{
|
|
if (m_state != AuthRequestSent) {
|
|
m_socket->write("auth cram-md5\r\n");
|
|
m_socket->flush();
|
|
m_authType = AuthCramMD5;
|
|
m_state = AuthRequestSent;
|
|
}
|
|
else {
|
|
const QByteArray response = m_username.toLatin1() + ' '
|
|
+ hmacMD5(m_password.toLatin1(), QByteArray::fromBase64(challenge)).toHex();
|
|
m_socket->write(response.toBase64() + "\r\n");
|
|
m_socket->flush();
|
|
m_state = AuthSent;
|
|
}
|
|
}
|
|
|
|
void Smtp::authPlain()
|
|
{
|
|
if (m_state != AuthRequestSent) {
|
|
m_authType = AuthPlain;
|
|
// Prepare Auth string
|
|
QByteArray auth;
|
|
auth += '\0';
|
|
auth += m_username.toLatin1();
|
|
qDebug() << "username: " << m_username.toLatin1();
|
|
auth += '\0';
|
|
auth += m_password.toLatin1();
|
|
qDebug() << "password: " << m_password.toLatin1();
|
|
// Send it
|
|
m_socket->write("auth plain " + auth.toBase64() + "\r\n");
|
|
m_socket->flush();
|
|
m_state = AuthSent;
|
|
}
|
|
}
|
|
|
|
void Smtp::authLogin()
|
|
{
|
|
if ((m_state != AuthRequestSent) && (m_state != AuthUsernameSent)) {
|
|
m_socket->write("auth login\r\n");
|
|
m_socket->flush();
|
|
m_authType = AuthLogin;
|
|
m_state = AuthRequestSent;
|
|
}
|
|
else if (m_state == AuthRequestSent) {
|
|
m_socket->write(m_username.toLatin1().toBase64() + "\r\n");
|
|
m_socket->flush();
|
|
m_state = AuthUsernameSent;
|
|
}
|
|
else {
|
|
m_socket->write(m_password.toLatin1().toBase64() + "\r\n");
|
|
m_socket->flush();
|
|
m_state = AuthSent;
|
|
}
|
|
}
|
|
|
|
void Smtp::logError(const QString &msg)
|
|
{
|
|
qDebug() << "Email Notification Error:" << msg;
|
|
Logger::instance()->addMessage(tr("Email Notification Error:") + ' ' + msg, Log::CRITICAL);
|
|
}
|
|
|
|
QString Smtp::getCurrentDateTime() const
|
|
{
|
|
// return date & time in the format specified in RFC 2822, section 3.3
|
|
const QDateTime nowDateTime = QDateTime::currentDateTime();
|
|
const QDate nowDate = nowDateTime.date();
|
|
const QLocale eng(QLocale::English);
|
|
|
|
const QString timeStr = nowDateTime.time().toString("HH:mm:ss");
|
|
const QString weekDayStr = eng.dayName(nowDate.dayOfWeek(), QLocale::ShortFormat);
|
|
const QString dayStr = QString::number(nowDate.day());
|
|
const QString monthStr = eng.monthName(nowDate.month(), QLocale::ShortFormat);
|
|
const QString yearStr = QString::number(nowDate.year());
|
|
|
|
QDateTime tmp = nowDateTime;
|
|
tmp.setTimeSpec(Qt::UTC);
|
|
const int timeOffsetHour = nowDateTime.secsTo(tmp) / 3600;
|
|
const int timeOffsetMin = nowDateTime.secsTo(tmp) / 60 - (60 * timeOffsetHour);
|
|
const int timeOffset = timeOffsetHour * 100 + timeOffsetMin;
|
|
// buf size = 11 to avoid format truncation warnings from snprintf
|
|
char buf[11] = {0};
|
|
std::snprintf(buf, sizeof(buf), "%+05d", timeOffset);
|
|
const QString timeOffsetStr = buf;
|
|
|
|
const QString ret = weekDayStr + ", " + dayStr + ' ' + monthStr + ' ' + yearStr + ' ' + timeStr + ' ' + timeOffsetStr;
|
|
return ret;
|
|
}
|
|
|
|
void Smtp::error(QAbstractSocket::SocketError socketError)
|
|
{
|
|
// Getting a remote host closed error is apparently normal, even when successfully sending
|
|
// an email
|
|
if (socketError != QAbstractSocket::RemoteHostClosedError)
|
|
logError(m_socket->errorString());
|
|
}
|