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-02 10:07:34 +00:00
|
|
|
class SrsHttpMessage;
|
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-02 10:07:34 +00:00
|
|
|
// linux path seprator
|
|
|
|
#define __PATH_SEP '/'
|
|
|
|
|
|
|
|
// 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-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-02 11:10:22 +00:00
|
|
|
* @pnext_path set the next path, if needed.
|
2014-04-02 10:07:34 +00:00
|
|
|
*/
|
2014-04-02 11:10:22 +00:00
|
|
|
virtual bool can_handle(const char* path, int length, const char** pnext_path);
|
2014-04-02 10:07:34 +00:00
|
|
|
/**
|
|
|
|
* use the handler to process the request.
|
|
|
|
*/
|
|
|
|
virtual int process_request(SrsSocket* skt, SrsHttpMessage* req, const char* path, int length);
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* find the best matched handler
|
|
|
|
*/
|
|
|
|
virtual int best_match(const char* path, int length, SrsHttpHandler** phandler, const char** pstart, int* plength);
|
2014-04-02 11:10:22 +00:00
|
|
|
public:
|
|
|
|
virtual SrsHttpHandler* res_status_line(std::stringstream& ss);
|
|
|
|
virtual SrsHttpHandler* res_content_type(std::stringstream& ss);
|
|
|
|
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-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-01 10:40:24 +00:00
|
|
|
|
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-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();
|
|
|
|
virtual std::string body();
|
|
|
|
virtual int64_t body_size();
|
|
|
|
virtual int64_t content_length();
|
|
|
|
virtual void set_url(std::string url);
|
|
|
|
virtual void set_state(SrsHttpParseState state);
|
|
|
|
virtual void set_header(http_parser* header);
|
|
|
|
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;
|
|
|
|
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();
|
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
|