2014-04-04 15:06:46 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
2014-12-31 12:32:09 +00:00
|
|
|
Copyright (c) 2013-2015 winlin
|
2014-04-04 15:06:46 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#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>
|
2014-07-26 12:07:12 +00:00
|
|
|
#include <srs_app_st_socket.hpp>
|
2014-04-04 15:06:46 +00:00
|
|
|
#include <srs_core_autofree.hpp>
|
|
|
|
#include <srs_app_config.hpp>
|
2014-06-08 05:03:03 +00:00
|
|
|
#include <srs_kernel_utility.hpp>
|
2015-01-18 01:12:53 +00:00
|
|
|
#include <srs_kernel_file.hpp>
|
|
|
|
#include <srs_kernel_flv.hpp>
|
2015-01-18 10:39:53 +00:00
|
|
|
#include <srs_protocol_rtmp.hpp>
|
2015-01-18 11:49:03 +00:00
|
|
|
#include <srs_app_source.hpp>
|
|
|
|
#include <srs_protocol_msg_array.hpp>
|
2015-01-18 01:12:53 +00:00
|
|
|
|
|
|
|
SrsVodStream::SrsVodStream(string root_dir)
|
|
|
|
: SrsGoHttpFileServer(root_dir)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsVodStream::~SrsVodStream()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsVodStream::serve_flv_stream(ISrsGoHttpResponseWriter* w, SrsHttpMessage* r, string fullpath, int offset)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
SrsFileReader fs;
|
|
|
|
|
|
|
|
// open flv file
|
|
|
|
if ((ret = fs.open(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;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsFlvVodStreamDecoder ffd;
|
|
|
|
|
|
|
|
// open fast decoder
|
|
|
|
if ((ret = ffd.initialize(&fs)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// save header, send later.
|
|
|
|
char flv_header[13];
|
|
|
|
|
|
|
|
// send flv header
|
|
|
|
if ((ret = ffd.read_header_ext(flv_header)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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_summary(&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.
|
|
|
|
w->header()->set_content_length((int)(sizeof(flv_header) + sh_size + left));
|
|
|
|
w->header()->set_content_type("video/x-flv");
|
|
|
|
|
|
|
|
// write flv header and sequence header.
|
|
|
|
if ((ret = w->write(flv_header, sizeof(flv_header))) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (sh_size > 0 && (ret = w->write(sh_data, sh_size)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write body.
|
|
|
|
if ((ret = ffd.lseek(offset)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send data
|
2015-01-18 01:16:14 +00:00
|
|
|
if ((ret = copy(w, &fs, r, left)) != ERROR_SUCCESS) {
|
2015-01-18 01:12:53 +00:00
|
|
|
srs_warn("read flv=%s size=%d failed, ret=%d", fullpath.c_str(), left, ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2015-01-18 11:49:03 +00:00
|
|
|
SrsFlvStreamWriter::SrsFlvStreamWriter(ISrsGoHttpResponseWriter* w)
|
|
|
|
{
|
|
|
|
writer = w;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsFlvStreamWriter::~SrsFlvStreamWriter()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsFlvStreamWriter::open(std::string /*file*/)
|
|
|
|
{
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsFlvStreamWriter::close()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsFlvStreamWriter::is_open()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsFlvStreamWriter::tellg()
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsFlvStreamWriter::write(void* buf, size_t count, ssize_t* pnwrite)
|
|
|
|
{
|
|
|
|
if (pnwrite) {
|
|
|
|
*pnwrite = count;
|
|
|
|
}
|
|
|
|
return writer->write((char*)buf, (int)count);
|
|
|
|
}
|
|
|
|
|
2015-01-18 10:50:15 +00:00
|
|
|
SrsLiveStream::SrsLiveStream(SrsSource* s, SrsRequest* r)
|
2015-01-18 10:00:40 +00:00
|
|
|
{
|
2015-01-18 10:50:15 +00:00
|
|
|
source = s;
|
|
|
|
req = r->copy();
|
2015-01-18 10:00:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsLiveStream::~SrsLiveStream()
|
|
|
|
{
|
2015-01-18 10:50:15 +00:00
|
|
|
srs_freep(req);
|
2015-01-18 10:00:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int SrsLiveStream::serve_http(ISrsGoHttpResponseWriter* w, SrsHttpMessage* r)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2015-01-18 11:49:03 +00:00
|
|
|
|
|
|
|
// create consumer of souce.
|
|
|
|
SrsConsumer* consumer = NULL;
|
|
|
|
if ((ret = source->create_consumer(consumer)) != ERROR_SUCCESS) {
|
|
|
|
srs_error("http: create consumer failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
SrsAutoFree(SrsConsumer, consumer);
|
|
|
|
srs_verbose("http: consumer created success.");
|
|
|
|
|
|
|
|
SrsMessageArray msgs(SRS_PERF_MW_MSGS);
|
|
|
|
// TODO: FIMXE: add pithy print.
|
|
|
|
|
|
|
|
// write http header for ts.
|
|
|
|
w->header()->set_content_length((int64_t)2 * 1024 * 1024 * 1024);
|
|
|
|
w->header()->set_content_type("video/x-flv");
|
|
|
|
|
|
|
|
// the memory writer.
|
|
|
|
SrsFlvStreamWriter writer(w);
|
|
|
|
|
|
|
|
SrsFlvEncoder enc;
|
|
|
|
if ((ret = enc.initialize(&writer)) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write flv header.
|
|
|
|
if ((ret = enc.write_header()) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
// get messages from consumer.
|
|
|
|
// each msg in msgs.msgs must be free, for the SrsMessageArray never free them.
|
|
|
|
int count = 0;
|
|
|
|
if ((ret = consumer->dump_packets(&msgs, count)) != ERROR_SUCCESS) {
|
|
|
|
srs_error("get messages from consumer failed. ret=%d", ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count <= 0) {
|
|
|
|
srs_info("mw sleep %dms for no msg", mw_sleep);
|
|
|
|
// directly use sleep, donot use consumer wait.
|
|
|
|
st_usleep(SRS_CONSTS_RTMP_PULSE_TIMEOUT_US);
|
|
|
|
|
|
|
|
// ignore when nothing got.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
srs_info("got %d msgs, min=%d, mw=%d", count,
|
|
|
|
SRS_PERF_MW_MIN_MSGS, SRS_CONSTS_RTMP_PULSE_TIMEOUT_US / 1000);
|
|
|
|
|
|
|
|
// sendout all messages.
|
|
|
|
ret = send_messages(&enc, msgs.msgs, count);
|
|
|
|
|
|
|
|
// free the messages.
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
SrsSharedPtrMessage* msg = msgs.msgs[i];
|
|
|
|
srs_freep(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
// check send error code.
|
|
|
|
if (ret != ERROR_SUCCESS) {
|
|
|
|
if (!srs_is_client_gracefully_close(ret)) {
|
|
|
|
srs_error("send messages to client failed. ret=%d", ret);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsLiveStream::send_messages(SrsFlvEncoder* enc, SrsSharedPtrMessage** msgs, int nb_msgs)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
|
|
|
for (int i = 0; i < nb_msgs; i++) {
|
|
|
|
SrsSharedPtrMessage* msg = msgs[i];
|
|
|
|
|
|
|
|
if (msg->is_audio()) {
|
|
|
|
ret = enc->write_audio(msg->timestamp, msg->payload, msg->size);
|
|
|
|
} else if (msg->is_video()) {
|
|
|
|
ret = enc->write_video(msg->timestamp, msg->payload, msg->size);
|
|
|
|
} else {
|
|
|
|
ret = enc->write_metadata(msg->timestamp, msg->payload, msg->size);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-18 10:00:40 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-18 12:07:54 +00:00
|
|
|
SrsLiveEntry::SrsLiveEntry()
|
|
|
|
{
|
|
|
|
stream = NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
SrsHttpServer::SrsHttpServer()
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
SrsHttpServer::~SrsHttpServer()
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
2015-01-18 12:07:54 +00:00
|
|
|
std::map<std::string, SrsLiveEntry*>::iterator it;
|
|
|
|
for (it = flvs.begin(); it != flvs.end(); ++it) {
|
|
|
|
SrsLiveEntry* entry = it->second;
|
|
|
|
srs_freep(entry);
|
|
|
|
}
|
2015-01-18 10:00:40 +00:00
|
|
|
flvs.clear();
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
int SrsHttpServer::initialize()
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2014-04-04 15:16:31 +00:00
|
|
|
|
2015-01-18 10:00:40 +00:00
|
|
|
// static file
|
|
|
|
// flv vod streaming.
|
|
|
|
if ((ret = mount_static_file()) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// remux rtmp to flv live streaming
|
|
|
|
if ((ret = mount_flv_streaming()) != ERROR_SUCCESS) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-18 10:39:53 +00:00
|
|
|
int SrsHttpServer::mount(SrsSource* s, SrsRequest* r)
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
2015-01-18 10:50:15 +00:00
|
|
|
if (flvs.find(r->vhost) == flvs.end()) {
|
|
|
|
srs_info("ignore mount flv stream for disabled");
|
2015-01-18 10:39:53 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-01-18 12:07:54 +00:00
|
|
|
|
|
|
|
SrsLiveEntry* entry = flvs[r->vhost];
|
|
|
|
|
|
|
|
// TODO: FIXME: supports reload.
|
|
|
|
if (entry->stream) {
|
|
|
|
entry->stream->entry->enabled = true;
|
|
|
|
return ret;
|
|
|
|
}
|
2015-01-18 10:50:15 +00:00
|
|
|
|
2015-01-18 12:07:54 +00:00
|
|
|
std::string mount = entry->mount;
|
2015-01-18 10:50:15 +00:00
|
|
|
|
|
|
|
// replace the vhost variable
|
|
|
|
mount = srs_string_replace(mount, "[vhost]", r->vhost);
|
|
|
|
mount = srs_string_replace(mount, "[app]", r->app);
|
|
|
|
mount = srs_string_replace(mount, "[stream]", r->stream);
|
|
|
|
|
|
|
|
// remove the default vhost mount
|
2015-01-18 11:49:03 +00:00
|
|
|
mount = srs_string_replace(mount, SRS_CONSTS_RTMP_DEFAULT_VHOST"/", "/");
|
2015-01-18 10:39:53 +00:00
|
|
|
|
2015-01-18 12:07:54 +00:00
|
|
|
entry->stream = new SrsLiveStream(s, r);
|
|
|
|
|
2015-01-18 10:50:15 +00:00
|
|
|
// mount the http flv stream.
|
2015-01-18 12:07:54 +00:00
|
|
|
if ((ret = mux.handle(mount, entry->stream)) != ERROR_SUCCESS) {
|
2015-01-18 10:50:15 +00:00
|
|
|
srs_error("http: mount flv stream for vhost=%s failed. ret=%d", r->vhost.c_str(), ret);
|
2015-01-18 10:39:53 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-01-18 10:50:15 +00:00
|
|
|
srs_trace("http: mount flv stream for vhost=%s, mount=%s", r->vhost.c_str(), mount.c_str());
|
2015-01-18 10:39:53 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsHttpServer::unmount(SrsSource* s, SrsRequest* r)
|
|
|
|
{
|
|
|
|
if (flvs.find(r->vhost) == flvs.end()) {
|
|
|
|
srs_info("ignore unmount flv stream for disabled");
|
|
|
|
return;
|
|
|
|
}
|
2015-01-18 12:07:54 +00:00
|
|
|
|
|
|
|
SrsLiveEntry* entry = flvs[r->vhost];
|
|
|
|
entry->stream->entry->enabled = false;
|
2015-01-18 10:39:53 +00:00
|
|
|
}
|
|
|
|
|
2015-01-18 10:00:40 +00:00
|
|
|
int SrsHttpServer::on_reload_vhost_http_updated()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
// TODO: FIXME: implements it.
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpServer::on_reload_vhost_http_flv_updated()
|
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
// TODO: FIXME: implements it.
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsHttpServer::mount_static_file()
|
|
|
|
{
|
|
|
|
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
|
|
|
|
2015-01-18 10:00:40 +00:00
|
|
|
// http static file and flv vod stream mount for each vhost.
|
2014-04-04 15:06:46 +00:00
|
|
|
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);
|
2015-01-18 08:38:26 +00:00
|
|
|
|
|
|
|
// replace the vhost variable
|
|
|
|
mount = srs_string_replace(mount, "[vhost]", vhost);
|
2015-01-18 10:00:40 +00:00
|
|
|
|
|
|
|
// remove the default vhost mount
|
2015-01-18 11:49:03 +00:00
|
|
|
mount = srs_string_replace(mount, SRS_CONSTS_RTMP_DEFAULT_VHOST"/", "/");
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2015-01-18 05:34:26 +00:00
|
|
|
// the dir mount must always ends with "/"
|
|
|
|
if (mount != "/" && mount.rfind("/") != mount.length() - 1) {
|
|
|
|
mount += "/";
|
|
|
|
}
|
|
|
|
|
2015-01-18 08:38:26 +00:00
|
|
|
// mount the http of vhost.
|
2015-01-18 01:12:53 +00:00
|
|
|
if ((ret = mux.handle(mount, new SrsVodStream(dir))) != ERROR_SUCCESS) {
|
2015-01-17 15:00:40 +00:00
|
|
|
srs_error("http: mount dir=%s for vhost=%s failed. ret=%d", dir.c_str(), vhost.c_str(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-04-04 15:16:31 +00:00
|
|
|
|
|
|
|
if (mount == "/") {
|
|
|
|
default_root_exists = true;
|
2015-01-18 05:34:26 +00:00
|
|
|
srs_warn("http: root mount to %s", dir.c_str());
|
2014-04-04 15:16:31 +00:00
|
|
|
}
|
2015-01-18 10:50:15 +00:00
|
|
|
srs_trace("http: vhost=%s mount to %s", vhost.c_str(), mount.c_str());
|
2014-04-04 15:16:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!default_root_exists) {
|
|
|
|
// add root
|
2015-01-17 15:00:40 +00:00
|
|
|
std::string dir = _srs_config->get_http_stream_dir();
|
2015-01-18 01:12:53 +00:00
|
|
|
if ((ret = mux.handle("/", new SrsVodStream(dir))) != ERROR_SUCCESS) {
|
2015-01-17 15:00:40 +00:00
|
|
|
srs_error("http: mount root dir=%s failed. ret=%d", dir.c_str(), ret);
|
2014-05-26 05:57:08 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-01-18 05:34:26 +00:00
|
|
|
srs_trace("http: root mount to %s", dir.c_str());
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-18 10:00:40 +00:00
|
|
|
int SrsHttpServer::mount_flv_streaming()
|
2015-01-18 09:17:07 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
2015-01-18 10:00:40 +00:00
|
|
|
|
|
|
|
// http flv live stream mount for each vhost.
|
|
|
|
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_flv_enabled(vhost)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-01-18 12:07:54 +00:00
|
|
|
SrsLiveEntry* entry = new SrsLiveEntry();
|
|
|
|
entry->vhost = vhost;
|
|
|
|
entry->mount = _srs_config->get_vhost_http_flv_mount(vhost);
|
|
|
|
flvs[vhost] = entry;
|
2015-01-18 10:00:40 +00:00
|
|
|
srs_trace("http flv live stream, vhost=%s, mount=%s",
|
2015-01-18 12:07:54 +00:00
|
|
|
vhost.c_str(), entry->mount.c_str());
|
2015-01-18 10:00:40 +00:00
|
|
|
}
|
|
|
|
|
2015-01-18 09:17:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
SrsHttpConn::SrsHttpConn(SrsServer* svr, st_netfd_t fd, SrsHttpServer* m)
|
|
|
|
: SrsConnection(svr, fd)
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
|
|
|
parser = new SrsHttpParser();
|
2015-01-17 15:00:40 +00:00
|
|
|
mux = m;
|
2014-04-04 15:06:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsHttpConn::~SrsHttpConn()
|
|
|
|
{
|
|
|
|
srs_freep(parser);
|
|
|
|
}
|
|
|
|
|
2014-06-19 07:28:05 +00:00
|
|
|
void SrsHttpConn::kbps_resample()
|
|
|
|
{
|
|
|
|
// TODO: FIXME: implements it
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsHttpConn::get_send_bytes_delta()
|
|
|
|
{
|
|
|
|
// TODO: FIXME: implements it
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t SrsHttpConn::get_recv_bytes_delta()
|
|
|
|
{
|
|
|
|
// TODO: FIXME: implements it
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-04 15:06:46 +00:00
|
|
|
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
|
2014-07-26 12:08:37 +00:00
|
|
|
SrsStSocket skt(stfd);
|
2014-04-04 15:06:46 +00:00
|
|
|
|
|
|
|
// 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.
|
2015-01-17 15:00:40 +00:00
|
|
|
SrsGoHttpResponseWriter writer(&skt);
|
|
|
|
if ((ret = process_request(&writer, req)) != ERROR_SUCCESS) {
|
2014-04-04 15:06:46 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
int SrsHttpConn::process_request(ISrsGoHttpResponseWriter* w, SrsHttpMessage* r)
|
2014-04-04 15:06:46 +00:00
|
|
|
{
|
|
|
|
int ret = ERROR_SUCCESS;
|
|
|
|
|
2014-05-30 01:20:51 +00:00
|
|
|
srs_trace("HTTP %s %s, content-length=%"PRId64"",
|
2015-01-17 15:00:40 +00:00
|
|
|
r->method_str().c_str(), r->url().c_str(), r->content_length());
|
2014-04-04 15:06:46 +00:00
|
|
|
|
2015-01-17 15:00:40 +00:00
|
|
|
// use default server mux to serve http request.
|
|
|
|
if ((ret = mux->mux.serve_http(w, r)) != ERROR_SUCCESS) {
|
|
|
|
if (!srs_is_client_gracefully_close(ret)) {
|
|
|
|
srs_error("serve http msg failed. ret=%d", ret);
|
|
|
|
}
|
2014-04-04 15:06:46 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2014-08-02 14:18:39 +00:00
|
|
|
|