2014-04-04 15:06:46 +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-15 06:01:57 +00:00
|
|
|
#ifdef SRS_AUTO_HTTP_SERVER
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
2014-05-26 05:57:08 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
using namespace std;
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
#include <srs_kernel_log.hpp>
|
|
|
|
#include <srs_kernel_error.hpp>
|
|
|
|
#include <srs_app_socket.hpp>
|
|
|
|
#include <srs_app_http.hpp>
|
|
|
|
#include <srs_core_autofree.hpp>
|
|
|
|
#include <srs_app_json.hpp>
|
|
|
|
#include <srs_app_config.hpp>
|
2014-05-28 07:37:06 +00:00
|
|
|
#include <srs_kernel_flv.hpp>
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2014-04-05 05:14:59 +00:00
|
|
|
#define SRS_HTTP_DEFAULT_PAGE "index.html"
|
|
|
|
|
2014-04-04 15:06:46 +00:00
|
|
|
SrsHttpRoot::SrsHttpRoot()
|
|
|
|
{
|
|
|
|
// TODO: FIXME: support reload vhosts.
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpRoot::~SrsHttpRoot()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpRoot::initialize()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2014-04-04 15:16:31 +00:00
|
|
|
|
|
|
|
bool default_root_exists = false;
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
// add other virtual path
|
|
|
|
SrsConfDirective* root = _srs_config->get_root();
|
|
|
|
for (int i = 0; i < (int)root->directives.size(); i++) {
|
|
|
|
SrsConfDirective* conf = root->at(i);
|
|
|
|
|
|
|
|
if (!conf->is_vhost()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string vhost = conf->arg0();
|
|
|
|
if (!_srs_config->get_vhost_http_enabled(vhost)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string mount = _srs_config->get_vhost_http_mount(vhost);
|
|
|
|
std::string dir = _srs_config->get_vhost_http_dir(vhost);
|
|
|
|
|
|
|
|
handlers.push_back(new SrsHttpVhost(vhost, mount, dir));
|
2014-04-04 15:16:31 +00:00
|
|
|
|
|
|
|
if (mount == "/") {
|
|
|
|
default_root_exists = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!default_root_exists) {
|
|
|
|
// add root
|
|
|
|
handlers.push_back(new SrsHttpVhost(
|
|
|
|
"__http__", "/", _srs_config->get_http_stream_dir()));
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-05 05:14:59 +00:00
|
|
|
int SrsHttpRoot::best_match(const char* path, int length, SrsHttpHandlerMatch** ppmatch)
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
2014-04-05 05:14:59 +00:00
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// find the best matched child handler.
|
|
|
|
std::vector<SrsHttpHandler*>::iterator it;
|
|
|
|
for (it = handlers.begin(); it != handlers.end(); ++it) {
|
|
|
|
SrsHttpHandler* h = *it;
|
|
|
|
|
|
|
|
// search all child handlers.
|
|
|
|
h->best_match(path, length, ppmatch);
|
|
|
|
}
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2014-04-05 05:14:59 +00:00
|
|
|
// if already matched by child, return.
|
|
|
|
if (*ppmatch) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// not matched, error.
|
|
|
|
return ERROR_HTTP_HANDLER_MATCH_URL;
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsHttpRoot::is_handler_valid(SrsHttpMessage* req, int& status_code, std::string& reason_phrase)
|
|
|
|
{
|
|
|
|
status_code = HTTP_InternalServerError;
|
|
|
|
reason_phrase = HTTP_InternalServerError_str;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpRoot::do_process_request(SrsSocket* skt, SrsHttpMessage* req)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpVhost::SrsHttpVhost(std::string vhost, std::string mount, std::string dir)
|
|
|
|
{
|
|
|
|
_vhost = vhost;
|
|
|
|
_mount = mount;
|
|
|
|
_dir = dir;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpVhost::~SrsHttpVhost()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsHttpVhost::can_handle(const char* path, int length, const char** /*pchild*/)
|
|
|
|
{
|
2014-04-05 05:14:59 +00:00
|
|
|
return srs_path_like(_mount.c_str(), path, length);
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsHttpVhost::is_handler_valid(SrsHttpMessage* req, int& status_code, std::string& reason_phrase)
|
|
|
|
{
|
2014-04-05 05:14:59 +00:00
|
|
|
std::string fullpath = get_request_file(req);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
if (::access(fullpath.c_str(), F_OK | R_OK) < 0) {
|
|
|
|
srs_warn("check file %s does not exists", fullpath.c_str());
|
|
|
|
|
|
|
|
status_code = HTTP_NotFound;
|
|
|
|
reason_phrase = HTTP_NotFound_str;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpVhost::do_process_request(SrsSocket* skt, SrsHttpMessage* req)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
2014-04-05 05:14:59 +00:00
|
|
|
std::string fullpath = get_request_file(req);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2014-04-22 04:15:13 +00:00
|
|
|
if (srs_string_ends_with(fullpath, ".ts")) {
|
2014-05-23 06:26:11 +00:00
|
|
|
return response_ts_file(skt, req, fullpath);
|
2014-05-23 09:23:33 +00:00
|
|
|
} else if (srs_string_ends_with(fullpath, ".flv") || srs_string_ends_with(fullpath, ".fhv")) {
|
2014-05-26 05:57:08 +00:00
|
|
|
std::string start = req->query_get("start");
|
|
|
|
if (start.empty()) {
|
|
|
|
return response_flv_file(skt, req, fullpath);
|
|
|
|
}
|
|
|
|
|
|
|
|
int offset = ::atoi(start.c_str());
|
|
|
|
if (offset <= 0) {
|
|
|
|
return response_flv_file(skt, req, fullpath);
|
|
|
|
}
|
|
|
|
|
|
|
|
return response_flv_file2(skt, req, fullpath, offset);
|
2014-05-23 06:26:11 +00:00
|
|
|
} else {
|
|
|
|
return response_regular_file(skt, req, fullpath);
|
|
|
|
}
|
2014-04-22 04:15:13 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpVhost::response_regular_file(SrsSocket* skt, SrsHttpMessage* req, string fullpath)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// TODO: FIXME: refine the file stream.
|
|
|
|
int fd = ::open(fullpath.c_str(), O_RDONLY);
|
|
|
|
if (fd < 0) {
|
|
|
|
ret = ERROR_HTTP_OPEN_FILE;
|
|
|
|
srs_warn("open file %s failed, ret=%d", fullpath.c_str(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t length = (int64_t)::lseek(fd, 0, SEEK_END);
|
|
|
|
::lseek(fd, 0, SEEK_SET);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
char* buf = new char[length];
|
|
|
|
SrsAutoFree(char, buf);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
// TODO: FIXME: use st_read.
|
|
|
|
if (::read(fd, buf, length) < 0) {
|
2014-04-04 15:06:46 +00:00
|
|
|
::close(fd);
|
2014-05-23 06:26:11 +00:00
|
|
|
ret = ERROR_HTTP_READ_FILE;
|
|
|
|
srs_warn("read file %s failed, ret=%d", fullpath.c_str(), ret);
|
2014-04-04 15:06:46 +00:00
|
|
|
return ret;
|
2014-05-23 06:26:11 +00:00
|
|
|
}
|
|
|
|
::close(fd);
|
|
|
|
|
|
|
|
std::string str;
|
|
|
|
str.append(buf, length);
|
|
|
|
|
|
|
|
if (srs_string_ends_with(fullpath, ".ts")) {
|
|
|
|
return res_mpegts(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".m3u8")) {
|
|
|
|
return res_m3u8(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".xml")) {
|
|
|
|
return res_xml(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".js")) {
|
|
|
|
return res_javascript(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".json")) {
|
|
|
|
return res_json(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".swf")) {
|
|
|
|
return res_swf(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".css")) {
|
|
|
|
return res_css(skt, req, str);
|
|
|
|
} else if (srs_string_ends_with(fullpath, ".ico")) {
|
|
|
|
return res_ico(skt, req, str);
|
2014-04-04 15:06:46 +00:00
|
|
|
} else {
|
2014-05-23 06:26:11 +00:00
|
|
|
return res_text(skt, req, str);
|
|
|
|
}
|
2014-04-22 04:15:13 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-05-23 09:23:33 +00:00
|
|
|
int SrsHttpVhost::response_flv_file(SrsSocket* skt, SrsHttpMessage* req, string fullpath)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// TODO: FIXME: use more advance cache.
|
|
|
|
// for ts video large file, use bytes to write it.
|
|
|
|
int fd = ::open(fullpath.c_str(), O_RDONLY);
|
|
|
|
if (fd < 0) {
|
|
|
|
ret = ERROR_HTTP_OPEN_FILE;
|
|
|
|
srs_warn("open file %s failed, ret=%d", fullpath.c_str(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t length = (int64_t)::lseek(fd, 0, SEEK_END);
|
|
|
|
::lseek(fd, 0, SEEK_SET);
|
|
|
|
|
|
|
|
// write http header for ts.
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
res_status_line(ss)->res_content_type_flv(ss)
|
|
|
|
->res_content_length(ss, (int)length);
|
|
|
|
|
|
|
|
if (req->requires_crossdomain()) {
|
|
|
|
res_enable_crossdomain(ss);
|
|
|
|
}
|
|
|
|
|
|
|
|
res_header_eof(ss);
|
|
|
|
|
|
|
|
// flush http header to peer
|
|
|
|
if ((ret = res_flush(skt, ss)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write body.
|
|
|
|
int64_t left = length;
|
|
|
|
char* buf = req->http_ts_send_buffer();
|
|
|
|
|
|
|
|
while (left > 0) {
|
|
|
|
ssize_t nread = -1;
|
|
|
|
// TODO: FIXME: use st_read.
|
|
|
|
if ((nread = ::read(fd, buf, HTTP_TS_SEND_BUFFER_SIZE)) < 0) {
|
|
|
|
ret = ERROR_HTTP_READ_FILE;
|
|
|
|
srs_warn("read file %s failed, ret=%d", fullpath.c_str(), ret);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
left -= nread;
|
|
|
|
if ((ret = skt->write(buf, nread, NULL)) != ERROR_SUCCESS) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
::close(fd);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2014-05-26 05:57:08 +00:00
|
|
|
|
|
|
|
int SrsHttpVhost::response_flv_file2(SrsSocket* skt, SrsHttpMessage* req, string fullpath, int offset)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
SrsFileStream fs;
|
|
|
|
|
|
|
|
// open flv file
|
|
|
|
if ((ret = fs.open_read(fullpath)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (offset > fs.filesize()) {
|
|
|
|
ret = ERROR_HTTP_FLV_OFFSET_OVERFLOW;
|
|
|
|
srs_warn("http flv streaming %s overflow. size=%"PRId64", offset=%d, ret=%d",
|
|
|
|
fullpath.c_str(), fs.filesize(), offset, ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsFlvFastDecoder ffd;
|
|
|
|
|
|
|
|
// open fast decoder
|
|
|
|
if ((ret = ffd.initialize(&fs)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// save header, send later.
|
|
|
|
char* flv_header = NULL;
|
|
|
|
int flv_size = 0;
|
|
|
|
|
|
|
|
// send flv header
|
|
|
|
if ((ret = ffd.read_header(&flv_header, &flv_size)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
SrsAutoFree(char, flv_header);
|
|
|
|
|
|
|
|
// save sequence header, send later
|
|
|
|
char* sh_data = NULL;
|
|
|
|
int sh_size = 0;
|
|
|
|
|
|
|
|
if (true) {
|
|
|
|
// send sequence header
|
|
|
|
int64_t start = 0;
|
|
|
|
if ((ret = ffd.read_sequence_header(&start, &sh_size)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (sh_size <= 0) {
|
|
|
|
ret = ERROR_HTTP_FLV_SEQUENCE_HEADER;
|
|
|
|
srs_warn("http flv streaming no sequence header. size=%d, ret=%d", sh_size, ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sh_data = new char[sh_size];
|
|
|
|
SrsAutoFree(char, sh_data);
|
|
|
|
if ((ret = fs.read(sh_data, sh_size, NULL)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// seek to data offset
|
|
|
|
int64_t left = fs.filesize() - offset;
|
|
|
|
|
|
|
|
// write http header for ts.
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
res_status_line(ss)->res_content_type_flv(ss)
|
|
|
|
->res_content_length(ss, (int)(flv_size + sh_size + left));
|
|
|
|
|
|
|
|
if (req->requires_crossdomain()) {
|
|
|
|
res_enable_crossdomain(ss);
|
|
|
|
}
|
|
|
|
|
|
|
|
res_header_eof(ss);
|
|
|
|
|
|
|
|
// flush http header to peer
|
|
|
|
if ((ret = res_flush(skt, ss)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flv_size > 0 && (ret = skt->write(flv_header, flv_size, NULL)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (sh_size > 0 && (ret = skt->write(sh_data, sh_size, NULL)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write body.
|
|
|
|
char* buf = req->http_ts_send_buffer();
|
|
|
|
if ((ret = ffd.lseek(offset)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send data
|
|
|
|
while (left > 0) {
|
|
|
|
ssize_t nread = -1;
|
|
|
|
if ((ret = fs.read(buf, HTTP_TS_SEND_BUFFER_SIZE, &nread)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
left -= nread;
|
|
|
|
if ((ret = skt->write(buf, nread, NULL)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2014-05-23 09:23:33 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
int SrsHttpVhost::response_ts_file(SrsSocket* skt, SrsHttpMessage* req, string fullpath)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// TODO: FIXME: use more advance cache.
|
|
|
|
// for ts video large file, use bytes to write it.
|
|
|
|
int fd = ::open(fullpath.c_str(), O_RDONLY);
|
|
|
|
if (fd < 0) {
|
|
|
|
ret = ERROR_HTTP_OPEN_FILE;
|
|
|
|
srs_warn("open file %s failed, ret=%d", fullpath.c_str(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t length = (int64_t)::lseek(fd, 0, SEEK_END);
|
|
|
|
::lseek(fd, 0, SEEK_SET);
|
|
|
|
|
|
|
|
// write http header for ts.
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
res_status_line(ss)->res_content_type_mpegts(ss)
|
|
|
|
->res_content_length(ss, (int)length);
|
2014-04-22 04:15:13 +00:00
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
if (req->requires_crossdomain()) {
|
|
|
|
res_enable_crossdomain(ss);
|
|
|
|
}
|
|
|
|
|
|
|
|
res_header_eof(ss);
|
|
|
|
|
|
|
|
// flush http header to peer
|
|
|
|
if ((ret = res_flush(skt, ss)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write body.
|
|
|
|
int64_t left = length;
|
|
|
|
char* buf = req->http_ts_send_buffer();
|
|
|
|
|
|
|
|
while (left > 0) {
|
|
|
|
ssize_t nread = -1;
|
2014-04-22 04:15:13 +00:00
|
|
|
// TODO: FIXME: use st_read.
|
2014-05-23 06:26:11 +00:00
|
|
|
if ((nread = ::read(fd, buf, HTTP_TS_SEND_BUFFER_SIZE)) < 0) {
|
2014-04-22 04:15:13 +00:00
|
|
|
ret = ERROR_HTTP_READ_FILE;
|
|
|
|
srs_warn("read file %s failed, ret=%d", fullpath.c_str(), ret);
|
2014-05-23 09:23:33 +00:00
|
|
|
break;
|
2014-04-22 04:15:13 +00:00
|
|
|
}
|
|
|
|
|
2014-05-23 06:26:11 +00:00
|
|
|
left -= nread;
|
|
|
|
if ((ret = skt->write(buf, nread, NULL)) != ERROR_SUCCESS) {
|
|
|
|
break;
|
2014-04-22 04:15:13 +00:00
|
|
|
}
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
2014-05-23 06:26:11 +00:00
|
|
|
::close(fd);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-05 05:14:59 +00:00
|
|
|
string SrsHttpVhost::get_request_file(SrsHttpMessage* req)
|
|
|
|
{
|
|
|
|
std::string fullpath = _dir + "/";
|
|
|
|
|
|
|
|
// if root, directly use the matched url.
|
|
|
|
if (_mount == "/") {
|
|
|
|
// add the dir
|
|
|
|
fullpath += req->match()->matched_url;
|
|
|
|
// if file speicified, add the file.
|
|
|
|
if (!req->match()->unmatched_url.empty()) {
|
|
|
|
fullpath += "/" + req->match()->unmatched_url;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// virtual path, ignore the virutal path.
|
|
|
|
fullpath += req->match()->unmatched_url;
|
|
|
|
}
|
|
|
|
|
|
|
|
// add default pages.
|
|
|
|
if (srs_string_ends_with(fullpath, "/")) {
|
|
|
|
fullpath += SRS_HTTP_DEFAULT_PAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fullpath;
|
|
|
|
}
|
|
|
|
|
2014-04-04 15:06:46 +00:00
|
|
|
string SrsHttpVhost::vhost()
|
|
|
|
{
|
|
|
|
return _vhost;
|
|
|
|
}
|
|
|
|
|
|
|
|
string SrsHttpVhost::mount()
|
|
|
|
{
|
|
|
|
return _mount;
|
|
|
|
}
|
|
|
|
|
|
|
|
string SrsHttpVhost::dir()
|
|
|
|
{
|
|
|
|
return _dir;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpConn::SrsHttpConn(SrsServer* srs_server, st_netfd_t client_stfd, SrsHttpHandler* _handler)
|
|
|
|
: SrsConnection(srs_server, client_stfd)
|
|
|
|
{
|
|
|
|
parser = new SrsHttpParser();
|
|
|
|
handler = _handler;
|
|
|
|
requires_crossdomain = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpConn::~SrsHttpConn()
|
|
|
|
{
|
|
|
|
srs_freep(parser);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::do_cycle()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
2014-05-30 01:20:51 +00:00
|
|
|
srs_trace("HTTP client ip=%s", ip.c_str());
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
// initialize parser
|
|
|
|
if ((ret = parser->initialize(HTTP_REQUEST)) != ERROR_SUCCESS) {
|
|
|
|
srs_error("http initialize http parser failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// underlayer socket
|
|
|
|
SrsSocket skt(stfd);
|
|
|
|
|
|
|
|
// process http messages.
|
|
|
|
for (;;) {
|
|
|
|
SrsHttpMessage* req = NULL;
|
|
|
|
|
|
|
|
// get a http message
|
|
|
|
if ((ret = parser->parse_message(&skt, &req)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if SUCCESS, always NOT-NULL and completed message.
|
|
|
|
srs_assert(req);
|
|
|
|
srs_assert(req->is_complete());
|
|
|
|
|
|
|
|
// always free it in this scope.
|
2014-05-14 01:41:41 +00:00
|
|
|
SrsAutoFree(SrsHttpMessage, req);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
// ok, handle http request.
|
|
|
|
if ((ret = process_request(&skt, req)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpConn::process_request(SrsSocket* skt, SrsHttpMessage* req)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
// parse uri to schema/server:port/path?query
|
|
|
|
if ((ret = req->parse_uri()) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-05-30 01:20:51 +00:00
|
|
|
srs_trace("HTTP %s %s, content-length=%"PRId64"",
|
|
|
|
req->method_str().c_str(), req->url().c_str(), req->content_length());
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
// TODO: maybe need to parse the url.
|
|
|
|
std::string url = req->path();
|
|
|
|
|
|
|
|
SrsHttpHandlerMatch* p = NULL;
|
|
|
|
if ((ret = handler->best_match(url.data(), url.length(), &p)) != ERROR_SUCCESS) {
|
|
|
|
srs_warn("failed to find the best match handler for url. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if success, p and pstart should be valid.
|
|
|
|
srs_assert(p);
|
|
|
|
srs_assert(p->handler);
|
|
|
|
srs_assert(p->matched_url.length() <= url.length());
|
|
|
|
srs_info("best match handler, matched_url=%s", p->matched_url.c_str());
|
|
|
|
|
|
|
|
req->set_match(p);
|
|
|
|
req->set_requires_crossdomain(requires_crossdomain);
|
|
|
|
|
|
|
|
// use handler to process request.
|
|
|
|
if ((ret = p->handler->process_request(skt, req)) != ERROR_SUCCESS) {
|
|
|
|
srs_warn("handler failed to process http request. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req->requires_crossdomain()) {
|
|
|
|
requires_crossdomain = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|