kde-playground/kdepimlibs/kioslave/imap4/imapparser.h
Ivailo Monev a9b00154bc generic: replace forward class declarations with header inclusions
Signed-off-by: Ivailo Monev <xakepa10@laimg.moc>
2019-07-31 18:05:48 +00:00

476 lines
13 KiB
C++

#ifndef _IMAPPARSER_H
#define _IMAPPARSER_H
/**********************************************************************
*
* imapparser.h - IMAP4rev1 Parser
* Copyright (C) 2001-2002 Michael Haeckel <haeckel@kde.org>
* Copyright (C) 2000 Sven Carstens <s.carstens@gmx.de>
*
* 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.
*
* Send comments and bug fixes to s.carstens@gmx.de
*
*********************************************************************/
#include "imap4-config.h"
#include "imaplist.h"
#include "imapcommand.h"
#include "imapinfo.h"
#include "mailheader.h"
#include <kio/authinfo.h>
#include <kio/slavebase.h>
#include <QHash>
#include <QList>
#include <QStringList>
#include <QByteArray>
class KUrl;
#include <QString>
class mailAddress;
class mimeHeader;
/** @brief a string used during parsing
* the string allows you to move the effective start of the string using
* str.pos++ and str.pos--.
* @bug it is possible to move past the beginning and end of the string
*/
class parseString
{
public:
parseString() : pos(0) {}
char operator[](int i) const { return data.length() > (i + pos) ? data[i + pos] : '\0'; }
bool isEmpty() const { return pos >= data.size(); }
QByteArray cstr() const
{
if ( pos >= data.size() ) {
return QByteArray();
}
return QByteArray( data.data() + pos, data.size() - pos );
}
int find(char c, int index = 0) const
{
int res = data.indexOf( c, index + pos );
return ( res == -1 ) ? res : ( res - pos );
}
// Warning: does not check for going past end of "data"
void takeLeftNoResize(QByteArray& dest, uint len) const
{
memmove( dest.data(), data.data() + pos, len );
}
// Warning: does not check for going past end of "data"
void takeMidNoResize(QByteArray& dest, uint start, uint len) const
{
memmove( dest.data(), data.data() + pos + start, len );
}
void clear()
{
data.resize( 0 );
pos = 0;
}
uint length()
{
return data.size() - pos;
}
void fromString(const QString &s)
{
clear();
data = s.toLatin1();
}
QByteArray data;
int pos;
};
class imapCache
{
public:
imapCache ()
{
myHeader = NULL;
mySize = 0;
myFlags = 0;
myUid = 0;
}
~imapCache ()
{
delete myHeader;
}
mailHeader *getHeader ()
{
return myHeader;
}
void setHeader (mailHeader * inHeader)
{
myHeader = inHeader;
}
ulong getSize ()
{
return mySize;
}
void setSize (ulong inSize)
{
mySize = inSize;
}
ulong getUid ()
{
return myUid;
}
void setUid (ulong inUid)
{
myUid = inUid;
}
ulong getFlags ()
{
return myFlags;
}
void setFlags (ulong inFlags)
{
myFlags = inFlags;
}
QByteArray getDate ()
{
return myDate;
}
void setDate (const QByteArray & _str)
{
myDate = _str;
}
void clear()
{
delete myHeader;
myHeader = NULL;
mySize = 0;
myFlags = 0;
myDate = QByteArray();
myUid = 0;
}
protected:
mailHeader * myHeader;
ulong mySize;
ulong myFlags;
ulong myUid;
QByteArray myDate;
};
class imapParser
{
public:
/** the different states the client can be in */
enum IMAP_STATE {
ISTATE_NO, /**< Not connected */
ISTATE_CONNECT, /**< Connected but not logged in */
ISTATE_LOGIN, /**< Logged in */
ISTATE_SELECT /**< A folder is currently selected */
};
public:
imapParser ();
virtual ~ imapParser ();
/** @brief Get the current state */
enum IMAP_STATE getState () { return currentState; }
/** @brief Set the current state */
void setState(enum IMAP_STATE state) { currentState = state; }
/* @brief return the currently selected mailbox */
const QString getCurrentBox ()
{
return KIMAP::decodeImapFolderName( currentBox );
}
/**
* @brief do setup and send the command to parseWriteLine
* @param aCmd The command to perform
* @return The completed command
*/
CommandPtr sendCommand (CommandPtr aCmd);
/**
* @brief perform a command and wait to parse the result
* @param aCmd The command to perform
* @return The completed command
*/
CommandPtr doCommand (CommandPtr aCmd);
/**
* @brief plaintext login
* @param aUser Username
* @param aPass Password
* @param resultInfo The resultinfo from the command
* @return success or failure
*/
bool clientLogin (const QString & aUser, const QString & aPass, QString & resultInfo);
/**
* @brief non-plaintext login
* @param aUser Username
* @param aPass Password
* @param aAuth authentication method
* @param isSSL are we using SSL
* @param resultInfo The resultinfo from the command
* @return success or failure
*/
bool clientAuthenticate (KIO::SlaveBase *slave, KIO::AuthInfo &ai, const QString & aFQDN,
const QString & aAuth, bool isSSL, QString & resultInfo);
/**
* main loop for the parser
* reads one line and dispatches it to the appropriate sub parser
*/
int parseLoop ();
/**
* @brief parses all untagged responses and passes them on to the
* following parsers
*/
void parseUntagged (parseString & result);
/** @brief parse a RECENT line */
void parseRecent (ulong value, parseString & result);
/** @brief parse a RESULT line */
void parseResult (QByteArray & result, parseString & rest,
const QString & command = QString());
/** @brief parse a CAPABILITY line */
void parseCapability (parseString & result);
/** @brief parse a FLAGS line */
void parseFlags (parseString & result);
/** @brief parse a LIST line */
void parseList (parseString & result);
/** @brief parse a LSUB line */
void parseLsub (parseString & result);
/** @brief parse a LISTRIGHTS line */
void parseListRights (parseString & result);
/** @brief parse a MYRIGHTS line */
void parseMyRights (parseString & result);
/** @brief parse a SEARCH line */
void parseSearch (parseString & result);
/** @brief parse a STATUS line */
void parseStatus (parseString & result);
/** @brief parse a EXISTS line */
void parseExists (ulong value, parseString & result);
/** @brief parse a EXPUNGE line */
void parseExpunge (ulong value, parseString & result);
/** @brief parse a ACL line */
void parseAcl (parseString & result);
/** @brief parse a ANNOTATION line */
void parseAnnotation (parseString & result);
/** @brief parse a NAMESPACE line */
void parseNamespace (parseString & result);
/** @brief parse a QUOTAROOT line */
void parseQuotaRoot (parseString & result);
/** @brief parse a QUOTA line */
void parseQuota (parseString & result);
/** @brief parse a custom command line */
void parseCustom (parseString & result);
/** @brief parse a OTHER-USER line */
void parseOtherUser (parseString & result);
/** @brief parse a DELEGATE line */
void parseDelegate (parseString & result);
/** @brief parse a OUT-OF-OFFICE line */
void parseOutOfOffice (parseString & result);
/**
* parses the results of a fetch command
* processes it with the following sub parsers
*/
void parseFetch (ulong value, parseString & inWords);
/** read a envelope from imap and parse the addresses */
mailHeader *parseEnvelope (parseString & inWords);
/** @brief parse an address list and return a list of addresses */
void parseAddressList (parseString & inWords, QList<mailAddress *>& list);
/** @brief parse an address and return the ref again */
const mailAddress& parseAddress (parseString & inWords, mailAddress& buffer);
/** parse the result of the body command */
void parseBody (parseString & inWords);
/** parse the body structure recursively */
mimeHeader *parseBodyStructure (parseString & inWords,
QString & section, mimeHeader * inHeader = 0);
/** parse only one not nested part */
mimeHeader *parseSimplePart (parseString & inWords, QString & section,
mimeHeader * localPart = 0);
/** parse a parameter list (name value pairs) */
QHash< QByteArray, QString > parseParameters (parseString & inWords);
/**
* parse the disposition list (disposition (name value pairs))
* the disposition has the key 'content-disposition'
*/
QHash< QByteArray, QString > parseDisposition (parseString & inWords);
// reimplement these
/** relay hook to send the fetched data directly to an upper level */
virtual void parseRelay (const QByteArray & buffer);
/** relay hook to announce the fetched data directly to an upper level
*/
virtual void parseRelay (ulong);
/** read at least len bytes */
virtual bool parseRead (QByteArray & buffer, long len, long relay = 0);
/** read at least a line (up to CRLF) */
virtual bool parseReadLine (QByteArray & buffer, long relay = 0);
/** write argument to server */
virtual void parseWriteLine (const QString &);
// generic parser routines
/** parse a parenthesized list */
void parseSentence (parseString & inWords);
/** parse a literal or word, may require more data */
QByteArray parseLiteral(parseString & inWords, bool relay = false,
bool stopAtBracket = false);
// static parser routines, can be used elsewhere
/** parse one word (maybe quoted) upto next space " ) ] } */
static QByteArray parseOneWord (parseString & inWords,
bool stopAtBracket = false);
/** parse one number using parseOneWord */
static bool parseOneNumber (parseString & inWords, ulong & num);
/** extract the box,section,list type, uid, uidvalidity,info from an url */
static void parseURL (const KUrl & _url, QString & _box, QString & _section,
QString & _type, QString & _uid, QString & _validity,
QString & _info);
/** @brief return the last handled foo
* @todo work out what a foo is
*/
imapCache *getLastHandled ()
{
return lastHandled;
}
/** @brief return the last results */
const QStringList & getResults ()
{
return lastResults;
}
/** @brief return the last status code */
const imapInfo & getStatus ()
{
return lastStatus;
}
/** return the select info */
const imapInfo & getSelected ()
{
return selectInfo;
}
const QByteArray & getContinuation ()
{
return continuation;
}
/** @brief see if server has a capability */
bool hasCapability (const QString &);
void removeCapability (const QString & cap);
static inline void skipWS (parseString & inWords)
{
char c;
while ( !inWords.isEmpty() &&
( ( c = inWords[0] ) == ' ' || c == '\t' || c == '\r' || c == '\n' ) ) {
inWords.pos++;
}
}
/** @brief find the namespace for the given box */
QString namespaceForBox( const QString & box );
protected:
/** the current state we're in */
enum IMAP_STATE currentState;
/** the box selected */
QString currentBox;
/** @brief here we store the result from select/examine and unsolicited updates */
imapInfo selectInfo;
/** @brief the results from the last status command */
imapInfo lastStatus;
/** @brief the results from the capabilities, split at ' ' */
QStringList imapCapabilities;
/** @brief the results from list/lsub/listrights commands */
QList < imapList > listResponses;
/** @brief queues handling the running commands */
QList <CommandPtr> sentQueue;
QList <CommandPtr> completeQueue;
/**
* everything we didn't handle, everything but the greeting is bogus
*/
QStringList unhandled;
/** the last continuation request (there MUST not be more than one pending) */
QByteArray continuation;
/** the last uid seen while a fetch */
QString seenUid;
imapCache *lastHandled;
ulong commandCounter;
/** @brief the results from search/acl commands */
QStringList lastResults;
/**
* @brief namespace prefix - delimiter association
* The namespace is cleaned before so that it does not contain the delimiter
*/
QMap<QString, QString> namespaceToDelimiter;
/**
* @brief list of namespaces in the form: section=namespace=delimiter
* section is 0 (personal), 1 (other users) or 2 (shared)
*/
QStringList imapNamespaces;
private:
/** we don't want to be able to copy this object */
imapParser & operator = (const imapParser &); // hide the copy ctor
};
#endif