2013-12-07 12:51:27 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
2014-01-01 02:37:12 +00:00
|
|
|
Copyright (c) 2013-2014 winlin
|
2013-12-07 12:51:27 +00:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
|
|
this software and associated documentation files (the "Software"), to deal in
|
|
|
|
the Software without restriction, including without limitation the rights to
|
|
|
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
the Software, and to permit persons to whom the Software is furnished to do so,
|
|
|
|
subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
|
|
copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
|
|
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
|
|
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2014-03-02 13:49:09 +00:00
|
|
|
#ifndef SRS_APP_HTTP_HPP
|
|
|
|
#define SRS_APP_HTTP_HPP
|
2013-12-07 12:51:27 +00:00
|
|
|
|
|
|
|
/*
|
2014-03-02 13:49:09 +00:00
|
|
|
#include <srs_app_http.hpp>
|
2013-12-07 12:51:27 +00:00
|
|
|
*/
|
|
|
|
#include <srs_core.hpp>
|
|
|
|
|
2014-04-01 10:40:24 +00:00
|
|
|
#ifdef SRS_HTTP_PARSER
|
|
|
|
|
|
|
|
#include <string>
|
2014-04-02 10:07:34 +00:00
|
|
|
#include <vector>
|
2014-04-02 11:10:22 +00:00
|
|
|
#include <sstream>
|
2014-04-01 10:40:24 +00:00
|
|
|
|
|
|
|
#include <http_parser.h>
|
|
|
|
|
2014-03-02 13:49:09 +00:00
|
|
|
#include <srs_app_st.hpp>
|
2014-03-01 02:14:25 +00:00
|
|
|
|
2014-04-01 10:40:24 +00:00
|
|
|
class SrsBuffer;
|
|
|
|
class SrsRequest;
|
|
|
|
class SrsSocket;
|
2014-04-03 03:49:14 +00:00
|
|
|
class SrsHttpUri;
|
2014-04-02 10:07:34 +00:00
|
|
|
class SrsHttpMessage;
|
2014-04-03 03:49:14 +00:00
|
|
|
class SrsHttpHandler;
|
2014-04-01 07:42:27 +00:00
|
|
|
|
2014-03-27 09:13:26 +00:00
|
|
|
// http specification
|
2014-04-01 09:30:55 +00:00
|
|
|
// CR = <US-ASCII CR, carriage return (13)>
|
|
|
|
#define __CR "\r" // 0x0D
|
|
|
|
// LF = <US-ASCII LF, linefeed (10)>
|
|
|
|
#define __LF "\n" // 0x0A
|
|
|
|
// SP = <US-ASCII SP, space (32)>
|
|
|
|
#define __SP " " // 0x20
|
|
|
|
// HT = <US-ASCII HT, horizontal-tab (9)>
|
|
|
|
#define __HT "\x09" // 0x09
|
|
|
|
|
|
|
|
// HTTP/1.1 defines the sequence CR LF as the end-of-line marker for all
|
|
|
|
// protocol elements except the entity-body (see appendix 19.3 for
|
|
|
|
// tolerant applications).
|
|
|
|
#define __CRLF "\r\n" // 0x0D0A
|
|
|
|
#define __CRLFCRLF "\r\n\r\n" // 0x0D0A0D0A
|
2014-03-27 09:13:26 +00:00
|
|
|
|
2014-04-03 07:17:02 +00:00
|
|
|
// 6.1.1 Status Code and Reason Phrase
|
|
|
|
#define HTTP_Continue 100
|
|
|
|
#define HTTP_SwitchingProtocols 101
|
|
|
|
#define HTTP_OK 200
|
|
|
|
#define HTTP_Created 201
|
|
|
|
#define HTTP_Accepted 202
|
|
|
|
#define HTTP_NonAuthoritativeInformation 203
|
|
|
|
#define HTTP_NoContent 204
|
|
|
|
#define HTTP_ResetContent 205
|
|
|
|
#define HTTP_PartialContent 206
|
|
|
|
#define HTTP_MultipleChoices 300
|
|
|
|
#define HTTP_MovedPermanently 301
|
|
|
|
#define HTTP_Found 302
|
|
|
|
#define HTTP_SeeOther 303
|
|
|
|
#define HTTP_NotModified 304
|
|
|
|
#define HTTP_UseProxy 305
|
|
|
|
#define HTTP_TemporaryRedirect 307
|
|
|
|
#define HTTP_BadRequest 400
|
|
|
|
#define HTTP_Unauthorized 401
|
|
|
|
#define HTTP_PaymentRequired 402
|
|
|
|
#define HTTP_Forbidden 403
|
|
|
|
#define HTTP_NotFound 404
|
|
|
|
#define HTTP_MethodNotAllowed 405
|
|
|
|
#define HTTP_NotAcceptable 406
|
|
|
|
#define HTTP_ProxyAuthenticationRequired 407
|
|
|
|
#define HTTP_RequestTimeout 408
|
|
|
|
#define HTTP_Conflict 409
|
|
|
|
#define HTTP_Gone 410
|
|
|
|
#define HTTP_LengthRequired 411
|
|
|
|
#define HTTP_PreconditionFailed 412
|
|
|
|
#define HTTP_RequestEntityTooLarge 413
|
|
|
|
#define HTTP_RequestURITooLarge 414
|
|
|
|
#define HTTP_UnsupportedMediaType 415
|
|
|
|
#define HTTP_RequestedRangeNotSatisfiable 416
|
|
|
|
#define HTTP_ExpectationFailed 417
|
|
|
|
#define HTTP_InternalServerError 500
|
|
|
|
#define HTTP_NotImplemented 501
|
|
|
|
#define HTTP_BadGateway 502
|
|
|
|
#define HTTP_ServiceUnavailable 503
|
|
|
|
#define HTTP_GatewayTimeout 504
|
|
|
|
#define HTTP_HTTPVersionNotSupported 505
|
|
|
|
|
|
|
|
#define HTTP_Continue_str "Continue"
|
|
|
|
#define HTTP_SwitchingProtocols_str "Switching Protocols"
|
|
|
|
#define HTTP_OK_str "OK"
|
|
|
|
#define HTTP_Created_str "Created "
|
|
|
|
#define HTTP_Accepted_str "Accepted"
|
|
|
|
#define HTTP_NonAuthoritativeInformation_str "Non Authoritative Information "
|
|
|
|
#define HTTP_NoContent_str "No Content "
|
|
|
|
#define HTTP_ResetContent_str "Reset Content"
|
|
|
|
#define HTTP_PartialContent_str "Partial Content"
|
|
|
|
#define HTTP_MultipleChoices_str "Multiple Choices "
|
|
|
|
#define HTTP_MovedPermanently_str "Moved Permanently"
|
|
|
|
#define HTTP_Found_str "Found"
|
|
|
|
#define HTTP_SeeOther_str "See Other"
|
|
|
|
#define HTTP_NotModified_str "Not Modified "
|
|
|
|
#define HTTP_UseProxy_str "Use Proxy"
|
|
|
|
#define HTTP_TemporaryRedirect_str "Temporary Redirect "
|
|
|
|
#define HTTP_BadRequest_str "Bad Request"
|
|
|
|
#define HTTP_Unauthorized_str "Unauthorized"
|
|
|
|
#define HTTP_PaymentRequired_str "Payment Required "
|
|
|
|
#define HTTP_Forbidden_str "Forbidden "
|
|
|
|
#define HTTP_NotFound_str "Not Found"
|
|
|
|
#define HTTP_MethodNotAllowed_str "Method Not Allowed"
|
|
|
|
#define HTTP_NotAcceptable_str "Not Acceptable "
|
|
|
|
#define HTTP_ProxyAuthenticationRequired_str "Proxy Authentication Required "
|
|
|
|
#define HTTP_RequestTimeout_str "Request Timeout"
|
|
|
|
#define HTTP_Conflict_str "Conflict"
|
|
|
|
#define HTTP_Gone_str "Gone"
|
|
|
|
#define HTTP_LengthRequired_str "Length Required"
|
|
|
|
#define HTTP_PreconditionFailed_str "Precondition Failed"
|
|
|
|
#define HTTP_RequestEntityTooLarge_str "Request Entity Too Large "
|
|
|
|
#define HTTP_RequestURITooLarge_str "Request URI Too Large"
|
|
|
|
#define HTTP_UnsupportedMediaType_str "Unsupported Media Type"
|
|
|
|
#define HTTP_RequestedRangeNotSatisfiable_str "Requested Range Not Satisfiable"
|
|
|
|
#define HTTP_ExpectationFailed_str "Expectation Failed "
|
|
|
|
#define HTTP_InternalServerError_str "Internal Server Error "
|
|
|
|
#define HTTP_NotImplemented_str "Not Implemented"
|
|
|
|
#define HTTP_BadGateway_str "Bad Gateway"
|
|
|
|
#define HTTP_ServiceUnavailable_str "Service Unavailable"
|
|
|
|
#define HTTP_GatewayTimeout_str "Gateway Timeout"
|
|
|
|
#define HTTP_HTTPVersionNotSupported_str "HTTP Version Not Supported"
|
|
|
|
|
2014-04-02 10:07:34 +00:00
|
|
|
// linux path seprator
|
|
|
|
#define __PATH_SEP '/'
|
2014-04-03 03:49:14 +00:00
|
|
|
// query string seprator
|
|
|
|
#define __QUERY_SEP '?'
|
|
|
|
|
|
|
|
// compare the path.
|
|
|
|
// full compare, extractly match.
|
|
|
|
extern bool srs_path_equals(const char* expect, const char* path, int nb_path);
|
2014-04-02 10:07:34 +00:00
|
|
|
|
|
|
|
// state of message
|
2014-04-01 10:40:24 +00:00
|
|
|
enum SrsHttpParseState {
|
|
|
|
SrsHttpParseStateInit = 0,
|
|
|
|
SrsHttpParseStateStart,
|
|
|
|
SrsHttpParseStateComplete
|
|
|
|
};
|
2013-12-07 14:06:53 +00:00
|
|
|
|
2014-04-03 03:49:14 +00:00
|
|
|
/**
|
|
|
|
* the matched handler info.
|
|
|
|
*/
|
|
|
|
class SrsHttpHandlerMatch
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SrsHttpHandler* handler;
|
|
|
|
std::string matched_url;
|
2014-04-03 07:17:02 +00:00
|
|
|
std::string unmatched_url;
|
2014-04-03 03:49:14 +00:00
|
|
|
public:
|
|
|
|
SrsHttpHandlerMatch();
|
|
|
|
};
|
|
|
|
|
2014-04-02 10:07:34 +00:00
|
|
|
/**
|
|
|
|
* resource handler for HTTP RESTful api.
|
|
|
|
*/
|
|
|
|
class SrsHttpHandler
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
/**
|
|
|
|
* we use handler chain to process request.
|
|
|
|
*/
|
|
|
|
std::vector<SrsHttpHandler*> handlers;
|
|
|
|
public:
|
|
|
|
SrsHttpHandler();
|
|
|
|
virtual ~SrsHttpHandler();
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* initialize the handler.
|
|
|
|
*/
|
|
|
|
virtual int initialize();
|
|
|
|
/**
|
|
|
|
* whether current handler can handle the specified path.
|
2014-04-03 03:49:14 +00:00
|
|
|
* @pchild set the next child path, if needed.
|
|
|
|
* for example, the root handler will reset pchild to path,
|
|
|
|
* to reparse the path use child handlers.
|
2014-04-02 10:07:34 +00:00
|
|
|
*/
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual bool can_handle(const char* path, int length, const char** pchild);
|
2014-04-02 10:07:34 +00:00
|
|
|
/**
|
|
|
|
* use the handler to process the request.
|
2014-04-03 03:49:14 +00:00
|
|
|
* @remark sub classes should override the do_process_request.
|
2014-04-02 10:07:34 +00:00
|
|
|
*/
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual int process_request(SrsSocket* skt, SrsHttpMessage* req);
|
2014-04-02 10:07:34 +00:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* find the best matched handler
|
|
|
|
*/
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual int best_match(const char* path, int length, SrsHttpHandlerMatch** ppmatch);
|
|
|
|
// factory methods
|
|
|
|
protected:
|
2014-04-03 07:17:02 +00:00
|
|
|
/**
|
|
|
|
* check whether the handler is valid.
|
|
|
|
* for example, user access /apis, actually it's not found,
|
|
|
|
* we will find the root handler to process it.
|
|
|
|
* @remark user can override this method, and should invoke it first.
|
|
|
|
* @see SrsApiRoot::is_handler_valid
|
|
|
|
*/
|
|
|
|
virtual bool is_handler_valid(SrsHttpMessage* req, int& status_code, std::string& reason_phrase);
|
|
|
|
/**
|
|
|
|
* do the actual process of request.
|
|
|
|
*/
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual int do_process_request(SrsSocket* skt, SrsHttpMessage* req);
|
|
|
|
// response writer
|
2014-04-02 11:10:22 +00:00
|
|
|
public:
|
|
|
|
virtual SrsHttpHandler* res_status_line(std::stringstream& ss);
|
2014-04-03 07:17:02 +00:00
|
|
|
virtual SrsHttpHandler* res_status_line_error(std::stringstream& ss, int code, std::string reason_phrase);
|
2014-04-02 11:10:22 +00:00
|
|
|
virtual SrsHttpHandler* res_content_type(std::stringstream& ss);
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual SrsHttpHandler* res_content_type_json(std::stringstream& ss);
|
2014-04-02 11:10:22 +00:00
|
|
|
virtual SrsHttpHandler* res_content_length(std::stringstream& ss, int64_t length);
|
|
|
|
virtual SrsHttpHandler* res_enable_crossdomain(std::stringstream& ss);
|
|
|
|
virtual SrsHttpHandler* res_header_eof(std::stringstream& ss);
|
|
|
|
virtual SrsHttpHandler* res_body(std::stringstream& ss, std::string body);
|
|
|
|
virtual int res_flush(SrsSocket* skt, std::stringstream& ss);
|
|
|
|
public:
|
|
|
|
virtual int res_options(SrsSocket* skt);
|
|
|
|
virtual int res_text(SrsSocket* skt, std::string body);
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual int res_json(SrsSocket* skt, std::string json);
|
2014-04-03 07:17:02 +00:00
|
|
|
virtual int res_error(SrsSocket* skt, int code, std::string reason_phrase, std::string body);
|
2014-04-03 03:49:14 +00:00
|
|
|
// object creator
|
2014-04-02 10:07:34 +00:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* create http api resource handler.
|
|
|
|
*/
|
|
|
|
static SrsHttpHandler* create_http_api();
|
|
|
|
/**
|
|
|
|
* create http stream resource handler.
|
|
|
|
*/
|
|
|
|
static SrsHttpHandler* create_http_stream();
|
|
|
|
};
|
|
|
|
|
2014-04-01 10:40:24 +00:00
|
|
|
/**
|
|
|
|
* the http message, request or response.
|
|
|
|
*/
|
|
|
|
class SrsHttpMessage
|
|
|
|
{
|
2014-04-02 04:03:49 +00:00
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* parsed url.
|
|
|
|
*/
|
|
|
|
std::string _url;
|
|
|
|
/**
|
|
|
|
* parsed http header.
|
|
|
|
*/
|
|
|
|
http_parser _header;
|
|
|
|
/**
|
|
|
|
* body object, in bytes.
|
|
|
|
* @remark, user can get body in string by get_body().
|
|
|
|
*/
|
|
|
|
SrsBuffer* _body;
|
|
|
|
/**
|
|
|
|
* parser state
|
|
|
|
* @remark, user can use is_complete() to determine the state.
|
|
|
|
*/
|
|
|
|
SrsHttpParseState _state;
|
2014-04-03 03:49:14 +00:00
|
|
|
/**
|
|
|
|
* uri parser
|
|
|
|
*/
|
|
|
|
SrsHttpUri* _uri;
|
|
|
|
/**
|
|
|
|
* best matched handler.
|
|
|
|
*/
|
|
|
|
SrsHttpHandlerMatch* _match;
|
2014-04-02 04:03:49 +00:00
|
|
|
public:
|
2014-04-01 10:40:24 +00:00
|
|
|
SrsHttpMessage();
|
|
|
|
virtual ~SrsHttpMessage();
|
2014-04-02 04:03:49 +00:00
|
|
|
public:
|
2014-04-01 10:40:24 +00:00
|
|
|
virtual void reset();
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual int parse_uri();
|
2014-04-02 04:03:49 +00:00
|
|
|
public:
|
2014-04-01 10:40:24 +00:00
|
|
|
virtual bool is_complete();
|
2014-04-02 04:03:49 +00:00
|
|
|
virtual u_int8_t method();
|
|
|
|
virtual std::string url();
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual std::string path();
|
|
|
|
virtual std::string query();
|
2014-04-02 04:03:49 +00:00
|
|
|
virtual std::string body();
|
|
|
|
virtual int64_t body_size();
|
|
|
|
virtual int64_t content_length();
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual SrsHttpHandlerMatch* match();
|
2014-04-02 04:03:49 +00:00
|
|
|
virtual void set_url(std::string url);
|
|
|
|
virtual void set_state(SrsHttpParseState state);
|
|
|
|
virtual void set_header(http_parser* header);
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual void set_match(SrsHttpHandlerMatch* match);
|
2014-04-02 04:03:49 +00:00
|
|
|
virtual void append_body(const char* body, int length);
|
2014-04-01 10:40:24 +00:00
|
|
|
};
|
2013-12-07 14:06:53 +00:00
|
|
|
|
2014-04-01 10:40:24 +00:00
|
|
|
/**
|
|
|
|
* wrapper for http-parser,
|
|
|
|
* provides HTTP message originted service.
|
|
|
|
*/
|
|
|
|
class SrsHttpParser
|
|
|
|
{
|
2014-04-02 04:03:49 +00:00
|
|
|
private:
|
|
|
|
http_parser_settings settings;
|
|
|
|
http_parser parser;
|
|
|
|
SrsHttpMessage* msg;
|
2014-04-01 10:40:24 +00:00
|
|
|
public:
|
|
|
|
SrsHttpParser();
|
|
|
|
virtual ~SrsHttpParser();
|
|
|
|
public:
|
2014-04-02 04:03:49 +00:00
|
|
|
/**
|
|
|
|
* initialize the http parser with specified type,
|
|
|
|
* one parser can only parse request or response messages.
|
|
|
|
*/
|
|
|
|
virtual int initialize(enum http_parser_type type);
|
|
|
|
/**
|
|
|
|
* always parse a http message,
|
|
|
|
* that is, the *ppmsg always NOT-NULL when return success.
|
|
|
|
* or error and *ppmsg must be NULL.
|
|
|
|
* @remark, if success, *ppmsg always NOT-NULL, *ppmsg always is_complete().
|
|
|
|
*/
|
|
|
|
virtual int parse_message(SrsSocket* skt, SrsHttpMessage** ppmsg);
|
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* parse the HTTP message to member field: msg.
|
|
|
|
*/
|
|
|
|
virtual int parse_message_imp(SrsSocket* skt);
|
|
|
|
private:
|
|
|
|
static int on_message_begin(http_parser* parser);
|
|
|
|
static int on_headers_complete(http_parser* parser);
|
|
|
|
static int on_message_complete(http_parser* parser);
|
|
|
|
static int on_url(http_parser* parser, const char* at, size_t length);
|
|
|
|
static int on_header_field(http_parser* parser, const char* at, size_t length);
|
|
|
|
static int on_header_value(http_parser* parser, const char* at, size_t length);
|
|
|
|
static int on_body(http_parser* parser, const char* at, size_t length);
|
2014-04-01 10:40:24 +00:00
|
|
|
};
|
2013-12-07 14:06:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* used to resolve the http uri.
|
|
|
|
*/
|
|
|
|
class SrsHttpUri
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
std::string url;
|
|
|
|
std::string schema;
|
|
|
|
std::string host;
|
|
|
|
int port;
|
|
|
|
std::string path;
|
2014-04-03 03:49:14 +00:00
|
|
|
std::string query;
|
2013-12-07 14:06:53 +00:00
|
|
|
public:
|
2014-03-18 03:32:58 +00:00
|
|
|
SrsHttpUri();
|
|
|
|
virtual ~SrsHttpUri();
|
2013-12-07 14:06:53 +00:00
|
|
|
public:
|
2014-03-18 03:32:58 +00:00
|
|
|
/**
|
|
|
|
* initialize the http uri.
|
|
|
|
*/
|
|
|
|
virtual int initialize(std::string _url);
|
2013-12-07 14:06:53 +00:00
|
|
|
public:
|
|
|
|
virtual const char* get_url();
|
|
|
|
virtual const char* get_schema();
|
|
|
|
virtual const char* get_host();
|
|
|
|
virtual int get_port();
|
2013-12-07 15:23:19 +00:00
|
|
|
virtual const char* get_path();
|
2014-04-03 03:49:14 +00:00
|
|
|
virtual const char* get_query();
|
2013-12-07 14:06:53 +00:00
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* get the parsed url field.
|
|
|
|
* @return return empty string if not set.
|
|
|
|
*/
|
|
|
|
virtual std::string get_uri_field(std::string uri, http_parser_url* hp_u, http_parser_url_fields field);
|
|
|
|
};
|
|
|
|
|
2013-12-07 12:51:27 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|