/*************************************************************************** sieve.cpp - description ------------------- begin : Thu Dec 20 18:47:08 EST 2001 copyright : (C) 2001 by Hamish Rodda email : meddie@yoyo.cc.monash.edu.au ***************************************************************************/ /*************************************************************************** * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License version 2 as * * published by the Free Software Foundation. * * * ***************************************************************************/ /** * Portions adapted from the SMTP ioslave. * Copyright (c) 2000, 2001 Alex Zepeda * Copyright (c) 2001 Michael Häckel * All rights reserved. * * Policy: the function where the error occurs calls error(). A result of * false, where it signifies an error, thus doesn't need to call error() itself. */ #ifdef HAVE_CONFIG_H # include #endif extern "C" { #include } #include "sieve.h" #include #include #include #include #include #include #include #include #include #include using std::exit; #include #include #include static const int debugArea = 7122; static inline #ifdef NDEBUG kndbgstream ksDebug() { return kdDebug( debugArea ); } kndbgstream ksDebug( bool cond ) { return kdDebug( cond, debugArea ); } #else kdbgstream ksDebug() { return kdDebug( debugArea ); } kdbgstream ksDebug( bool cond ) { return kdDebug( cond, debugArea ); } #endif #define SIEVE_DEFAULT_PORT 2000 static sasl_callback_t callbacks[] = { { SASL_CB_ECHOPROMPT, NULL, NULL }, { SASL_CB_NOECHOPROMPT, NULL, NULL }, { SASL_CB_GETREALM, NULL, NULL }, { SASL_CB_USER, NULL, NULL }, { SASL_CB_AUTHNAME, NULL, NULL }, { SASL_CB_PASS, NULL, NULL }, { SASL_CB_CANON_USER, NULL, NULL }, { SASL_CB_LIST_END, NULL, NULL } }; static const unsigned int SIEVE_DEFAULT_RECIEVE_BUFFER = 512; using namespace KIO; extern "C" { KDE_EXPORT int kdemain(int argc, char **argv) { KInstance instance("kio_sieve" ); ksDebug() << "*** Starting kio_sieve " << endl; if (argc != 4) { ksDebug() << "Usage: kio_sieve protocol domain-socket1 domain-socket2" << endl; exit(-1); } if ( sasl_client_init( NULL ) != SASL_OK ) { fprintf(stderr, "SASL library initialization failed!\n"); ::exit (-1); } kio_sieveProtocol slave(argv[2], argv[3]); slave.dispatchLoop(); sasl_done(); ksDebug() << "*** kio_sieve Done" << endl; return 0; } } /* ---------------------------------------------------------------------------------- */ kio_sieveResponse::kio_sieveResponse() { clear(); } /* ---------------------------------------------------------------------------------- */ const uint& kio_sieveResponse::getType() const { return rType; } /* ---------------------------------------------------------------------------------- */ const uint kio_sieveResponse::getQuantity() const { return quantity; } /* ---------------------------------------------------------------------------------- */ const TQCString& kio_sieveResponse::getAction() const { return key; } /* ---------------------------------------------------------------------------------- */ const TQCString& kio_sieveResponse::getKey() const { return key; } /* ---------------------------------------------------------------------------------- */ const TQCString& kio_sieveResponse::getVal() const { return val; } /* ---------------------------------------------------------------------------------- */ const TQCString& kio_sieveResponse::getExtra() const { return extra; } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::setQuantity(const uint& newTQty) { rType = TQUANTITY; quantity = newTQty; } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::setAction(const TQCString& newAction) { rType = ACTION; key = newAction.copy(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::setKey(const TQCString& newKey) { rType = KEY_VAL_PAIR; key = newKey.copy(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::setVal(const TQCString& newVal) { val = newVal.copy(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::setExtra(const TQCString& newExtra) { extra = newExtra.copy(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveResponse::clear() { rType = NONE; extra = key = val = TQCString(""); quantity = 0; } /* ---------------------------------------------------------------------------------- */ kio_sieveProtocol::kio_sieveProtocol(const TQCString &pool_socket, const TQCString &app_socket) : TCPSlaveBase( SIEVE_DEFAULT_PORT, "sieve", pool_socket, app_socket, false) , m_connMode(NORMAL) , m_supportsTLS(false) , m_shouldBeConnected(false) , m_allowUnencrypted(false) { } /* ---------------------------------------------------------------------------------- */ kio_sieveProtocol::~kio_sieveProtocol() { if ( isConnectionValid() ) disconnect(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::setHost (const TQString &host, int port, const TQString &user, const TQString &pass) { if ( isConnectionValid() && ( m_sServer != host || m_iPort != port || m_sUser != user || m_sPass != pass ) ) { disconnect(); } m_sServer = host; m_iPort = port ? port : m_iDefaultPort; m_sUser = user; m_sPass = pass; m_supportsTLS = false; } /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::openConnection() { m_connMode = CONNECTION_ORIENTED; connect(); } bool kio_sieveProtocol::parseCapabilities(bool requestCapabilities/* = false*/) { ksDebug() << k_funcinfo << endl; // Setup... bool ret = false; if (requestCapabilities) { sendData("CAPABILITY"); } while (receiveData()) { ksDebug() << "Looping receive" << endl; if (r.getType() == kio_sieveResponse::ACTION) { if ( r.getAction().contains("ok", false) != -1 ) { ksDebug() << "Sieve server ready & awaiting authentication." << endl; break; } else ksDebug() << "Unknown action " << r.getAction() << "." << endl; } else if (r.getKey() == "IMPLEMENTATION") { if (r.getVal().contains("sieve", false) != -1) { ksDebug() << "Connected to Sieve server: " << r.getVal() << endl; ret = true; setMetaData("implementation", r.getVal()); m_implementation = r.getVal(); } } else if (r.getKey() == "SASL") { // Save list of available SASL methods m_sasl_caps = TQStringList::split(' ', r.getVal()); ksDebug() << "Server SASL authentication methods: " << m_sasl_caps.join(", ") << endl; setMetaData("saslMethods", r.getVal()); } else if (r.getKey() == "SIEVE") { // Save script capabilities; report back as meta data: ksDebug() << "Server script capabilities: " << TQStringList::split(' ', r.getVal()).join(", ") << endl; setMetaData("sieveExtensions", r.getVal()); } else if (r.getKey() == "STARTTLS") { // The server supports TLS ksDebug() << "Server supports TLS" << endl; m_supportsTLS = true; setMetaData("tlsSupported", "true"); } else { ksDebug() << "Unrecognised key." << endl; } } if (!m_supportsTLS) { setMetaData("tlsSupported", "false"); } return ret; } /* ---------------------------------------------------------------------------------- */ /** * Checks if connection parameters have changed. * If it it, close the current connection */ void kio_sieveProtocol::changeCheck( const KURL &url ) { TQString auth; if (!metaData("sasl").isEmpty()) auth = metaData("sasl").upper(); else { TQString query = url.query(); if ( query.startsWith("?") ) query.remove( 0, 1 ); TQStringList q = TQStringList::split( ",", query ); TQStringList::iterator it; for ( it = q.begin(); it != q.end(); ++it ) { if ( TQString( (*it).section('=',0,0) ).lower() == "x-mech" ) { auth = TQString( (*it).section('=',1) ).upper(); break; } } } ksDebug() << "auth: " << auth << " m_sAuth: " << m_sAuth << endl; if ( m_sAuth != auth ) { m_sAuth = auth; if ( isConnectionValid() ) disconnect(); } // For TLS, only disconnect if we are unencrypted and are // no longer allowed (otherwise, it's still fine): const bool allowUnencryptedNow = url.queryItem("x-allow-unencrypted") == "true" ; if ( m_allowUnencrypted && !allowUnencryptedNow ) if ( isConnectionValid() ) disconnect(); m_allowUnencrypted = allowUnencryptedNow; } /* ---------------------------------------------------------------------------------- */ /** * Connects to the server. * returns false and calls error() if an error occurred. */ bool kio_sieveProtocol::connect(bool useTLSIfAvailable) { ksDebug() << k_funcinfo << endl; if (isConnectionValid()) return true; infoMessage(i18n("Connecting to %1...").tqarg( m_sServer)); if (m_connMode == CONNECTION_ORIENTED && m_shouldBeConnected) { error(ERR_CONNECTION_BROKEN, i18n("The connection to the server was lost.")); return false; } setBlockConnection(true); if (!connectToHost(m_sServer, m_iPort, true)) { return false; } if (!parseCapabilities()) { closeDescriptor(); error(ERR_UNSUPPORTED_PROTOCOL, i18n("Server identification failed.")); return false; } // Attempt to start TLS if ( !m_allowUnencrypted && !canUseTLS() ) { error( ERR_SLAVE_DEFINED, i18n("Can not use TLS. Please enable TLS in the KDE cryptography setting.") ); disconnect(); return false; } if ( !m_allowUnencrypted && useTLSIfAvailable && canUseTLS() && !m_supportsTLS && messageBox( WarningContinueCancel, i18n("TLS encryption was requested, but your Sieve server does not advertise TLS in its capabilities.\n" "You can choose to try to initiate TLS negotiations nonetheless, or cancel the operation."), i18n("Server Does Not Advertise TLS"), i18n("&Start TLS nonetheless"), i18n("&Cancel") ) != KMessageBox::Continue ) { error( ERR_USER_CANCELED, i18n("TLS encryption requested, but not supported by server.") ); disconnect(); return false; } // FIXME find a test server and test that this works if (useTLSIfAvailable && canUseTLS()) { sendData("STARTTLS"); if (operationSuccessful()) { ksDebug() << "TLS has been accepted. Starting TLS..." << endl << "WARNING this is untested and may fail." << endl; int retval = startTLS(); if (retval == 1) { ksDebug() << "TLS enabled successfully." << endl; // reparse capabilities: parseCapabilities( requestCapabilitiesAfterStartTLS() ); } else { ksDebug() << "TLS initiation failed, code " << retval << endl; if ( m_allowUnencrypted ) { disconnect(true); return connect(false); } if ( retval != -3 ) messageBox( Information, i18n("Your Sieve server claims to support TLS, " "but negotiation was unsuccessful."), i18n("Connection Failed") ); disconnect(true); return false; } } else if ( !m_allowUnencrypted ) { ksDebug() << "Server incapable of TLS." << endl; disconnect(); error( ERR_SLAVE_DEFINED, i18n("The server does not seem to support TLS. " "Disable TLS if you want to connect without encryption.") ); return false; } else ksDebug() << "Server incapable of TLS. Transmitted documents will be unencrypted." << endl; } else ksDebug() << "We are incapable of TLS. Transmitted documents will be unencrypted." << endl; assert( m_allowUnencrypted || usingTLS() ); infoMessage(i18n("Authenticating user...")); if (!authenticate()) { disconnect(); error(ERR_COULD_NOT_AUTHENTICATE, i18n("Authentication failed.")); return false; } m_shouldBeConnected = true; return true; } /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::closeConnection() { m_connMode = CONNECTION_ORIENTED; disconnect(); } /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::disconnect(bool forcibly) { if (!forcibly) { sendData("LOGOUT"); // This crashes under certain conditions as described in // http://intevation.de/roundup/kolab/issue2442 // Fixing KIO::TCPSlaveBase::atEnd() for !fd would also work but 3.x is on life support. //if (!operationSuccessful()) // ksDebug() << "Server did not logout cleanly." << endl; } closeDescriptor(); m_shouldBeConnected = false; } /* ---------------------------------------------------------------------------------- */ /*void kio_sieveProtocol::slave_status() { slaveStatus(isConnectionValid() ? m_sServer : "", isConnectionValid()); finished(); }*/ /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::special(const TQByteArray &data) { int tmp; TQDataStream stream(data, IO_ReadOnly); KURL url; stream >> tmp; switch (tmp) { case 1: stream >> url; if (!activate(url)) return; break; case 2: if (!deactivate()) return; break; case 3: parseCapabilities(true); break; } infoMessage(i18n("Done.")); finished(); } /* ---------------------------------------------------------------------------------- */ bool kio_sieveProtocol::activate(const KURL& url) { changeCheck( url ); if (!connect()) return false; infoMessage(i18n("Activating script...")); TQString filename = url.fileName(false); if (filename.isEmpty()) { error(ERR_DOES_NOT_EXIST, url.prettyURL()); return false; } if (!sendData("SETACTIVE \"" + filename.utf8() + "\"")) return false; if (operationSuccessful()) { ksDebug() << "Script activation complete." << endl; return true; } else { error(ERR_INTERNAL_SERVER, i18n("There was an error activating the script.")); return false; } } /* ---------------------------------------------------------------------------------- */ bool kio_sieveProtocol::deactivate() { if (!connect()) return false; if (!sendData("SETACTIVE \"\"")) return false; if (operationSuccessful()) { ksDebug() << "Script deactivation complete." << endl; return true; } else { error(ERR_INTERNAL_SERVER, i18n("There was an error deactivating the script.")); return false; } } static void append_lf2crlf( TQByteArray & out, const TQByteArray & in ) { if ( in.isEmpty() ) return; const unsigned int oldOutSize = out.size(); out.resize( oldOutSize + 2 * in.size() ); const char * s = in.begin(); const char * const end = in.end(); char * d = out.begin() + oldOutSize; char last = '\0'; while ( s < end ) { if ( *s == '\n' && last != '\r' ) *d++ = '\r'; *d++ = last = *s++; } out.resize( d - out.begin() ); } void kio_sieveProtocol::put(const KURL& url, int /*permissions*/, bool /*overwrite*/, bool /*resume*/) { changeCheck( url ); if (!connect()) return; infoMessage(i18n("Sending data...")); TQString filename = url.fileName(false); if (filename.isEmpty()) { error(ERR_MALFORMED_URL, url.prettyURL()); return; } TQByteArray data; for (;;) { dataReq(); TQByteArray buffer; const int newSize = readData(buffer); append_lf2crlf( data, buffer ); if ( newSize < 0 ) { // read error: network in unknown state so disconnect error(ERR_COULD_NOT_READ, i18n("KIO data supply error.")); return; } if ( newSize == 0 ) break; } // script size int bufLen = (int)data.size(); totalSize(bufLen); // timsieved 1.1.0: // C: HAVESPACE "rejected" 74 // S: NO "Number expected" // C: HAVESPACE 74 // S: NO "Missing script name" // S: HAVESPACE "rejected" "74" // C: NO "Number expected" // => broken, we can't use it :-( // (will be fixed in Cyrus 2.1.10) #ifndef HAVE_BROKEN_TIMSIEVED // first, check quota (it's a SHOULD in draft std) if (!sendData("HAVESPACE \"" + filename.utf8() + "\" " + TQCString().setNum( bufLen ))) return; if (!operationSuccessful()) { error(ERR_DISK_FULL, i18n("Quota exceeded")); return; } #endif if (!sendData("PUTSCRIPT \"" + filename.utf8() + "\" {" + TQCString().setNum( bufLen ) + "+}")) return; // atEnd() lies so the code below doesn't work. /*if (!atEnd()) { // We are not expecting any data here, so if the server has responded // with anything but OK we treat it as an error. char * buf = new char[2]; while (!atEnd()) { ksDebug() << "Reading..." << endl; read(buf, 1); ksDebug() << "Trailing [" << buf[0] << "]" << endl; } ksDebug() << "End of data." << endl; delete[] buf; if (!operationSuccessful()) { error(ERR_UNSUPPORTED_PROTOCOL, i18n("A protocol error occurred " "while trying to negotiate script uploading.\n" "The server responded:\n%1") .tqarg(r.getAction().right(r.getAction().length() - 3))); return; } }*/ // upload data to the server if (write(data, bufLen) != bufLen) { error(ERR_COULD_NOT_WRITE, i18n("Network error.")); disconnect(true); return; } // finishing CR/LF if (!sendData("")) return; processedSize(bufLen); infoMessage(i18n("Verifying upload completion...")); if (operationSuccessful()) ksDebug() << "Script upload complete." << endl; else { /* The managesieve server parses received scripts and rejects * scripts which are not syntactically correct. Here we expect * to receive a message detailing the error (only the first * error is reported. */ if (r.getAction().length() > 3) { // make a copy of the extra info TQCString extra = r.getAction().right(r.getAction().length() - 3); // send the extra message off for re-processing receiveData(false, &extra); if (r.getType() == kio_sieveResponse::TQUANTITY) { // length of the error message uint len = r.getQuantity(); TQCString errmsg(len + 1); read(errmsg.data(), len); error(ERR_INTERNAL_SERVER, i18n("The script did not upload successfully.\n" "This is probably due to errors in the script.\n" "The server responded:\n%1").tqarg(TQString(errmsg))); // clear the rest of the incoming data receiveData(); } else if (r.getType() == kio_sieveResponse::KEY_VAL_PAIR) { error(ERR_INTERNAL_SERVER, i18n("The script did not upload successfully.\n" "This is probably due to errors in the script.\n" "The server responded:\n%1").tqarg(TQString(r.getKey()))); } else error(ERR_INTERNAL_SERVER, i18n("The script did not upload successfully.\n" "The script may contain errors.")); } else error(ERR_INTERNAL_SERVER, i18n("The script did not upload successfully.\n" "The script may contain errors.")); } //if ( permissions != -1 ) // chmod( url, permissions ); infoMessage(i18n("Done.")); finished(); } static void inplace_crlf2lf( TQByteArray & in ) { if ( in.isEmpty() ) return; TQByteArray & out = in; // inplace const char * s = in.begin(); const char * const end = in.end(); char * d = out.begin(); char last = '\0'; while ( s < end ) { if ( *s == '\n' && last == '\r' ) --d; *d++ = last = *s++; } out.resize( d - out.begin() ); } /* ---------------------------------------------------------------------------------- */ void kio_sieveProtocol::get(const KURL& url) { changeCheck( url ); if (!connect()) return; infoMessage(i18n("Retrieving data...")); TQString filename = url.fileName(false); if (filename.isEmpty()) { error(ERR_MALFORMED_URL, url.prettyURL()); return; } //SlaveBase::mimetype( TQString("text/plain") ); // "application/sieve"); if (!sendData("GETSCRIPT \"" + filename.utf8() + "\"")) return; if (receiveData() && r.getType() == kio_sieveResponse::TQUANTITY) { // determine script size ssize_t total_len = r.getQuantity(); totalSize( total_len ); int recv_len = 0; do { // wait for data... if ( !waitForResponse( 600 ) ) { error( KIO::ERR_SERVER_TIMEOUT, m_sServer ); disconnect( true ); return; } // ...read data... // Only read as much as we need, otherwise we slurp in the OK that // operationSuccessful() is expecting below. TQByteArray dat( kMin( total_len - recv_len, ssize_t(64 * 1024 )) ); ssize_t this_recv_len = read( dat.data(), dat.size() ); if ( this_recv_len < 1 && !isConnectionValid() ) { error( KIO::ERR_CONNECTION_BROKEN, m_sServer ); disconnect( true ); return; } dat.resize( this_recv_len ); inplace_crlf2lf( dat ); // send data to slaveinterface data( dat ); recv_len += this_recv_len; processedSize( recv_len ); } while ( recv_len < total_len ); infoMessage(i18n("Finishing up...") ); data(TQByteArray()); if (operationSuccessful()) ksDebug() << "Script retrieval complete." << endl; else ksDebug() << "Script retrieval failed." << endl; } else { error(ERR_UNSUPPORTED_PROTOCOL, i18n("A protocol error occurred " "while trying to negotiate script downloading.")); return; } infoMessage(i18n("Done.")); finished(); } void kio_sieveProtocol::del(const KURL &url, bool isfile) { if (!isfile) { error(ERR_INTERNAL, i18n("Folders are not supported.")); return; } changeCheck( url ); if (!connect()) return; infoMessage(i18n("Deleting file...")); TQString filename = url.fileName(false); if (filename.isEmpty()) { error(ERR_MALFORMED_URL, url.prettyURL()); return; } if (!sendData("DELETESCRIPT \"" + filename.utf8() + "\"")) return; if (operationSuccessful()) ksDebug() << "Script deletion successful." << endl; else { error(ERR_INTERNAL_SERVER, i18n("The server would not delete the file.")); return; } infoMessage(i18n("Done.")); finished(); } void kio_sieveProtocol::chmod(const KURL& url, int permissions) { switch ( permissions ) { case 0700: // activate activate(url); break; case 0600: // deactivate deactivate(); break; default: // unsupported error(ERR_CANNOT_CHMOD, i18n("Cannot chmod to anything but 0700 (active) or 0600 (inactive script).")); return; } finished(); } #if defined(_AIX) && defined(stat) #undef stat #endif void kio_sieveProtocol::stat(const KURL& url) { changeCheck( url ); if (!connect()) return; UDSEntry entry; TQString filename = url.fileName(false); if (filename.isEmpty()) { UDSAtom atom; atom.m_uds = KIO::UDS_NAME; atom.m_str = "/"; entry.append(atom); atom.m_uds = KIO::UDS_FILE_TYPE; atom.m_long = S_IFDIR; entry.append(atom); atom.m_uds = KIO::UDS_ACCESS; atom.m_long = 0700; entry.append(atom); statEntry(entry); } else { if (!sendData("LISTSCRIPTS")) return; while(receiveData()) { if (r.getType() == kio_sieveResponse::ACTION) { if (r.getAction().contains("OK", false) == 1) // Script list completed break; } else if (filename == TQString::fromUtf8(r.getKey())) { entry.clear(); UDSAtom atom; atom.m_uds = KIO::UDS_NAME; atom.m_str = TQString::fromUtf8(r.getKey()); entry.append(atom); atom.m_uds = KIO::UDS_FILE_TYPE; atom.m_long = S_IFREG; entry.append(atom); atom.m_uds = KIO::UDS_ACCESS; if ( r.getExtra() == "ACTIVE" ) atom.m_long = 0700; // mark exec'able else atom.m_long = 0600; entry.append(atom); atom.m_uds = KIO::UDS_MIME_TYPE; atom.m_str = "application/sieve"; entry.append(atom); //setMetaData("active", (r.getExtra() == "ACTIVE") ? "yes" : "no"); statEntry(entry); // cannot break here because we need to clear // the rest of the incoming data. } } } finished(); } void kio_sieveProtocol::listDir(const KURL& url) { changeCheck( url ); if (!connect()) return; if (!sendData("LISTSCRIPTS")) return; UDSEntry entry; while(receiveData()) { if (r.getType() == kio_sieveResponse::ACTION) { if (r.getAction().contains("OK", false) == 1) // Script list completed. break; } else { entry.clear(); UDSAtom atom; atom.m_uds = KIO::UDS_NAME; atom.m_str = TQString::fromUtf8(r.getKey()); entry.append(atom); atom.m_uds = KIO::UDS_FILE_TYPE; atom.m_long = S_IFREG; entry.append(atom); atom.m_uds = KIO::UDS_ACCESS; if ( r.getExtra() == "ACTIVE" ) atom.m_long = 0700; // mark exec'able else atom.m_long = 0600; entry.append(atom); atom.m_uds = KIO::UDS_MIME_TYPE; atom.m_str = "application/sieve"; entry.append(atom); //asetMetaData("active", (r.getExtra() == "ACTIVE") ? "true" : "false"); ksDebug() << "Listing script " << r.getKey() << endl; listEntry(entry , false); } } listEntry(entry, true); finished(); } /* ---------------------------------------------------------------------------------- */ bool kio_sieveProtocol::saslInteract( void *in, AuthInfo &ai ) { ksDebug() << "sasl_interact" << endl; sasl_interact_t *interact = ( sasl_interact_t * ) in; //some mechanisms do not require username && pass, so it doesn't need a popup //window for getting this info for ( ; interact->id != SASL_CB_LIST_END; interact++ ) { if ( interact->id == SASL_CB_AUTHNAME || interact->id == SASL_CB_PASS ) { if (m_sUser.isEmpty() || m_sPass.isEmpty()) { if (!openPassDlg(ai)) { error(ERR_ABORTED, i18n("No authentication details supplied.")); return false; } m_sUser = ai.username; m_sPass = ai.password; } break; } } interact = ( sasl_interact_t * ) in; while( interact->id != SASL_CB_LIST_END ) { ksDebug() << "SASL_INTERACT id: " << interact->id << endl; switch( interact->id ) { case SASL_CB_USER: case SASL_CB_AUTHNAME: ksDebug() << "SASL_CB_[AUTHNAME|USER]: '" << m_sUser << "'" << endl; interact->result = strdup( m_sUser.utf8() ); interact->len = strlen( (const char *) interact->result ); break; case SASL_CB_PASS: ksDebug() << "SASL_CB_PASS: [hidden] " << endl; interact->result = strdup( m_sPass.utf8() ); interact->len = strlen( (const char *) interact->result ); break; default: interact->result = NULL; interact->len = 0; break; } interact++; } return true; } #define SASLERROR error(ERR_COULD_NOT_AUTHENTICATE, i18n("An error occurred during authentication: %1").tqarg( \ TQString::fromUtf8( sasl_errdetail( conn ) ))); bool kio_sieveProtocol::authenticate() { int result; sasl_conn_t *conn = NULL; sasl_interact_t *client_interact = NULL; const char *out = NULL; uint outlen; const char *mechusing = NULL; TQByteArray challenge, tmp; /* Retrieve authentication details from user. * Note: should this require realm as well as user & pass details * before it automatically skips the prompt? * Note2: encoding issues with PLAIN login? */ AuthInfo ai; ai.url.setProtocol("sieve"); ai.url.setHost(m_sServer); ai.url.setPort(m_iPort); ai.username = m_sUser; ai.password = m_sPass; ai.keepPassword = true; ai.caption = i18n("Sieve Authentication Details"); ai.comment = i18n("Please enter your authentication details for your sieve account " "(usually the same as your email password):"); result = sasl_client_new( "sieve", m_sServer.latin1(), 0, 0, callbacks, 0, &conn ); if ( result != SASL_OK ) { ksDebug() << "sasl_client_new failed with: " << result << endl; SASLERROR return false; } TQStringList strList; // strList.append("NTLM"); if ( !m_sAuth.isEmpty() ) strList.append( m_sAuth ); else strList = m_sasl_caps; do { result = sasl_client_start(conn, strList.join(" ").latin1(), &client_interact, &out, &outlen, &mechusing); if (result == SASL_INTERACT) if ( !saslInteract( client_interact, ai ) ) { sasl_dispose( &conn ); return false; }; } while ( result == SASL_INTERACT ); if ( result != SASL_CONTINUE && result != SASL_OK ) { ksDebug() << "sasl_client_start failed with: " << result << endl; SASLERROR sasl_dispose( &conn ); return false; } ksDebug() << "Preferred authentication method is " << mechusing << "." << endl; TQString firstCommand = "AUTHENTICATE \"" + TQString::fromLatin1( mechusing ) + "\""; tmp.setRawData( out, outlen ); KCodecs::base64Encode( tmp, challenge ); tmp.resetRawData( out, outlen ); if ( !challenge.isEmpty() ) { firstCommand += " \""; firstCommand += TQString::fromLatin1( challenge.data(), challenge.size() ); firstCommand += "\""; } if (!sendData( firstCommand.latin1() )) return false; TQCString command; do { receiveData(); if (operationResult() != OTHER) break; ksDebug() << "Challenge len " << r.getQuantity() << endl; if (r.getType() != kio_sieveResponse::TQUANTITY) { sasl_dispose( &conn ); error(ERR_SLAVE_DEFINED, i18n("A protocol error occurred during authentication.\n" "Choose a different authentication method to %1.").tqarg(mechusing)); return false; } uint qty = r.getQuantity(); receiveData(); if (r.getType() != kio_sieveResponse::ACTION && r.getAction().length() != qty) { sasl_dispose( &conn ); error(ERR_UNSUPPORTED_PROTOCOL, i18n("A protocol error occurred during authentication.\n" "Choose a different authentication method to %1.").tqarg(mechusing)); return false; } tmp.setRawData( r.getAction().data(), qty ); KCodecs::base64Decode( tmp, challenge ); tmp.resetRawData( r.getAction().data(), qty ); // ksDebug() << "S: [" << r.getAction() << "]." << endl; // ksDebug() << "S-1: [" << TQCString(challenge.data(), challenge.size()+1) << "]." << endl; do { result = sasl_client_step(conn, challenge.isEmpty() ? 0 : challenge.data(), challenge.size(), &client_interact, &out, &outlen); if (result == SASL_INTERACT) if ( !saslInteract( client_interact, ai ) ) { sasl_dispose( &conn ); return false; }; } while ( result == SASL_INTERACT ); ksDebug() << "sasl_client_step: " << result << endl; if ( result != SASL_CONTINUE && result != SASL_OK ) { ksDebug() << "sasl_client_step failed with: " << result << endl; SASLERROR sasl_dispose( &conn ); return false; } tmp.setRawData( out, outlen ); KCodecs::base64Encode( tmp, challenge ); tmp.resetRawData( out, outlen ); sendData("\"" + TQCString( challenge.data(), challenge.size()+1 ) + "\""); // ksDebug() << "C: [" << TQCString(challenge.data(), challenge.size()+1) << "]." << endl; // ksDebug() << "C-1: [" << out << "]." << endl; } while ( true ); ksDebug() << "Challenges finished." << endl; sasl_dispose( &conn ); if (operationResult() == OK) { // Authentication succeeded. return true; } else { // Authentication failed. error(ERR_COULD_NOT_AUTHENTICATE, i18n("Authentication failed.\nMost likely the password is wrong.\nThe server responded:\n%1").tqarg( TQString(r.getAction()) ) ); return false; } } /* --------------------------------------------------------------------------- */ void kio_sieveProtocol::mimetype(const KURL & url) { ksDebug() << "Requesting mimetype for " << url.prettyURL() << endl; if (url.fileName(false).isEmpty()) mimeType( "inode/directory" ); else mimeType( "application/sieve" ); finished(); } /* --------------------------------------------------------------------------- */ bool kio_sieveProtocol::sendData(const TQCString &data) { TQCString write_buf = data + "\r\n"; //ksDebug() << "C: " << data << endl; // Write the command ssize_t write_buf_len = write_buf.length(); if (write(write_buf.data(), write_buf_len) != write_buf_len) { error(ERR_COULD_NOT_WRITE, i18n("Network error.")); disconnect(true); return false; } return true; } /* --------------------------------------------------------------------------- */ bool kio_sieveProtocol::receiveData(bool waitForData, TQCString *reparse) { TQCString interpret; int start, end; if (!reparse) { if (!waitForData) // is there data waiting? if (atEnd()) return false; // read data from the server char buffer[SIEVE_DEFAULT_RECIEVE_BUFFER]; readLine(buffer, SIEVE_DEFAULT_RECIEVE_BUFFER - 1); buffer[SIEVE_DEFAULT_RECIEVE_BUFFER-1] = '\0'; // strip LF/CR interpret = TQCString(buffer).left(tqstrlen(buffer) - 2); } else { interpret = reparse->copy(); } r.clear(); //ksDebug() << "S: " << interpret << endl; switch(interpret[0]) { case '{': { // expecting {quantity} start = 0; end = interpret.find("+}", start + 1); // some older versions of Cyrus enclose the literal size just in { } instead of { +} if ( end == -1 ) end = interpret.find('}', start + 1); bool ok = false; r.setQuantity(interpret.mid(start + 1, end - start - 1).toUInt( &ok )); if (!ok) { disconnect(); error(ERR_INTERNAL_SERVER, i18n("A protocol error occurred.")); return false; } return true; } case '"': // expecting "key" "value" pairs break; default: // expecting single string r.setAction(interpret); return true; } start = 0; end = interpret.find(34, start + 1); if (end == -1) { ksDebug() << "Possible insufficient buffer size." << endl; r.setKey(interpret.right(interpret.length() - start)); return true; } r.setKey(interpret.mid(start + 1, end - start - 1)); start = interpret.find(34, end + 1); if (start == -1) { if ((int)interpret.length() > end) // skip " and space r.setExtra(interpret.right(interpret.length() - end - 2)); return true; } end = interpret.find(34, start + 1); if (end == -1) { ksDebug() << "Possible insufficient buffer size." << endl; r.setVal(interpret.right(interpret.length() - start)); return true; } r.setVal(interpret.mid(start + 1, end - start - 1)); return true; } bool kio_sieveProtocol::operationSuccessful() { while (receiveData(false)) { if (r.getType() == kio_sieveResponse::ACTION) { TQCString response = r.getAction().left(2); if (response == "OK") { return true; } else if (response == "NO") { return false; } } } return false; } int kio_sieveProtocol::operationResult() { if (r.getType() == kio_sieveResponse::ACTION) { TQCString response = r.getAction().left(2); if (response == "OK") { return OK; } else if (response == "NO") { return NO; } else if (response == "BY"/*E*/) { return BYE; } } return OTHER; } bool kio_sieveProtocol::requestCapabilitiesAfterStartTLS() const { // Cyrus didn't send CAPABILITIES after STARTTLS until 2.3.11, which is // not standard conform, but we need to support that anyway. // m_implementation looks like this 'Cyrus timsieved v2.2.12' for Cyrus btw. TQRegExp regExp( "Cyrus\\stimsieved\\sv(\\d+)\\.(\\d+)\\.(\\d+)([-\\w]*)", false ); if ( regExp.search( m_implementation ) >= 0 ) { const int major = regExp.cap( 1 ).toInt(); const int minor = regExp.cap( 2 ).toInt(); const int patch = regExp.cap( 3 ).toInt(); const TQString vendor = regExp.cap( 4 ); if ( major < 2 || (major == 2 && (minor < 3 || (minor == 3 && patch < 11))) || (vendor == "-kolab-nocaps") ) { ksDebug() << k_funcinfo << "Enabling compat mode for Cyrus < 2.3.11 or Cyrus marked as \"kolab-nocaps\"" << endl; return true; } } return false; }