kdelibs/kioslave/http/http.h
2014-11-13 01:04:59 +02:00

602 lines
17 KiB
C++

/*
Copyright (C) 2000,2001 Dawit Alemayehu <adawit@kde.org>
Copyright (C) 2000,2001 Waldo Bastian <bastian@kde.org>
Copyright (C) 2000,2001 George Staikos <staikos@kde.org>
Copyright (C) 2001,2002 Hamish Rodda <rodda@kde.org>
Copyright (C) 2007 Daniel Nicoletti <mirttex@users.sourceforge.net>
Copyright (C) 2008,2009 Andreas Hartmetz <ahartmetz@gmail.com>
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.
*/
#ifndef HTTP_H
#define HTTP_H
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdio.h>
#include <time.h>
#include <QtCore/QList>
#include <QtCore/QStringList>
#include <QtNetwork/QLocalSocket>
#include <kurl.h>
#include "kio/tcpslavebase.h"
#include "kio/http.h"
class QDomNodeList;
class QFile;
class QIODevice;
namespace KIO {
class AuthInfo;
}
class HeaderTokenizer;
class KAbstractHttpAuthentication;
class HTTPProtocol : public QObject, public KIO::TCPSlaveBase
{
Q_OBJECT
public:
HTTPProtocol( const QByteArray &protocol, const QByteArray &pool,
const QByteArray &app );
virtual ~HTTPProtocol();
/** HTTP version **/
enum HTTP_REV {HTTP_None, HTTP_Unknown, HTTP_10, HTTP_11, SHOUTCAST};
/** Authorization method used **/
enum AUTH_SCHEME {AUTH_None, AUTH_Basic, AUTH_NTLM, AUTH_Digest, AUTH_Negotiate};
/** DAV-specific request elements for the current connection **/
struct DAVRequest
{
DAVRequest ()
{
overwrite = false;
depth = 0;
}
QString desturl;
bool overwrite;
int depth;
};
enum CacheIOMode {
NoCache = 0,
ReadFromCache = 1,
WriteToCache = 2
};
struct CacheTag
{
CacheTag()
{
useCache = false;
ioMode = NoCache;
bytesCached = 0;
file = 0;
expireDate = 0;
servedDate = 0;
lastModifiedDate = 0;
fileUseCount = 0;
}
enum CachePlan {
UseCached = 0,
ValidateCached,
IgnoreCached
};
CachePlan plan(int maxCacheAge) const;
QByteArray serialize() const;
bool deserialize(const QByteArray &);
KIO::CacheControl policy; // ### initialize in the constructor?
bool useCache; // Whether the cache should be used
enum CacheIOMode ioMode; // Write to cache file, read from it, or don't use it.
quint32 fileUseCount;
quint32 bytesCached;
QString etag; // entity tag header as described in the HTTP standard.
QFile *file; // file on disk - either a QTemporaryFile (write) or QFile (read)
qint64 servedDate; // Date when the resource was served by the origin server
qint64 lastModifiedDate; // Last modified.
qint64 expireDate; // Date when the cache entry will expire
QString charset;
};
/** The request for the current connection **/
struct HTTPRequest
{
HTTPRequest ()
{
method = KIO::HTTP_UNKNOWN;
offset = 0;
endoffset = 0;
allowTransferCompression = false;
disablePassDialog = false;
doNotWWWAuthenticate = false;
doNotProxyAuthenticate = false;
preferErrorPage = false;
useCookieJar = false;
}
QByteArray methodString() const;
KUrl url;
QString encoded_hostname; //### can be calculated on-the-fly
// Persistent connections
bool isKeepAlive;
int keepAliveTimeout; // Timeout in seconds.
KIO::HTTP_METHOD method;
QString methodStringOverride; // Overrides method if non-empty.
QByteArray sentMethodString; // Stores http method actually sent
KIO::filesize_t offset;
KIO::filesize_t endoffset;
QString windowId; // Window Id this request is related to.
// Header fields
QString referrer;
QString charsets;
QString languages;
QString userAgent;
// Previous and current response codes
unsigned int responseCode;
unsigned int prevResponseCode;
// Miscellaneous
QString id;
DAVRequest davData;
KUrl redirectUrl;
KUrl proxyUrl;
QStringList proxyUrls;
bool isPersistentProxyConnection;
bool allowTransferCompression;
bool disablePassDialog;
bool doNotWWWAuthenticate;
bool doNotProxyAuthenticate;
// Indicates whether an error page or error message is preferred.
bool preferErrorPage;
// Use the cookie jar (or pass cookies to the application as metadata instead)
bool useCookieJar;
// Cookie flags
enum { CookiesAuto, CookiesManual, CookiesNone } cookieMode;
CacheTag cacheTag;
};
/** State of the current connection to the server **/
struct HTTPServerState
{
HTTPServerState()
{
isKeepAlive = false;
isPersistentProxyConnection = false;
}
void initFrom(const HTTPRequest &request)
{
url = request.url;
encoded_hostname = request.encoded_hostname;
isKeepAlive = request.isKeepAlive;
proxyUrl = request.proxyUrl;
isPersistentProxyConnection = request.isPersistentProxyConnection;
}
void updateCredentials(const HTTPRequest &request)
{
if (url.host() == request.url.host() && url.port() == request.url.port()) {
url.setUserName(request.url.userName());
url.setPassword(request.url.password());
}
if (proxyUrl.host() == request.proxyUrl.host() &&
proxyUrl.port() == request.proxyUrl.port()) {
proxyUrl.setUserName(request.proxyUrl.userName());
proxyUrl.setPassword(request.proxyUrl.password());
}
}
void clear()
{
url.clear();
encoded_hostname.clear();
proxyUrl.clear();
isKeepAlive = false;
isPersistentProxyConnection = false;
}
KUrl url;
QString encoded_hostname;
KUrl proxyUrl;
bool isKeepAlive;
bool isPersistentProxyConnection;
};
//---------------------- Re-implemented methods ----------------
virtual void setHost(const QString& host, quint16 port, const QString& user,
const QString& pass);
virtual void slave_status();
virtual void get( const KUrl& url );
virtual void put( const KUrl& url, int _mode, KIO::JobFlags flags );
//----------------- Re-implemented methods for WebDAV -----------
virtual void listDir( const KUrl& url );
virtual void mkdir( const KUrl& url, int _permissions );
virtual void rename( const KUrl& src, const KUrl& dest, KIO::JobFlags flags );
virtual void copy( const KUrl& src, const KUrl& dest, int _permissions, KIO::JobFlags flags );
virtual void del( const KUrl& url, bool _isfile );
// ask the host whether it supports WebDAV & cache this info
bool davHostOk();
// send generic DAV request
void davGeneric( const KUrl& url, KIO::HTTP_METHOD method, qint64 size = -1 );
// Send requests to lock and unlock resources
void davLock( const KUrl& url, const QString& scope,
const QString& type, const QString& owner );
void davUnlock( const KUrl& url );
// Calls httpClose() and finished()
void davFinished();
// Handle error conditions
QString davError( int code = -1, const QString &url = QString() );
//---------------------------- End WebDAV -----------------------
/**
* Special commands supported by this slave :
* 1 - HTTP POST
* 2 - Cache has been updated
* 3 - SSL Certificate Cache has been updated
* 4 - HTTP multi get
* 5 - DAV LOCK (see
* 6 - DAV UNLOCK README.webdav)
*/
virtual void special( const QByteArray &data );
virtual void mimetype( const KUrl& url);
virtual void stat( const KUrl& url );
virtual void reparseConfiguration();
/**
* Forced close of connection
*/
virtual void closeConnection();
void post( const KUrl& url, qint64 size = -1 );
void multiGet(const QByteArray &data);
bool maybeSetRequestUrl(const KUrl &);
/**
* Generate and send error message based on response code.
*/
bool sendHttpError();
/**
* Call SlaveBase::errorPage() and remember that we've called it
*/
bool sendErrorPageNotification();
/**
* Check network status
*/
bool isOffline();
protected Q_SLOTS:
void slotData(const QByteArray &);
void slotFilterError(const QString &text);
void error(int errid, const QString &text);
void proxyAuthenticationForSocket(const QNetworkProxy &, QAuthenticator *);
void saveProxyAuthenticationForSocket();
protected:
int readChunked(); ///< Read a chunk
int readLimited(); ///< Read maximum m_iSize bytes.
int readUnlimited(); ///< Read as much as possible.
/**
* A thin wrapper around TCPSlaveBase::write() that will retry writing as
* long as no error occurs.
*/
ssize_t write(const void *buf, size_t nbytes);
using SlaveBase::write;
/**
* Add an encoding on to the appropriate stack this
* is nececesary because transfer encodings and
* content encodings must be handled separately.
*/
void addEncoding(const QString &, QStringList &);
quint16 defaultPort() const;
// The methods between here and sendQuery() are helpers for sendQuery().
/**
* Return true if the request is already "done", false otherwise.
*
* @p cacheHasPage will be set to true if the page was found, false otherwise.
*/
bool satisfyRequestFromCache(bool *cacheHasPage);
QString formatRequestUri() const;
/**
* create HTTP authentications response(s), if any
*/
QString authenticationHeader();
bool sendQuery();
/**
* Close transfer
*/
void httpClose(bool keepAlive);
/**
* Open connection
*/
bool httpOpenConnection();
/**
* Close connection
*/
void httpCloseConnection();
/**
* Check whether to keep or close the connection.
*/
bool httpShouldCloseConnection();
void forwardHttpResponseHeader(bool forwardImmediately = true);
/**
* fix common mimetype errors by webservers.
*
* Helper for readResponseHeader().
*/
void fixupResponseMimetype();
/**
* fix common content-encoding errors by webservers.
*
* Helper for readResponseHeader().
*/
void fixupResponseContentEncoding();
bool readResponseHeader();
bool parseHeaderFromCache();
void parseContentDisposition(const QString &disposition);
bool sendBody();
bool sendCachedBody();
// where dataInternal == true, the content is to be made available
// to an internal function.
bool readBody( bool dataInternal = false );
/**
* Performs a WebDAV stat or list
*/
void davSetRequest( const QByteArray& requestXML );
void davStatList( const KUrl& url, bool stat = true );
void davParsePropstats( const QDomNodeList& propstats, KIO::UDSEntry& entry );
void davParseActiveLocks( const QDomNodeList& activeLocks,
uint& lockCount );
/**
* Returns the error code from a "HTTP/1.1 code Code Name" string
*/
int codeFromResponse( const QString& response );
/**
* Extracts locks from metadata
* Returns the appropriate If: header
*/
QString davProcessLocks();
/**
* Send a cookie to the cookiejar
*/
void addCookies( const QString &url, const QByteArray &cookieHeader);
/**
* Look for cookies in the cookiejar
*/
QString findCookies( const QString &url);
void cacheParseResponseHeader(const HeaderTokenizer &tokenizer);
QString cacheFilePathFromUrl(const KUrl &url) const;
bool cacheFileOpenRead();
bool cacheFileOpenWrite();
void cacheFileClose();
void sendCacheCleanerCommand(const QByteArray &command);
QByteArray cacheFileReadPayload(int maxLength);
void cacheFileWritePayload(const QByteArray &d);
void cacheFileWriteTextHeader();
/**
* check URL to guard against hash collisions, and load the etag for validation
*/
bool cacheFileReadTextHeader1(const KUrl &desiredUrl);
/**
* load the rest of the text fields
*/
bool cacheFileReadTextHeader2();
void setCacheabilityMetadata(bool cachingAllowed);
/**
* Do everything proceedUntilResponseHeader does, and also get the response body.
* This is being used as a replacement for proceedUntilResponseHeader() in
* situations where we actually expect the response to have a body / payload data.
*
* where dataInternal == true, the content is to be made available
* to an internal function.
*/
void proceedUntilResponseContent( bool dataInternal = false );
/**
* Ensure we are connected, send our query, and get the response header.
*/
bool proceedUntilResponseHeader();
/**
* Resets any per session settings.
*/
void resetSessionSettings();
/**
* Resets variables related to parsing a response.
*/
void resetResponseParsing();
/**
* Resets any per connection settings. These are different from
* per-session settings in that they must be invalidated every time
* a request is made, e.g. a retry to re-send the header to the
* server, as compared to only when a new request arrives.
*/
void resetConnectionSettings();
/**
* Caches the POST data in a temporary buffer.
*
* Depending on size of content, the temporary buffer might be
* created either in memory or on disk as (a temporary file).
*/
void cachePostData(const QByteArray&);
/**
* Clears the POST data buffer.
*
* Note that calling this function results in the POST data buffer
* getting completely deleted.
*/
void clearPostDataBuffer();
/**
* Returns true on successful retrieval of all content data.
*/
bool retrieveAllData();
/**
* Saves HTTP authentication data.
*/
void saveAuthenticationData();
/**
* Handles HTTP authentication.
*/
bool handleAuthenticationHeader(const HeaderTokenizer* tokenizer);
/**
* Handles file -> webdav put requests.
*/
void copyPut(const KUrl& src, const KUrl& dest, KIO::JobFlags flags);
/**
* Stats a remote DAV file and returns true if it exists.
*/
bool davStatDestination();
protected:
HTTPServerState m_server;
HTTPRequest m_request;
QList<HTTPRequest> m_requestQueue;
// Processing related
KIO::filesize_t m_iSize; ///< Expected size of message
KIO::filesize_t m_iPostDataSize;
KIO::filesize_t m_iBytesLeft; ///< # of bytes left to receive in this message.
KIO::filesize_t m_iContentLeft; ///< # of content bytes left
QByteArray m_receiveBuf; ///< Receive buffer
bool m_dataInternal; ///< Data is for internal consumption
bool m_isChunked; ///< Chunked transfer encoding
bool m_isBusy; ///< Busy handling request queue.
bool m_isEOF;
bool m_isEOD;
//--- Settings related to a single response only
bool m_isRedirection; ///< Indicates current request is a redirection
QStringList m_responseHeaders; ///< All headers
// Language/Encoding related
QStringList m_transferEncodings;
QStringList m_contentEncodings;
QString m_contentMD5;
QString m_mimeType; // TODO QByteArray?
//--- WebDAV
// Data structure to hold data which will be passed to an internal func.
QByteArray m_webDavDataBuf;
QStringList m_davCapabilities;
bool m_davHostOk;
bool m_davHostUnsupported;
//----------
// Mimetype determination
bool m_cpMimeBuffer;
QByteArray m_mimeTypeBuffer;
// Holds the POST data so it won't get lost on if we
// happend to get a 401/407 response when submitting
// a form.
QIODevice* m_POSTbuf;
// Cache related
int m_maxCacheAge; ///< Maximum age of a cache entry in seconds.
long m_maxCacheSize; ///< Maximum cache size in Kb.
QString m_strCacheDir; ///< Location of the cache.
QLocalSocket m_cacheCleanerConnection; ///< Connection to the cache cleaner process
// Operation mode
QByteArray m_protocol;
KAbstractHttpAuthentication *m_wwwAuth;
KAbstractHttpAuthentication *m_proxyAuth;
// For proxy auth when it's handled by the Qt/KDE socket classes
QAuthenticator *m_socketProxyAuth;
// Indicates whether there was some error.
int m_iError;
// Whether we are loading an error page (we should close the connection afterwards)
bool m_isLoadingErrorPage;
// Values that determine the remote connection timeouts.
int m_remoteRespTimeout;
// EOF Retry count
quint8 m_iEOFRetryCount;
QByteArray m_unreadBuf;
void clearUnreadBuffer();
void unread(char *buf, size_t size);
size_t readBuffered(char *buf, size_t size, bool unlimited = true);
bool readDelimitedText(char *buf, int *idx, int end, int numNewlines);
};
#endif