2014-03-31 03:35:22 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
|
|
|
Copyright (c) 2013-2014 winlin
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <srs_app_http_conn.hpp>
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
#include <sstream>
|
|
|
|
using namespace std;
|
|
|
|
|
2014-03-31 03:35:22 +00:00
|
|
|
#include <srs_kernel_log.hpp>
|
|
|
|
#include <srs_kernel_error.hpp>
|
2014-04-01 04:36:56 +00:00
|
|
|
#include <srs_app_socket.hpp>
|
2014-04-01 06:28:19 +00:00
|
|
|
#include <srs_app_http.hpp>
|
|
|
|
#include <srs_kernel_buffer.hpp>
|
2014-04-01 04:36:56 +00:00
|
|
|
|
|
|
|
#define SRS_HTTP_HEADER_BUFFER 1024
|
2014-03-31 03:35:22 +00:00
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
SrsHttpRequest::SrsHttpRequest()
|
|
|
|
{
|
|
|
|
body = new SrsBuffer();
|
|
|
|
state = SrsHttpParseStateInit;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpRequest::~SrsHttpRequest()
|
|
|
|
{
|
|
|
|
srs_freep(body);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsHttpRequest::reset()
|
|
|
|
{
|
|
|
|
state = SrsHttpParseStateInit;
|
|
|
|
body->clear();
|
|
|
|
url = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsHttpRequest::is_complete()
|
|
|
|
{
|
|
|
|
return state == SrsHttpParseStateComplete;
|
|
|
|
}
|
|
|
|
|
2014-03-31 03:35:22 +00:00
|
|
|
SrsHttpConn::SrsHttpConn(SrsServer* srs_server, st_netfd_t client_stfd)
|
|
|
|
: SrsConnection(srs_server, client_stfd)
|
|
|
|
{
|
2014-04-01 06:28:19 +00:00
|
|
|
req = new SrsHttpRequest();
|
2014-03-31 03:35:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
SrsHttpConn::~SrsHttpConn()
|
2014-03-31 03:35:22 +00:00
|
|
|
{
|
2014-04-01 06:28:19 +00:00
|
|
|
srs_freep(req);
|
2014-03-31 03:35:22 +00:00
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
int SrsHttpConn::do_cycle()
|
2014-03-31 03:35:22 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
if ((ret = get_peer_ip()) != ERROR_SUCCESS) {
|
|
|
|
srs_error("get peer ip failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
srs_trace("http get peer ip success. ip=%s", ip);
|
2014-04-01 04:36:56 +00:00
|
|
|
|
|
|
|
// setup http parser
|
|
|
|
http_parser_settings settings;
|
|
|
|
|
|
|
|
memset(&settings, 0, sizeof(settings));
|
|
|
|
settings.on_message_begin = on_message_begin;
|
|
|
|
settings.on_url = on_url;
|
|
|
|
settings.on_header_field = on_header_field;
|
|
|
|
settings.on_header_value = on_header_value;
|
|
|
|
settings.on_headers_complete = on_headers_complete;
|
|
|
|
settings.on_body = on_body;
|
|
|
|
settings.on_message_complete = on_message_complete;
|
|
|
|
|
|
|
|
http_parser parser;
|
|
|
|
http_parser_init(&parser, HTTP_REQUEST);
|
|
|
|
// callback object ptr.
|
|
|
|
parser.data = (void*)this;
|
|
|
|
|
|
|
|
// underlayer socket
|
|
|
|
SrsSocket skt(stfd);
|
|
|
|
|
|
|
|
for (;;) {
|
2014-04-01 06:28:19 +00:00
|
|
|
if ((ret = parse_request(&skt, &parser, &settings)) != ERROR_SUCCESS) {
|
2014-04-01 04:36:56 +00:00
|
|
|
if (!srs_is_client_gracefully_close(ret)) {
|
|
|
|
srs_error("http client cycle failed. ret=%d", ret);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
int SrsHttpConn::parse_request(SrsSocket* skt, http_parser* parser, http_parser_settings* settings)
|
2014-04-01 04:36:56 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// reset response header.
|
2014-04-01 06:28:19 +00:00
|
|
|
req->reset();
|
2014-04-01 04:36:56 +00:00
|
|
|
|
|
|
|
// parser header.
|
|
|
|
char buf[SRS_HTTP_HEADER_BUFFER];
|
|
|
|
for (;;) {
|
|
|
|
ssize_t nread;
|
|
|
|
if ((ret = skt->read(buf, (size_t)sizeof(buf), &nread)) != ERROR_SUCCESS) {
|
|
|
|
if (!srs_is_client_gracefully_close(ret)) {
|
|
|
|
srs_error("read body from server failed. ret=%d", ret);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t nparsed = http_parser_execute(parser, settings, buf, nread);
|
|
|
|
srs_info("read_size=%d, nparsed=%d", (int)nread, (int)nparsed);
|
|
|
|
|
|
|
|
// check header size.
|
2014-04-01 06:28:19 +00:00
|
|
|
if (req->is_complete()) {
|
|
|
|
srs_trace("http request parsed, method=%d, url=%s, content-length=%"PRId64"",
|
|
|
|
req->header.method, req->url.c_str(), req->header.content_length);
|
2014-04-01 04:36:56 +00:00
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
return process_request(skt);
|
2014-04-01 04:36:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (nparsed != nread) {
|
|
|
|
ret = ERROR_HTTP_PARSE_HEADER;
|
|
|
|
srs_error("parse response error, parsed(%d)!=read(%d), ret=%d", (int)nparsed, (int)nread, ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
int SrsHttpConn::process_request(SrsSocket* skt)
|
2014-04-01 04:36:56 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2014-03-31 03:35:22 +00:00
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
if (req->header.method == HTTP_OPTIONS) {
|
|
|
|
char data[] = "HTTP/1.1 200 OK" __CRLF
|
|
|
|
"Content-Length: 0"__CRLF
|
|
|
|
"Server: SRS/"RTMP_SIG_SRS_VERSION""__CRLF
|
|
|
|
"Allow: DELETE, GET, HEAD, OPTIONS, POST, PUT"__CRLF
|
|
|
|
"Access-Control-Allow-Origin: *"__CRLF
|
|
|
|
"Access-Control-Allow-Methods: GET, POST, HEAD, PUT, DELETE"__CRLF
|
|
|
|
"Access-Control-Allow-Headers: Cache-Control,X-Proxy-Authorization,X-Requested-With,Content-Type"__CRLF
|
|
|
|
"Content-Type: text/html;charset=utf-8"__CRLFCRLF
|
|
|
|
"";
|
|
|
|
return skt->write(data, sizeof(data), NULL);
|
|
|
|
} else {
|
|
|
|
std::string tilte = "SRS/"RTMP_SIG_SRS_VERSION;
|
|
|
|
tilte += " hello http/1.1~\n";
|
2014-03-31 03:35:22 +00:00
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
std::stringstream ss;
|
|
|
|
ss << "HTTP/1.1 200 OK " << __CRLF
|
|
|
|
<< "Content-Length: "<< tilte.length() + req->body->size() << __CRLF
|
|
|
|
<< "Server: SRS/"RTMP_SIG_SRS_VERSION"" << __CRLF
|
|
|
|
<< "Allow: DELETE, GET, HEAD, OPTIONS, POST, PUT" << __CRLF
|
|
|
|
<< "Access-Control-Allow-Origin: *" << __CRLF
|
|
|
|
<< "Access-Control-Allow-Methods: GET, POST, HEAD, PUT, DELETE" << __CRLF
|
|
|
|
<< "Access-Control-Allow-Headers: Cache-Control,X-Proxy-Authorization,X-Requested-With,Content-Type" << __CRLF
|
|
|
|
<< "Content-Type: text/html;charset=utf-8" << __CRLFCRLF
|
|
|
|
<< tilte << (req->body->empty()? "":req->body->bytes())
|
|
|
|
<< "";
|
|
|
|
return skt->write(ss.str().c_str(), ss.str().length(), NULL);
|
|
|
|
}
|
2014-04-01 04:36:56 +00:00
|
|
|
|
2014-03-31 03:35:22 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2014-04-01 04:36:56 +00:00
|
|
|
|
|
|
|
int SrsHttpConn::on_message_begin(http_parser* parser)
|
|
|
|
{
|
|
|
|
SrsHttpConn* obj = (SrsHttpConn*)parser->data;
|
2014-04-01 06:28:19 +00:00
|
|
|
obj->req->state = SrsHttpParseStateStart;
|
|
|
|
|
|
|
|
srs_info("***MESSAGE BEGIN***");
|
|
|
|
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::on_headers_complete(http_parser* parser)
|
|
|
|
{
|
|
|
|
SrsHttpConn* obj = (SrsHttpConn*)parser->data;
|
2014-04-01 06:28:19 +00:00
|
|
|
memcpy(&obj->req->header, parser, sizeof(http_parser));
|
|
|
|
|
|
|
|
srs_info("***HEADERS COMPLETE***");
|
2014-04-01 04:36:56 +00:00
|
|
|
|
|
|
|
// see http_parser.c:1570, return 1 to skip body.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::on_message_complete(http_parser* parser)
|
|
|
|
{
|
|
|
|
SrsHttpConn* obj = (SrsHttpConn*)parser->data;
|
|
|
|
// save the parser when header parse completed.
|
2014-04-01 06:28:19 +00:00
|
|
|
obj->req->state = SrsHttpParseStateComplete;
|
|
|
|
|
|
|
|
srs_info("***MESSAGE COMPLETE***\n");
|
|
|
|
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::on_url(http_parser* parser, const char* at, size_t length)
|
|
|
|
{
|
|
|
|
SrsHttpConn* obj = (SrsHttpConn*)parser->data;
|
2014-04-01 06:28:19 +00:00
|
|
|
|
|
|
|
if (length > 0) {
|
|
|
|
obj->req->url.append(at, (int)length);
|
|
|
|
}
|
|
|
|
|
|
|
|
srs_info("Method: %d, Url: %.*s", parser->method, (int)length, at);
|
|
|
|
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
int SrsHttpConn::on_header_field(http_parser* /*parser*/, const char* at, size_t length)
|
2014-04-01 04:36:56 +00:00
|
|
|
{
|
2014-04-01 06:28:19 +00:00
|
|
|
srs_info("Header field: %.*s", (int)length, at);
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-01 06:28:19 +00:00
|
|
|
int SrsHttpConn::on_header_value(http_parser* /*parser*/, const char* at, size_t length)
|
2014-04-01 04:36:56 +00:00
|
|
|
{
|
2014-04-01 06:28:19 +00:00
|
|
|
srs_info("Header value: %.*s", (int)length, at);
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::on_body(http_parser* parser, const char* at, size_t length)
|
|
|
|
{
|
|
|
|
SrsHttpConn* obj = (SrsHttpConn*)parser->data;
|
2014-04-01 06:28:19 +00:00
|
|
|
|
|
|
|
if (length > 0) {
|
|
|
|
obj->req->body->append(at, (int)length);
|
|
|
|
}
|
|
|
|
|
|
|
|
srs_info("Body: %.*s", (int)length, at);
|
|
|
|
|
2014-04-01 04:36:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|