464 lines
15 KiB
C
464 lines
15 KiB
C
|
/*
|
||
|
* libjingle
|
||
|
* Copyright 2004--2005, Google Inc.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
*
|
||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer.
|
||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer in the documentation
|
||
|
* and/or other materials provided with the distribution.
|
||
|
* 3. The name of the author may not be used to endorse or promote products
|
||
|
* derived from this software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
||
|
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
||
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
||
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
||
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
||
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
#ifndef TALK_BASE_HTTPCOMMON_H__
|
||
|
#define TALK_BASE_HTTPCOMMON_H__
|
||
|
|
||
|
#include <map>
|
||
|
#include <string>
|
||
|
#include <vector>
|
||
|
#include "talk/base/basictypes.h"
|
||
|
#include "talk/base/common.h"
|
||
|
#include "talk/base/scoped_ptr.h"
|
||
|
#include "talk/base/stringutils.h"
|
||
|
#include "talk/base/stream.h"
|
||
|
|
||
|
namespace talk_base {
|
||
|
|
||
|
class CryptString;
|
||
|
class SocketAddress;
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
// Constants
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
enum HttpCode {
|
||
|
HC_OK = 200,
|
||
|
HC_NON_AUTHORITATIVE = 203,
|
||
|
HC_NO_CONTENT = 204,
|
||
|
HC_PARTIAL_CONTENT = 206,
|
||
|
|
||
|
HC_MULTIPLE_CHOICES = 300,
|
||
|
HC_MOVED_PERMANENTLY = 301,
|
||
|
HC_FOUND = 302,
|
||
|
HC_SEE_OTHER = 303,
|
||
|
HC_NOT_MODIFIED = 304,
|
||
|
HC_MOVED_TEMPORARILY = 307,
|
||
|
|
||
|
HC_BAD_REQUEST = 400,
|
||
|
HC_UNAUTHORIZED = 401,
|
||
|
HC_FORBIDDEN = 403,
|
||
|
HC_NOT_FOUND = 404,
|
||
|
HC_PROXY_AUTHENTICATION_REQUIRED = 407,
|
||
|
HC_GONE = 410,
|
||
|
|
||
|
HC_INTERNAL_SERVER_ERROR = 500,
|
||
|
HC_NOT_IMPLEMENTED = 501,
|
||
|
HC_SERVICE_UNAVAILABLE = 503,
|
||
|
};
|
||
|
|
||
|
enum HttpVersion {
|
||
|
HVER_1_0, HVER_1_1, HVER_UNKNOWN,
|
||
|
HVER_LAST = HVER_UNKNOWN
|
||
|
};
|
||
|
|
||
|
enum HttpVerb {
|
||
|
HV_GET, HV_POST, HV_PUT, HV_DELETE, HV_CONNECT, HV_HEAD,
|
||
|
HV_LAST = HV_HEAD
|
||
|
};
|
||
|
|
||
|
enum HttpError {
|
||
|
HE_NONE,
|
||
|
HE_PROTOCOL, // Received non-valid HTTP data
|
||
|
HE_DISCONNECTED, // Connection closed unexpectedly
|
||
|
HE_OVERFLOW, // Received too much data for internal buffers
|
||
|
HE_CONNECT_FAILED, // The socket failed to connect.
|
||
|
HE_SOCKET_ERROR, // An error occurred on a connected socket
|
||
|
HE_SHUTDOWN, // Http object is being destroyed
|
||
|
HE_OPERATION_CANCELLED, // Connection aborted locally
|
||
|
HE_AUTH, // Proxy Authentication Required
|
||
|
HE_CERTIFICATE_EXPIRED, // During SSL negotiation
|
||
|
HE_STREAM, // Problem reading or writing to the document
|
||
|
HE_CACHE, // Problem reading from cache
|
||
|
HE_DEFAULT
|
||
|
};
|
||
|
|
||
|
enum HttpHeader {
|
||
|
HH_AGE,
|
||
|
HH_CACHE_CONTROL,
|
||
|
HH_CONNECTION,
|
||
|
HH_CONTENT_DISPOSITION,
|
||
|
HH_CONTENT_LENGTH,
|
||
|
HH_CONTENT_RANGE,
|
||
|
HH_CONTENT_TYPE,
|
||
|
HH_COOKIE,
|
||
|
HH_DATE,
|
||
|
HH_ETAG,
|
||
|
HH_EXPIRES,
|
||
|
HH_HOST,
|
||
|
HH_IF_MODIFIED_SINCE,
|
||
|
HH_IF_NONE_MATCH,
|
||
|
HH_KEEP_ALIVE,
|
||
|
HH_LAST_MODIFIED,
|
||
|
HH_LOCATION,
|
||
|
HH_PROXY_AUTHENTICATE,
|
||
|
HH_PROXY_AUTHORIZATION,
|
||
|
HH_PROXY_CONNECTION,
|
||
|
HH_RANGE,
|
||
|
HH_SET_COOKIE,
|
||
|
HH_TE,
|
||
|
HH_TRAILERS,
|
||
|
HH_TRANSFER_ENCODING,
|
||
|
HH_UPGRADE,
|
||
|
HH_USER_AGENT,
|
||
|
HH_WWW_AUTHENTICATE,
|
||
|
HH_LAST = HH_WWW_AUTHENTICATE
|
||
|
};
|
||
|
|
||
|
const uint16 HTTP_DEFAULT_PORT = 80;
|
||
|
const uint16 HTTP_SECURE_PORT = 443;
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
// Utility Functions
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
inline HttpError mkerr(HttpError err, HttpError def_err = HE_DEFAULT) {
|
||
|
return (err != HE_NONE) ? err : def_err;
|
||
|
}
|
||
|
|
||
|
const char* ToString(HttpVersion version);
|
||
|
bool FromString(HttpVersion& version, const std::string& str);
|
||
|
|
||
|
const char* ToString(HttpVerb verb);
|
||
|
bool FromString(HttpVerb& verb, const std::string& str);
|
||
|
|
||
|
const char* ToString(HttpHeader header);
|
||
|
bool FromString(HttpHeader& header, const std::string& str);
|
||
|
|
||
|
inline bool HttpCodeIsInformational(uint32 code) { return ((code / 100) == 1); }
|
||
|
inline bool HttpCodeIsSuccessful(uint32 code) { return ((code / 100) == 2); }
|
||
|
inline bool HttpCodeIsRedirection(uint32 code) { return ((code / 100) == 3); }
|
||
|
inline bool HttpCodeIsClientError(uint32 code) { return ((code / 100) == 4); }
|
||
|
inline bool HttpCodeIsServerError(uint32 code) { return ((code / 100) == 5); }
|
||
|
|
||
|
bool HttpCodeHasBody(uint32 code);
|
||
|
bool HttpCodeIsCacheable(uint32 code);
|
||
|
bool HttpHeaderIsEndToEnd(HttpHeader header);
|
||
|
bool HttpHeaderIsCollapsible(HttpHeader header);
|
||
|
|
||
|
struct HttpData;
|
||
|
bool HttpShouldKeepAlive(const HttpData& data);
|
||
|
|
||
|
typedef std::pair<std::string, std::string> HttpAttribute;
|
||
|
typedef std::vector<HttpAttribute> HttpAttributeList;
|
||
|
void HttpComposeAttributes(const HttpAttributeList& attributes, char separator,
|
||
|
std::string* composed);
|
||
|
void HttpParseAttributes(const char * data, size_t len,
|
||
|
HttpAttributeList& attributes);
|
||
|
bool HttpHasAttribute(const HttpAttributeList& attributes,
|
||
|
const std::string& name,
|
||
|
std::string* value);
|
||
|
bool HttpHasNthAttribute(HttpAttributeList& attributes,
|
||
|
size_t index,
|
||
|
std::string* name,
|
||
|
std::string* value);
|
||
|
|
||
|
// Convert RFC1123 date (DoW, DD Mon YYYY HH:MM:SS TZ) to unix timestamp
|
||
|
bool HttpDateToSeconds(const std::string& date, time_t* seconds);
|
||
|
|
||
|
inline uint16 HttpDefaultPort(bool secure) {
|
||
|
return secure ? HTTP_SECURE_PORT : HTTP_DEFAULT_PORT;
|
||
|
}
|
||
|
|
||
|
// Returns the http server notation for a given address
|
||
|
std::string HttpAddress(const SocketAddress& address, bool secure);
|
||
|
|
||
|
// functional for insensitive std::string compare
|
||
|
struct iless {
|
||
|
bool operator()(const std::string& lhs, const std::string& rhs) const {
|
||
|
return (::_stricmp(lhs.c_str(), rhs.c_str()) < 0);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// put quotes around a string and escape any quotes inside it
|
||
|
std::string quote(const std::string& str);
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
// Url
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
template<class CTYPE>
|
||
|
class Url {
|
||
|
public:
|
||
|
typedef typename Traits<CTYPE>::string string;
|
||
|
|
||
|
// TODO: Implement Encode/Decode
|
||
|
static int Encode(const CTYPE* source, CTYPE* destination, size_t len);
|
||
|
static int Encode(const string& source, string& destination);
|
||
|
static int Decode(const CTYPE* source, CTYPE* destination, size_t len);
|
||
|
static int Decode(const string& source, string& destination);
|
||
|
|
||
|
Url(const string& url) { do_set_url(url.c_str(), url.size()); }
|
||
|
Url(const string& path, const string& host, uint16 port = HTTP_DEFAULT_PORT)
|
||
|
: host_(host), port_(port), secure_(HTTP_SECURE_PORT == port)
|
||
|
{ set_full_path(path); }
|
||
|
|
||
|
bool valid() const { return !host_.empty(); }
|
||
|
void clear() {
|
||
|
host_.clear();
|
||
|
port_ = HTTP_DEFAULT_PORT;
|
||
|
secure_ = false;
|
||
|
path_.assign(1, static_cast<CTYPE>('/'));
|
||
|
query_.clear();
|
||
|
}
|
||
|
|
||
|
void set_url(const string& val) {
|
||
|
do_set_url(val.c_str(), val.size());
|
||
|
}
|
||
|
string url() const {
|
||
|
string val; do_get_url(&val); return val;
|
||
|
}
|
||
|
|
||
|
void set_address(const string& val) {
|
||
|
do_set_address(val.c_str(), val.size());
|
||
|
}
|
||
|
string address() const {
|
||
|
string val; do_get_address(&val); return val;
|
||
|
}
|
||
|
|
||
|
void set_full_path(const string& val) {
|
||
|
do_set_full_path(val.c_str(), val.size());
|
||
|
}
|
||
|
string full_path() const {
|
||
|
string val; do_get_full_path(&val); return val;
|
||
|
}
|
||
|
|
||
|
void set_host(const string& val) { host_ = val; }
|
||
|
const string& host() const { return host_; }
|
||
|
|
||
|
void set_port(uint16 val) { port_ = val; }
|
||
|
uint16 port() const { return port_; }
|
||
|
|
||
|
void set_secure(bool val) { secure_ = val; }
|
||
|
bool secure() const { return secure_; }
|
||
|
|
||
|
void set_path(const string& val) {
|
||
|
if (val.empty()) {
|
||
|
path_.assign(1, static_cast<CTYPE>('/'));
|
||
|
} else {
|
||
|
ASSERT(val[0] == static_cast<CTYPE>('/'));
|
||
|
path_ = val;
|
||
|
}
|
||
|
}
|
||
|
const string& path() const { return path_; }
|
||
|
|
||
|
void set_query(const string& val) {
|
||
|
ASSERT(val.empty() || (val[0] == static_cast<CTYPE>('?')));
|
||
|
query_ = val;
|
||
|
}
|
||
|
const string& query() const { return query_; }
|
||
|
|
||
|
bool get_attribute(const string& name, string* value) const;
|
||
|
|
||
|
private:
|
||
|
void do_set_url(const CTYPE* val, size_t len);
|
||
|
void do_set_address(const CTYPE* val, size_t len);
|
||
|
void do_set_full_path(const CTYPE* val, size_t len);
|
||
|
|
||
|
void do_get_url(string* val) const;
|
||
|
void do_get_address(string* val) const;
|
||
|
void do_get_full_path(string* val) const;
|
||
|
|
||
|
string host_, path_, query_;
|
||
|
uint16 port_;
|
||
|
bool secure_;
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
// HttpData
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
struct HttpData {
|
||
|
typedef std::multimap<std::string, std::string, iless> HeaderMap;
|
||
|
typedef HeaderMap::const_iterator const_iterator;
|
||
|
typedef HeaderMap::iterator iterator;
|
||
|
|
||
|
HttpVersion version;
|
||
|
scoped_ptr<StreamInterface> document;
|
||
|
|
||
|
HttpData() : version(HVER_1_1) { }
|
||
|
|
||
|
enum HeaderCombine { HC_YES, HC_NO, HC_AUTO, HC_REPLACE, HC_NEW };
|
||
|
void changeHeader(const std::string& name, const std::string& value,
|
||
|
HeaderCombine combine);
|
||
|
inline void addHeader(const std::string& name, const std::string& value,
|
||
|
bool append = true) {
|
||
|
changeHeader(name, value, append ? HC_AUTO : HC_NO);
|
||
|
}
|
||
|
inline void setHeader(const std::string& name, const std::string& value,
|
||
|
bool overwrite = true) {
|
||
|
changeHeader(name, value, overwrite ? HC_REPLACE : HC_NEW);
|
||
|
}
|
||
|
// Returns count of erased headers
|
||
|
size_t clearHeader(const std::string& name);
|
||
|
// Returns iterator to next header
|
||
|
iterator clearHeader(iterator header);
|
||
|
|
||
|
// keep in mind, this may not do what you want in the face of multiple headers
|
||
|
bool hasHeader(const std::string& name, std::string* value) const;
|
||
|
|
||
|
inline const_iterator begin() const {
|
||
|
return headers_.begin();
|
||
|
}
|
||
|
inline const_iterator end() const {
|
||
|
return headers_.end();
|
||
|
}
|
||
|
inline iterator begin() {
|
||
|
return headers_.begin();
|
||
|
}
|
||
|
inline iterator end() {
|
||
|
return headers_.end();
|
||
|
}
|
||
|
inline const_iterator begin(const std::string& name) const {
|
||
|
return headers_.lower_bound(name);
|
||
|
}
|
||
|
inline const_iterator end(const std::string& name) const {
|
||
|
return headers_.upper_bound(name);
|
||
|
}
|
||
|
inline iterator begin(const std::string& name) {
|
||
|
return headers_.lower_bound(name);
|
||
|
}
|
||
|
inline iterator end(const std::string& name) {
|
||
|
return headers_.upper_bound(name);
|
||
|
}
|
||
|
|
||
|
// Convenience methods using HttpHeader
|
||
|
inline void changeHeader(HttpHeader header, const std::string& value,
|
||
|
HeaderCombine combine) {
|
||
|
changeHeader(ToString(header), value, combine);
|
||
|
}
|
||
|
inline void addHeader(HttpHeader header, const std::string& value,
|
||
|
bool append = true) {
|
||
|
addHeader(ToString(header), value, append);
|
||
|
}
|
||
|
inline void setHeader(HttpHeader header, const std::string& value,
|
||
|
bool overwrite = true) {
|
||
|
setHeader(ToString(header), value, overwrite);
|
||
|
}
|
||
|
inline void clearHeader(HttpHeader header) {
|
||
|
clearHeader(ToString(header));
|
||
|
}
|
||
|
inline bool hasHeader(HttpHeader header, std::string* value) const {
|
||
|
return hasHeader(ToString(header), value);
|
||
|
}
|
||
|
inline const_iterator begin(HttpHeader header) const {
|
||
|
return headers_.lower_bound(ToString(header));
|
||
|
}
|
||
|
inline const_iterator end(HttpHeader header) const {
|
||
|
return headers_.upper_bound(ToString(header));
|
||
|
}
|
||
|
inline iterator begin(HttpHeader header) {
|
||
|
return headers_.lower_bound(ToString(header));
|
||
|
}
|
||
|
inline iterator end(HttpHeader header) {
|
||
|
return headers_.upper_bound(ToString(header));
|
||
|
}
|
||
|
|
||
|
void setContent(const std::string& content_type, StreamInterface* document);
|
||
|
void setDocumentAndLength(StreamInterface* document);
|
||
|
|
||
|
virtual size_t formatLeader(char* buffer, size_t size) const = 0;
|
||
|
virtual HttpError parseLeader(const char* line, size_t len) = 0;
|
||
|
|
||
|
protected:
|
||
|
virtual ~HttpData() { }
|
||
|
void clear(bool release_document);
|
||
|
void copy(const HttpData& src);
|
||
|
|
||
|
private:
|
||
|
HeaderMap headers_;
|
||
|
};
|
||
|
|
||
|
struct HttpRequestData : public HttpData {
|
||
|
HttpVerb verb;
|
||
|
std::string path;
|
||
|
|
||
|
HttpRequestData() : verb(HV_GET) { }
|
||
|
|
||
|
void clear(bool release_document);
|
||
|
void copy(const HttpRequestData& src);
|
||
|
|
||
|
virtual size_t formatLeader(char* buffer, size_t size) const;
|
||
|
virtual HttpError parseLeader(const char* line, size_t len);
|
||
|
|
||
|
bool getAbsoluteUri(std::string* uri) const;
|
||
|
bool getRelativeUri(std::string* host, std::string* path) const;
|
||
|
};
|
||
|
|
||
|
struct HttpResponseData : public HttpData {
|
||
|
uint32 scode;
|
||
|
std::string message;
|
||
|
|
||
|
HttpResponseData() : scode(HC_INTERNAL_SERVER_ERROR) { }
|
||
|
void clear(bool release_document);
|
||
|
void copy(const HttpResponseData& src);
|
||
|
|
||
|
// Convenience methods
|
||
|
void set_success(uint32 scode = HC_OK);
|
||
|
void set_success(const std::string& content_type, StreamInterface* document,
|
||
|
uint32 scode = HC_OK);
|
||
|
void set_redirect(const std::string& location,
|
||
|
uint32 scode = HC_MOVED_TEMPORARILY);
|
||
|
void set_error(uint32 scode);
|
||
|
|
||
|
virtual size_t formatLeader(char* buffer, size_t size) const;
|
||
|
virtual HttpError parseLeader(const char* line, size_t len);
|
||
|
};
|
||
|
|
||
|
struct HttpTransaction {
|
||
|
HttpRequestData request;
|
||
|
HttpResponseData response;
|
||
|
};
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
// Http Authentication
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
struct HttpAuthContext {
|
||
|
std::string auth_method;
|
||
|
HttpAuthContext(const std::string& auth) : auth_method(auth) { }
|
||
|
virtual ~HttpAuthContext() { }
|
||
|
};
|
||
|
|
||
|
enum HttpAuthResult { HAR_RESPONSE, HAR_IGNORE, HAR_CREDENTIALS, HAR_ERROR };
|
||
|
|
||
|
// 'context' is used by this function to record information between calls.
|
||
|
// Start by passing a null pointer, then pass the same pointer each additional
|
||
|
// call. When the authentication attempt is finished, delete the context.
|
||
|
HttpAuthResult HttpAuthenticate(
|
||
|
const char * challenge, size_t len,
|
||
|
const SocketAddress& server,
|
||
|
const std::string& method, const std::string& uri,
|
||
|
const std::string& username, const CryptString& password,
|
||
|
HttpAuthContext *& context, std::string& response, std::string& auth_method);
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
} // namespace talk_base
|
||
|
|
||
|
#endif // TALK_BASE_HTTPCOMMON_H__
|