diff --git a/README.md b/README.md
index 31044195d..bacdbeb57 100755
--- a/README.md
+++ b/README.md
@@ -566,6 +566,7 @@ Supported operating systems and hardware:
### SRS 2.0 history
+* v2.0, 2015-05-23, fix [#391](https://github.com/simple-rtmp-server/srs/issues/391) copy request for async call.
* v2.0, 2015-05-22, fix [#397](https://github.com/simple-rtmp-server/srs/issues/397) the USER_HZ maybe not 100. 2.0.165
* v2.0, 2015-05-22, for [#400](https://github.com/simple-rtmp-server/srs/issues/400), parse when got entire http header, by feilong. 2.0.164.
* v2.0, 2015-05-19, merge from bravo system, add the rtmfp to bms(commercial srs). 2.0.163.
diff --git a/trunk/configure b/trunk/configure
index 7f8727442..d0841c79a 100755
--- a/trunk/configure
+++ b/trunk/configure
@@ -159,7 +159,7 @@ MODULE_DEPENDS=("CORE" "KERNEL")
ModuleLibIncs=(${SRS_OBJS_DIR} ${LibSSLRoot})
MODULE_FILES=("srs_rtmp_amf0" "srs_rtmp_io" "srs_rtmp_stack" "srs_rtmp_sdk"
"srs_rtmp_handshake" "srs_rtmp_utility" "srs_rtmp_msg_array" "srs_rtmp_buffer"
- "srs_raw_avc" "srs_rtsp_stack")
+ "srs_raw_avc" "srs_rtsp_stack" "srs_http_stack")
RTMP_INCS="src/protocol"; MODULE_DIR=${RTMP_INCS} . auto/modules.sh
RTMP_OBJS="${MODULE_OBJS[@]}"
#
@@ -169,7 +169,7 @@ if [ $SRS_EXPORT_LIBRTMP_PROJECT = NO ]; then
MODULE_DEPENDS=("CORE" "KERNEL" "RTMP")
ModuleLibIncs=(${LibSTRoot} ${LibHttpParserRoot} ${SRS_OBJS_DIR})
MODULE_FILES=("srs_app_server" "srs_app_conn" "srs_app_rtmp_conn" "srs_app_st_socket" "srs_app_source"
- "srs_app_refer" "srs_app_hls" "srs_app_forward" "srs_app_encoder" "srs_app_http"
+ "srs_app_refer" "srs_app_hls" "srs_app_forward" "srs_app_encoder"
"srs_app_thread" "srs_app_bandwidth" "srs_app_st" "srs_app_log" "srs_app_config"
"srs_app_pithy_print" "srs_app_reload" "srs_app_http_api" "srs_app_http_conn" "srs_app_http_hooks"
"srs_app_json" "srs_app_ingest" "srs_app_ffmpeg" "srs_app_utility" "srs_app_dvr" "srs_app_edge"
diff --git a/trunk/ide/srs_upp/srs_upp.upp b/trunk/ide/srs_upp/srs_upp.upp
index e302bb98b..edba5ddc9 100755
--- a/trunk/ide/srs_upp/srs_upp.upp
+++ b/trunk/ide/srs_upp/srs_upp.upp
@@ -46,6 +46,8 @@ file
../../src/kernel/srs_kernel_utility.hpp,
../../src/kernel/srs_kernel_utility.cpp,
protocol readonly separator,
+ ../../src/protocol/srs_http_stack.hpp,
+ ../../src/protocol/srs_http_stack.cpp,
../../src/protocol/srs_raw_avc.hpp,
../../src/protocol/srs_raw_avc.cpp,
../../src/protocol/srs_rtmp_amf0.hpp,
@@ -91,8 +93,6 @@ file
../../src/app/srs_app_heartbeat.cpp,
../../src/app/srs_app_hls.hpp,
../../src/app/srs_app_hls.cpp,
- ../../src/app/srs_app_http.hpp,
- ../../src/app/srs_app_http.cpp,
../../src/app/srs_app_http_api.hpp,
../../src/app/srs_app_http_api.cpp,
../../src/app/srs_app_http_client.hpp,
diff --git a/trunk/ide/srs_vs2010/srs.vcxproj b/trunk/ide/srs_vs2010/srs.vcxproj
index f7d0a9ff1..76a86d491 100755
--- a/trunk/ide/srs_vs2010/srs.vcxproj
+++ b/trunk/ide/srs_vs2010/srs.vcxproj
@@ -74,7 +74,6 @@
-
@@ -119,6 +118,7 @@
+
@@ -155,7 +155,6 @@
-
@@ -201,6 +200,7 @@
+
diff --git a/trunk/ide/srs_xcode/srs_xcode.xcodeproj/project.pbxproj b/trunk/ide/srs_xcode/srs_xcode.xcodeproj/project.pbxproj
index 6daf2f9b7..ebc5dce14 100644
--- a/trunk/ide/srs_xcode/srs_xcode.xcodeproj/project.pbxproj
+++ b/trunk/ide/srs_xcode/srs_xcode.xcodeproj/project.pbxproj
@@ -7,6 +7,7 @@
objects = {
/* Begin PBXBuildFile section */
+ 3C0E1B8D1B0F5ADF003ADEF7 /* srs_http_stack.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C0E1B8B1B0F5ADF003ADEF7 /* srs_http_stack.cpp */; };
3C1231F61AAE652D00CE8F6C /* srs_core_autofree.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1231F01AAE652C00CE8F6C /* srs_core_autofree.cpp */; };
3C1231F71AAE652D00CE8F6C /* srs_core_performance.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1231F21AAE652C00CE8F6C /* srs_core_performance.cpp */; };
3C1231F81AAE652D00CE8F6C /* srs_core.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1231F41AAE652D00CE8F6C /* srs_core.cpp */; };
@@ -48,7 +49,6 @@
3C1232A01AAE81D900CE8F6C /* srs_app_http_client.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1232641AAE81D900CE8F6C /* srs_app_http_client.cpp */; };
3C1232A11AAE81D900CE8F6C /* srs_app_http_conn.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1232661AAE81D900CE8F6C /* srs_app_http_conn.cpp */; };
3C1232A21AAE81D900CE8F6C /* srs_app_http_hooks.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1232681AAE81D900CE8F6C /* srs_app_http_hooks.cpp */; };
- 3C1232A31AAE81D900CE8F6C /* srs_app_http.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C12326A1AAE81D900CE8F6C /* srs_app_http.cpp */; };
3C1232A41AAE81D900CE8F6C /* srs_app_ingest.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C12326C1AAE81D900CE8F6C /* srs_app_ingest.cpp */; };
3C1232A51AAE81D900CE8F6C /* srs_app_json.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C12326E1AAE81D900CE8F6C /* srs_app_json.cpp */; };
3C1232A61AAE81D900CE8F6C /* srs_app_kbps.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3C1232701AAE81D900CE8F6C /* srs_app_kbps.cpp */; };
@@ -122,6 +122,8 @@
/* End PBXCopyFilesBuildPhase section */
/* Begin PBXFileReference section */
+ 3C0E1B8B1B0F5ADF003ADEF7 /* srs_http_stack.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_http_stack.cpp; path = ../../../src/protocol/srs_http_stack.cpp; sourceTree = ""; };
+ 3C0E1B8C1B0F5ADF003ADEF7 /* srs_http_stack.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_http_stack.hpp; path = ../../../src/protocol/srs_http_stack.hpp; sourceTree = ""; };
3C1231E51AAE64A400CE8F6C /* srs_xcode */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = srs_xcode; sourceTree = BUILT_PRODUCTS_DIR; };
3C1231F01AAE652C00CE8F6C /* srs_core_autofree.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_core_autofree.cpp; path = ../../../src/core/srs_core_autofree.cpp; sourceTree = ""; };
3C1231F11AAE652C00CE8F6C /* srs_core_autofree.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_core_autofree.hpp; path = ../../../src/core/srs_core_autofree.hpp; sourceTree = ""; };
@@ -205,8 +207,6 @@
3C1232671AAE81D900CE8F6C /* srs_app_http_conn.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_app_http_conn.hpp; path = ../../../src/app/srs_app_http_conn.hpp; sourceTree = ""; };
3C1232681AAE81D900CE8F6C /* srs_app_http_hooks.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_app_http_hooks.cpp; path = ../../../src/app/srs_app_http_hooks.cpp; sourceTree = ""; };
3C1232691AAE81D900CE8F6C /* srs_app_http_hooks.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_app_http_hooks.hpp; path = ../../../src/app/srs_app_http_hooks.hpp; sourceTree = ""; };
- 3C12326A1AAE81D900CE8F6C /* srs_app_http.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_app_http.cpp; path = ../../../src/app/srs_app_http.cpp; sourceTree = ""; };
- 3C12326B1AAE81D900CE8F6C /* srs_app_http.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_app_http.hpp; path = ../../../src/app/srs_app_http.hpp; sourceTree = ""; };
3C12326C1AAE81D900CE8F6C /* srs_app_ingest.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_app_ingest.cpp; path = ../../../src/app/srs_app_ingest.cpp; sourceTree = ""; };
3C12326D1AAE81D900CE8F6C /* srs_app_ingest.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = srs_app_ingest.hpp; path = ../../../src/app/srs_app_ingest.hpp; sourceTree = ""; };
3C12326E1AAE81D900CE8F6C /* srs_app_json.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = srs_app_json.cpp; path = ../../../src/app/srs_app_json.cpp; sourceTree = ""; };
@@ -487,6 +487,8 @@
3C12322C1AAE819900CE8F6C /* protocol */ = {
isa = PBXGroup;
children = (
+ 3C0E1B8B1B0F5ADF003ADEF7 /* srs_http_stack.cpp */,
+ 3C0E1B8C1B0F5ADF003ADEF7 /* srs_http_stack.hpp */,
3C12322D1AAE81A400CE8F6C /* srs_raw_avc.cpp */,
3C12322E1AAE81A400CE8F6C /* srs_raw_avc.hpp */,
3C12322F1AAE81A400CE8F6C /* srs_rtmp_amf0.cpp */,
@@ -550,8 +552,6 @@
3C1232671AAE81D900CE8F6C /* srs_app_http_conn.hpp */,
3C1232681AAE81D900CE8F6C /* srs_app_http_hooks.cpp */,
3C1232691AAE81D900CE8F6C /* srs_app_http_hooks.hpp */,
- 3C12326A1AAE81D900CE8F6C /* srs_app_http.cpp */,
- 3C12326B1AAE81D900CE8F6C /* srs_app_http.hpp */,
3C12326C1AAE81D900CE8F6C /* srs_app_ingest.cpp */,
3C12326D1AAE81D900CE8F6C /* srs_app_ingest.hpp */,
3C12326E1AAE81D900CE8F6C /* srs_app_json.cpp */,
@@ -886,6 +886,7 @@
3C1232411AAE81A400CE8F6C /* srs_raw_avc.cpp in Sources */,
3C1232491AAE81A400CE8F6C /* srs_rtmp_utility.cpp in Sources */,
3C663F191AB0155100286D8B /* srs_publish.c in Sources */,
+ 3C0E1B8D1B0F5ADF003ADEF7 /* srs_http_stack.cpp in Sources */,
3C1232A01AAE81D900CE8F6C /* srs_app_http_client.cpp in Sources */,
3C689F981AB6AAAC00C9CEEE /* key.c in Sources */,
3C12329B1AAE81D900CE8F6C /* srs_app_ffmpeg.cpp in Sources */,
@@ -936,7 +937,6 @@
3C1232AF1AAE81D900CE8F6C /* srs_app_rtsp.cpp in Sources */,
3CC52DDD1ACE4023006FEB01 /* srs_utest_reload.cpp in Sources */,
3C689FA11AB6AAC800C9CEEE /* sync.c in Sources */,
- 3C1232A31AAE81D900CE8F6C /* srs_app_http.cpp in Sources */,
3C12329A1AAE81D900CE8F6C /* srs_app_encoder.cpp in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
diff --git a/trunk/src/app/srs_app_async_call.cpp b/trunk/src/app/srs_app_async_call.cpp
index 24f7228ca..28968de71 100644
--- a/trunk/src/app/srs_app_async_call.cpp
+++ b/trunk/src/app/srs_app_async_call.cpp
@@ -31,65 +31,64 @@ using namespace std;
// the sleep interval for http async callback.
#define SRS_AUTO_ASYNC_CALLBACL_SLEEP_US 300000
-ISrsDvrAsyncCall::ISrsDvrAsyncCall()
+ISrsAsyncCallTask::ISrsAsyncCallTask()
{
}
-ISrsDvrAsyncCall::~ISrsDvrAsyncCall()
+ISrsAsyncCallTask::~ISrsAsyncCallTask()
{
}
-SrsDvrAsyncCallThread::SrsDvrAsyncCallThread()
+SrsAsyncCallWorker::SrsAsyncCallWorker()
{
- pthread = new SrsThread("async", this, SRS_AUTO_ASYNC_CALLBACL_SLEEP_US, true);
+ pthread = new SrsReusableThread("async", this, SRS_AUTO_ASYNC_CALLBACL_SLEEP_US);
}
-SrsDvrAsyncCallThread::~SrsDvrAsyncCallThread()
+SrsAsyncCallWorker::~SrsAsyncCallWorker()
{
- stop();
srs_freep(pthread);
- std::vector::iterator it;
- for (it = callbacks.begin(); it != callbacks.end(); ++it) {
- ISrsDvrAsyncCall* call = *it;
- srs_freep(call);
+ std::vector::iterator it;
+ for (it = tasks.begin(); it != tasks.end(); ++it) {
+ ISrsAsyncCallTask* task = *it;
+ srs_freep(task);
}
- callbacks.clear();
+ tasks.clear();
}
-int SrsDvrAsyncCallThread::call(ISrsDvrAsyncCall* c)
+int SrsAsyncCallWorker::execute(ISrsAsyncCallTask* t)
{
int ret = ERROR_SUCCESS;
- callbacks.push_back(c);
+ tasks.push_back(t);
return ret;
}
-int SrsDvrAsyncCallThread::start()
+int SrsAsyncCallWorker::start()
{
return pthread->start();
}
-void SrsDvrAsyncCallThread::stop()
+void SrsAsyncCallWorker::stop()
{
pthread->stop();
}
-int SrsDvrAsyncCallThread::cycle()
+int SrsAsyncCallWorker::cycle()
{
int ret = ERROR_SUCCESS;
- std::vector copies = callbacks;
- callbacks.clear();
+ std::vector copies = tasks;
+ tasks.clear();
- std::vector::iterator it;
+ std::vector::iterator it;
for (it = copies.begin(); it != copies.end(); ++it) {
- ISrsDvrAsyncCall* call = *it;
- if ((ret = call->call()) != ERROR_SUCCESS) {
- srs_warn("ignore async callback %s, ret=%d", call->to_string().c_str(), ret);
+ ISrsAsyncCallTask* task = *it;
+ if ((ret = task->call()) != ERROR_SUCCESS) {
+ srs_warn("ignore async callback %s, ret=%d", task->to_string().c_str(), ret);
}
- srs_freep(call);
+ srs_freep(task);
}
return ret;
diff --git a/trunk/src/app/srs_app_async_call.hpp b/trunk/src/app/srs_app_async_call.hpp
index e014e3cc7..42f334ff8 100644
--- a/trunk/src/app/srs_app_async_call.hpp
+++ b/trunk/src/app/srs_app_async_call.hpp
@@ -42,32 +42,36 @@ CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
* a video and pass it to the dvr again.
* futhurmore, the aync call never block the main worker thread.
*/
-class ISrsDvrAsyncCall
+class ISrsAsyncCallTask
{
public:
- ISrsDvrAsyncCall();
- virtual ~ISrsDvrAsyncCall();
+ ISrsAsyncCallTask();
+ virtual ~ISrsAsyncCallTask();
public:
virtual int call() = 0;
virtual std::string to_string() = 0;
};
/**
-* the async callback for dvr.
-*/
-class SrsDvrAsyncCallThread : public ISrsThreadHandler
+ * the async callback for dvr.
+ * when worker call with the task, the worker will do it in isolate thread.
+ * that is, the task is execute/call in async mode.
+ */
+class SrsAsyncCallWorker : public ISrsReusableThreadHandler
{
private:
- SrsThread* pthread;
- std::vector callbacks;
+ SrsReusableThread* pthread;
+ std::vector tasks;
public:
- SrsDvrAsyncCallThread();
- virtual ~SrsDvrAsyncCallThread();
+ SrsAsyncCallWorker();
+ virtual ~SrsAsyncCallWorker();
public:
- virtual int call(ISrsDvrAsyncCall* c);
+ virtual int execute(ISrsAsyncCallTask* t);
public:
virtual int start();
virtual void stop();
+// interface ISrsReusableThreadHandler
+public:
virtual int cycle();
};
diff --git a/trunk/src/app/srs_app_conn.cpp b/trunk/src/app/srs_app_conn.cpp
index 4c483ba1c..a03a17489 100644
--- a/trunk/src/app/srs_app_conn.cpp
+++ b/trunk/src/app/srs_app_conn.cpp
@@ -45,12 +45,17 @@ SrsConnection::SrsConnection(IConnectionManager* cm, st_netfd_t c)
// so we never use joinable.
// TODO: FIXME: maybe other thread need to stop it.
// @see: https://github.com/simple-rtmp-server/srs/issues/78
- pthread = new SrsThread("conn", this, 0, false);
+ pthread = new SrsOneCycleThread("conn", this);
}
SrsConnection::~SrsConnection()
{
- stop();
+ /**
+ * when delete the connection, stop the connection,
+ * close the underlayer socket, delete the thread.
+ */
+ srs_close_stfd(stfd);
+ srs_freep(pthread);
}
int SrsConnection::start()
@@ -83,9 +88,6 @@ int SrsConnection::cycle()
if (ret == ERROR_SOCKET_CLOSED) {
srs_warn("client disconnect peer. ret=%d", ret);
}
-
- // set loop to stop to quit.
- pthread->stop_loop();
return ERROR_SUCCESS;
}
@@ -101,10 +103,4 @@ int SrsConnection::srs_id()
return id;
}
-void SrsConnection::stop()
-{
- srs_close_stfd(stfd);
- srs_freep(pthread);
-}
-
diff --git a/trunk/src/app/srs_app_conn.hpp b/trunk/src/app/srs_app_conn.hpp
index 29a6eace7..e00add8ce 100644
--- a/trunk/src/app/srs_app_conn.hpp
+++ b/trunk/src/app/srs_app_conn.hpp
@@ -58,14 +58,14 @@ public:
* all connections accept from listener must extends from this base class,
* server will add the connection to manager, and delete it when remove.
*/
-class SrsConnection : public virtual ISrsThreadHandler, public virtual IKbpsDelta
+class SrsConnection : public virtual ISrsOneCycleThreadHandler, public virtual IKbpsDelta
{
private:
/**
* each connection start a green thread,
* when thread stop, the connection will be delete by server.
*/
- SrsThread* pthread;
+ SrsOneCycleThread* pthread;
/**
* the id of connection.
*/
@@ -97,6 +97,8 @@ public:
* to remove the client by server->remove(this).
*/
virtual int start();
+// interface ISrsOneCycleThreadHandler
+public:
/**
* the thread cycle function,
* when serve connection completed, terminate the loop which will terminate the thread,
@@ -119,12 +121,6 @@ protected:
* for concrete connection to do the cycle.
*/
virtual int do_cycle() = 0;
-private:
- /**
- * when delete the connection, stop the connection,
- * close the underlayer socket, delete the thread.
- */
- virtual void stop();
};
#endif
diff --git a/trunk/src/app/srs_app_dvr.cpp b/trunk/src/app/srs_app_dvr.cpp
index 10725fe75..1ed737f44 100644
--- a/trunk/src/app/srs_app_dvr.cpp
+++ b/trunk/src/app/srs_app_dvr.cpp
@@ -498,12 +498,13 @@ int SrsFlvSegment::on_reload_vhost_dvr(std::string /*vhost*/)
SrsDvrAsyncCallOnDvr::SrsDvrAsyncCallOnDvr(SrsRequest* r, string p)
{
- req = r;
+ req = r->copy();
path = p;
}
SrsDvrAsyncCallOnDvr::~SrsDvrAsyncCallOnDvr()
{
+ srs_freep(req);
}
int SrsDvrAsyncCallOnDvr::call()
@@ -547,7 +548,7 @@ SrsDvrPlan::SrsDvrPlan()
dvr_enabled = false;
segment = new SrsFlvSegment(this);
- async = new SrsDvrAsyncCallThread();
+ async = new SrsAsyncCallWorker();
}
SrsDvrPlan::~SrsDvrPlan()
@@ -628,7 +629,7 @@ int SrsDvrPlan::on_reap_segment()
{
int ret = ERROR_SUCCESS;
- if ((ret = async->call(new SrsDvrAsyncCallOnDvr(req, segment->get_path()))) != ERROR_SUCCESS) {
+ if ((ret = async->execute(new SrsDvrAsyncCallOnDvr(req, segment->get_path()))) != ERROR_SUCCESS) {
return ret;
}
diff --git a/trunk/src/app/srs_app_dvr.hpp b/trunk/src/app/srs_app_dvr.hpp
index 4fb653a20..42e3521a1 100644
--- a/trunk/src/app/srs_app_dvr.hpp
+++ b/trunk/src/app/srs_app_dvr.hpp
@@ -178,7 +178,7 @@ public:
/**
* the dvr async call.
*/
-class SrsDvrAsyncCallOnDvr : public ISrsDvrAsyncCall
+class SrsDvrAsyncCallOnDvr : public ISrsAsyncCallTask
{
private:
std::string path;
@@ -206,7 +206,7 @@ public:
SrsRequest* req;
protected:
SrsFlvSegment* segment;
- SrsDvrAsyncCallThread* async;
+ SrsAsyncCallWorker* async;
bool dvr_enabled;
public:
SrsDvrPlan();
diff --git a/trunk/src/app/srs_app_edge.cpp b/trunk/src/app/srs_app_edge.cpp
index 3a8483be9..c95480c15 100644
--- a/trunk/src/app/srs_app_edge.cpp
+++ b/trunk/src/app/srs_app_edge.cpp
@@ -70,7 +70,7 @@ SrsEdgeIngester::SrsEdgeIngester()
origin_index = 0;
stream_id = 0;
stfd = NULL;
- pthread = new SrsThread("edge-igs", this, SRS_EDGE_INGESTER_SLEEP_US, true);
+ pthread = new SrsReusableThread2("edge-igs", this, SRS_EDGE_INGESTER_SLEEP_US);
}
SrsEdgeIngester::~SrsEdgeIngester()
@@ -171,7 +171,7 @@ int SrsEdgeIngester::ingest()
SrsPithyPrint* pprint = SrsPithyPrint::create_edge();
SrsAutoFree(SrsPithyPrint, pprint);
- while (pthread->can_loop()) {
+ while (!pthread->interrupted()) {
pprint->elapse();
// pithy print
@@ -397,7 +397,7 @@ SrsEdgeForwarder::SrsEdgeForwarder()
origin_index = 0;
stream_id = 0;
stfd = NULL;
- pthread = new SrsThread("edge-fwr", this, SRS_EDGE_FORWARDER_SLEEP_US, true);
+ pthread = new SrsReusableThread2("edge-fwr", this, SRS_EDGE_FORWARDER_SLEEP_US);
queue = new SrsMessageQueue();
send_error_code = ERROR_SUCCESS;
}
@@ -489,7 +489,7 @@ int SrsEdgeForwarder::cycle()
SrsMessageArray msgs(SYS_MAX_EDGE_SEND_MSGS);
- while (pthread->can_loop()) {
+ while (!pthread->interrupted()) {
if (send_error_code != ERROR_SUCCESS) {
st_usleep(SRS_EDGE_FORWARDER_ERROR_US);
continue;
diff --git a/trunk/src/app/srs_app_edge.hpp b/trunk/src/app/srs_app_edge.hpp
index 99147e62c..aa0cfc82d 100644
--- a/trunk/src/app/srs_app_edge.hpp
+++ b/trunk/src/app/srs_app_edge.hpp
@@ -75,7 +75,7 @@ enum SrsEdgeUserState
/**
* edge used to ingest stream from origin.
*/
-class SrsEdgeIngester : public ISrsThreadHandler
+class SrsEdgeIngester : public ISrsReusableThread2Handler
{
private:
int stream_id;
@@ -83,7 +83,7 @@ private:
SrsSource* _source;
SrsPlayEdge* _edge;
SrsRequest* _req;
- SrsThread* pthread;
+ SrsReusableThread2* pthread;
st_netfd_t stfd;
ISrsProtocolReaderWriter* io;
SrsKbps* kbps;
@@ -96,7 +96,7 @@ public:
virtual int initialize(SrsSource* source, SrsPlayEdge* edge, SrsRequest* req);
virtual int start();
virtual void stop();
-// interface ISrsThreadHandler
+// interface ISrsReusableThread2Handler
public:
virtual int cycle();
private:
@@ -110,7 +110,7 @@ private:
/**
* edge used to forward stream to origin.
*/
-class SrsEdgeForwarder : public ISrsThreadHandler
+class SrsEdgeForwarder : public ISrsReusableThread2Handler
{
private:
int stream_id;
@@ -118,7 +118,7 @@ private:
SrsSource* _source;
SrsPublishEdge* _edge;
SrsRequest* _req;
- SrsThread* pthread;
+ SrsReusableThread2* pthread;
st_netfd_t stfd;
ISrsProtocolReaderWriter* io;
SrsKbps* kbps;
@@ -144,7 +144,7 @@ public:
virtual int initialize(SrsSource* source, SrsPublishEdge* edge, SrsRequest* req);
virtual int start();
virtual void stop();
-// interface ISrsThreadHandler
+// interface ISrsReusableThread2Handler
public:
virtual int cycle();
public:
diff --git a/trunk/src/app/srs_app_encoder.cpp b/trunk/src/app/srs_app_encoder.cpp
index 3bcbc230c..a64c81c6d 100644
--- a/trunk/src/app/srs_app_encoder.cpp
+++ b/trunk/src/app/srs_app_encoder.cpp
@@ -44,7 +44,7 @@ static std::vector _transcoded_url;
SrsEncoder::SrsEncoder()
{
- pthread = new SrsThread("encoder", this, SRS_RTMP_ENCODER_SLEEP_US, true);
+ pthread = new SrsReusableThread("encoder", this, SRS_RTMP_ENCODER_SLEEP_US);
pprint = SrsPithyPrint::create_encoder();
}
diff --git a/trunk/src/app/srs_app_encoder.hpp b/trunk/src/app/srs_app_encoder.hpp
index f728e7579..557c29201 100644
--- a/trunk/src/app/srs_app_encoder.hpp
+++ b/trunk/src/app/srs_app_encoder.hpp
@@ -45,13 +45,13 @@ class SrsFFMPEG;
* the encoder for a stream,
* may use multiple ffmpegs to transcode the specified stream.
*/
-class SrsEncoder : public ISrsThreadHandler
+class SrsEncoder : public ISrsReusableThreadHandler
{
private:
std::string input_stream_name;
std::vector ffmpegs;
private:
- SrsThread* pthread;
+ SrsReusableThread* pthread;
SrsPithyPrint* pprint;
public:
SrsEncoder();
@@ -59,7 +59,7 @@ public:
public:
virtual int on_publish(SrsRequest* req);
virtual void on_unpublish();
-// interface ISrsThreadHandler.
+// interface ISrsReusableThreadHandler.
public:
virtual int cycle();
virtual void on_thread_stop();
diff --git a/trunk/src/app/srs_app_forward.cpp b/trunk/src/app/srs_app_forward.cpp
index 1d77d8854..4e68c0e9f 100644
--- a/trunk/src/app/srs_app_forward.cpp
+++ b/trunk/src/app/srs_app_forward.cpp
@@ -59,7 +59,7 @@ SrsForwarder::SrsForwarder(SrsSource* _source)
kbps = new SrsKbps();
stream_id = 0;
- pthread = new SrsThread("forward", this, SRS_FORWARDER_SLEEP_US, true);
+ pthread = new SrsReusableThread2("forward", this, SRS_FORWARDER_SLEEP_US);
queue = new SrsMessageQueue();
jitter = new SrsRtmpJitter();
@@ -407,7 +407,7 @@ int SrsForwarder::forward()
}
}
- while (pthread->can_loop()) {
+ while (!pthread->interrupted()) {
pprint->elapse();
// read from client.
diff --git a/trunk/src/app/srs_app_forward.hpp b/trunk/src/app/srs_app_forward.hpp
index dbad1c827..68b4314ab 100644
--- a/trunk/src/app/srs_app_forward.hpp
+++ b/trunk/src/app/srs_app_forward.hpp
@@ -48,7 +48,7 @@ class SrsKbps;
* forward the stream to other servers.
*/
// TODO: FIXME: refine the error log, comments it.
-class SrsForwarder : public ISrsThreadHandler
+class SrsForwarder : public ISrsReusableThread2Handler
{
private:
// the ep to forward, server[:port].
@@ -57,7 +57,7 @@ private:
int stream_id;
private:
st_netfd_t stfd;
- SrsThread* pthread;
+ SrsReusableThread2* pthread;
private:
SrsSource* source;
ISrsProtocolReaderWriter* io;
@@ -95,7 +95,7 @@ public:
* @param shared_video, directly ptr, copy it if need to save it.
*/
virtual int on_video(SrsSharedPtrMessage* shared_video);
-// interface ISrsThreadHandler.
+// interface ISrsReusableThread2Handler.
public:
virtual int cycle();
private:
diff --git a/trunk/src/app/srs_app_heartbeat.cpp b/trunk/src/app/srs_app_heartbeat.cpp
index ecb9b9196..165e38e90 100644
--- a/trunk/src/app/srs_app_heartbeat.cpp
+++ b/trunk/src/app/srs_app_heartbeat.cpp
@@ -33,9 +33,9 @@ using namespace std;
#include
#include
#include
-#include
#include
#include
+#include
SrsHttpHeartbeat::SrsHttpHeartbeat()
{
@@ -82,14 +82,14 @@ void SrsHttpHeartbeat::heartbeat()
return;
}
- SrsHttpMessage* msg = NULL;
+ ISrsHttpMessage* msg = NULL;
if ((ret = http.post(uri.get_path(), req, &msg)) != ERROR_SUCCESS) {
srs_info("http post hartbeart uri failed. "
"url=%s, request=%s, response=%s, ret=%d",
url.c_str(), req.c_str(), res.c_str(), ret);
return;
}
- SrsAutoFree(SrsHttpMessage, msg);
+ SrsAutoFree(ISrsHttpMessage, msg);
std::string res;
if ((ret = msg->body_read_all(res)) != ERROR_SUCCESS) {
diff --git a/trunk/src/app/srs_app_hls.cpp b/trunk/src/app/srs_app_hls.cpp
index cf322b22b..0ec4caedf 100644
--- a/trunk/src/app/srs_app_hls.cpp
+++ b/trunk/src/app/srs_app_hls.cpp
@@ -172,7 +172,7 @@ void SrsHlsSegment::update_duration(int64_t current_frame_dts)
SrsDvrAsyncCallOnHls::SrsDvrAsyncCallOnHls(SrsRequest* r, string p, string t, string m, string mu, int s, double d)
{
- req = r;
+ req = r->copy();
path = p;
ts_url = t;
m3u8 = m;
@@ -183,6 +183,7 @@ SrsDvrAsyncCallOnHls::SrsDvrAsyncCallOnHls(SrsRequest* r, string p, string t, st
SrsDvrAsyncCallOnHls::~SrsDvrAsyncCallOnHls()
{
+ srs_freep(req);
}
int SrsDvrAsyncCallOnHls::call()
@@ -221,12 +222,13 @@ string SrsDvrAsyncCallOnHls::to_string()
SrsDvrAsyncCallOnHlsNotify::SrsDvrAsyncCallOnHlsNotify(SrsRequest* r, string u)
{
- req = r;
+ req = r->copy();
ts_url = u;
}
SrsDvrAsyncCallOnHlsNotify::~SrsDvrAsyncCallOnHlsNotify()
{
+ srs_freep(req);
}
int SrsDvrAsyncCallOnHlsNotify::call()
@@ -284,7 +286,7 @@ SrsHlsMuxer::SrsHlsMuxer()
acodec = SrsCodecAudioReserved1;
should_write_cache = false;
should_write_file = true;
- async = new SrsDvrAsyncCallThread();
+ async = new SrsAsyncCallWorker();
context = new SrsTsContext();
}
@@ -667,7 +669,7 @@ int SrsHlsMuxer::segment_close(string log_desc)
segments.push_back(current);
// use async to call the http hooks, for it will cause thread switch.
- if ((ret = async->call(new SrsDvrAsyncCallOnHls(req,
+ if ((ret = async->execute(new SrsDvrAsyncCallOnHls(req,
current->full_path, current->uri, m3u8, m3u8_url,
current->sequence_no, current->duration))) != ERROR_SUCCESS)
{
@@ -675,7 +677,7 @@ int SrsHlsMuxer::segment_close(string log_desc)
}
// use async to call the http hooks, for it will cause thread switch.
- if ((ret = async->call(new SrsDvrAsyncCallOnHlsNotify(req, current->uri))) != ERROR_SUCCESS) {
+ if ((ret = async->execute(new SrsDvrAsyncCallOnHlsNotify(req, current->uri))) != ERROR_SUCCESS) {
return ret;
}
diff --git a/trunk/src/app/srs_app_hls.hpp b/trunk/src/app/srs_app_hls.hpp
index a79872a10..ccfd72a7e 100644
--- a/trunk/src/app/srs_app_hls.hpp
+++ b/trunk/src/app/srs_app_hls.hpp
@@ -159,7 +159,7 @@ public:
/**
* the hls async call: on_hls
*/
-class SrsDvrAsyncCallOnHls : public ISrsDvrAsyncCall
+class SrsDvrAsyncCallOnHls : public ISrsAsyncCallTask
{
private:
std::string path;
@@ -180,7 +180,7 @@ public:
/**
* the hls async call: on_hls_notify
*/
-class SrsDvrAsyncCallOnHlsNotify : public ISrsDvrAsyncCall
+class SrsDvrAsyncCallOnHlsNotify : public ISrsAsyncCallTask
{
private:
std::string ts_url;
@@ -215,7 +215,7 @@ private:
double hls_aof_ratio;
double hls_fragment;
double hls_window;
- SrsDvrAsyncCallThread* async;
+ SrsAsyncCallWorker* async;
private:
// whether use floor algorithm for timestamp.
bool hls_ts_floor;
diff --git a/trunk/src/app/srs_app_http.cpp b/trunk/src/app/srs_app_http.cpp
deleted file mode 100644
index ff7bf19b9..000000000
--- a/trunk/src/app/srs_app_http.cpp
+++ /dev/null
@@ -1,1700 +0,0 @@
-/*
-The MIT License (MIT)
-
-Copyright (c) 2013-2015 SRS(simple-rtmp-server)
-
-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
-
-#ifdef SRS_AUTO_HTTP_PARSER
-
-#include
-#include
-#include
-using namespace std;
-
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-
-#define SRS_DEFAULT_HTTP_PORT 80
-
-// for http parser macros
-#define SRS_CONSTS_HTTP_OPTIONS HTTP_OPTIONS
-#define SRS_CONSTS_HTTP_GET HTTP_GET
-#define SRS_CONSTS_HTTP_POST HTTP_POST
-#define SRS_CONSTS_HTTP_PUT HTTP_PUT
-#define SRS_CONSTS_HTTP_DELETE HTTP_DELETE
-
-// for ead all of http body, read each time.
-#define SRS_HTTP_READ_CACHE_BYTES 4096
-
-#define SRS_HTTP_DEFAULT_PAGE "index.html"
-
-int srs_go_http_response_json(ISrsHttpResponseWriter* w, string data)
-{
- w->header()->set_content_length(data.length());
- w->header()->set_content_type("application/json");
-
- return w->write((char*)data.data(), (int)data.length());
-}
-
-// get the status text of code.
-string srs_generate_http_status_text(int status)
-{
- static std::map _status_map;
- if (_status_map.empty()) {
- _status_map[SRS_CONSTS_HTTP_Continue ] = SRS_CONSTS_HTTP_Continue_str ;
- _status_map[SRS_CONSTS_HTTP_SwitchingProtocols ] = SRS_CONSTS_HTTP_SwitchingProtocols_str ;
- _status_map[SRS_CONSTS_HTTP_OK ] = SRS_CONSTS_HTTP_OK_str ;
- _status_map[SRS_CONSTS_HTTP_Created ] = SRS_CONSTS_HTTP_Created_str ;
- _status_map[SRS_CONSTS_HTTP_Accepted ] = SRS_CONSTS_HTTP_Accepted_str ;
- _status_map[SRS_CONSTS_HTTP_NonAuthoritativeInformation ] = SRS_CONSTS_HTTP_NonAuthoritativeInformation_str ;
- _status_map[SRS_CONSTS_HTTP_NoContent ] = SRS_CONSTS_HTTP_NoContent_str ;
- _status_map[SRS_CONSTS_HTTP_ResetContent ] = SRS_CONSTS_HTTP_ResetContent_str ;
- _status_map[SRS_CONSTS_HTTP_PartialContent ] = SRS_CONSTS_HTTP_PartialContent_str ;
- _status_map[SRS_CONSTS_HTTP_MultipleChoices ] = SRS_CONSTS_HTTP_MultipleChoices_str ;
- _status_map[SRS_CONSTS_HTTP_MovedPermanently ] = SRS_CONSTS_HTTP_MovedPermanently_str ;
- _status_map[SRS_CONSTS_HTTP_Found ] = SRS_CONSTS_HTTP_Found_str ;
- _status_map[SRS_CONSTS_HTTP_SeeOther ] = SRS_CONSTS_HTTP_SeeOther_str ;
- _status_map[SRS_CONSTS_HTTP_NotModified ] = SRS_CONSTS_HTTP_NotModified_str ;
- _status_map[SRS_CONSTS_HTTP_UseProxy ] = SRS_CONSTS_HTTP_UseProxy_str ;
- _status_map[SRS_CONSTS_HTTP_TemporaryRedirect ] = SRS_CONSTS_HTTP_TemporaryRedirect_str ;
- _status_map[SRS_CONSTS_HTTP_BadRequest ] = SRS_CONSTS_HTTP_BadRequest_str ;
- _status_map[SRS_CONSTS_HTTP_Unauthorized ] = SRS_CONSTS_HTTP_Unauthorized_str ;
- _status_map[SRS_CONSTS_HTTP_PaymentRequired ] = SRS_CONSTS_HTTP_PaymentRequired_str ;
- _status_map[SRS_CONSTS_HTTP_Forbidden ] = SRS_CONSTS_HTTP_Forbidden_str ;
- _status_map[SRS_CONSTS_HTTP_NotFound ] = SRS_CONSTS_HTTP_NotFound_str ;
- _status_map[SRS_CONSTS_HTTP_MethodNotAllowed ] = SRS_CONSTS_HTTP_MethodNotAllowed_str ;
- _status_map[SRS_CONSTS_HTTP_NotAcceptable ] = SRS_CONSTS_HTTP_NotAcceptable_str ;
- _status_map[SRS_CONSTS_HTTP_ProxyAuthenticationRequired ] = SRS_CONSTS_HTTP_ProxyAuthenticationRequired_str ;
- _status_map[SRS_CONSTS_HTTP_RequestTimeout ] = SRS_CONSTS_HTTP_RequestTimeout_str ;
- _status_map[SRS_CONSTS_HTTP_Conflict ] = SRS_CONSTS_HTTP_Conflict_str ;
- _status_map[SRS_CONSTS_HTTP_Gone ] = SRS_CONSTS_HTTP_Gone_str ;
- _status_map[SRS_CONSTS_HTTP_LengthRequired ] = SRS_CONSTS_HTTP_LengthRequired_str ;
- _status_map[SRS_CONSTS_HTTP_PreconditionFailed ] = SRS_CONSTS_HTTP_PreconditionFailed_str ;
- _status_map[SRS_CONSTS_HTTP_RequestEntityTooLarge ] = SRS_CONSTS_HTTP_RequestEntityTooLarge_str ;
- _status_map[SRS_CONSTS_HTTP_RequestURITooLarge ] = SRS_CONSTS_HTTP_RequestURITooLarge_str ;
- _status_map[SRS_CONSTS_HTTP_UnsupportedMediaType ] = SRS_CONSTS_HTTP_UnsupportedMediaType_str ;
- _status_map[SRS_CONSTS_HTTP_RequestedRangeNotSatisfiable ] = SRS_CONSTS_HTTP_RequestedRangeNotSatisfiable_str ;
- _status_map[SRS_CONSTS_HTTP_ExpectationFailed ] = SRS_CONSTS_HTTP_ExpectationFailed_str ;
- _status_map[SRS_CONSTS_HTTP_InternalServerError ] = SRS_CONSTS_HTTP_InternalServerError_str ;
- _status_map[SRS_CONSTS_HTTP_NotImplemented ] = SRS_CONSTS_HTTP_NotImplemented_str ;
- _status_map[SRS_CONSTS_HTTP_BadGateway ] = SRS_CONSTS_HTTP_BadGateway_str ;
- _status_map[SRS_CONSTS_HTTP_ServiceUnavailable ] = SRS_CONSTS_HTTP_ServiceUnavailable_str ;
- _status_map[SRS_CONSTS_HTTP_GatewayTimeout ] = SRS_CONSTS_HTTP_GatewayTimeout_str ;
- _status_map[SRS_CONSTS_HTTP_HTTPVersionNotSupported ] = SRS_CONSTS_HTTP_HTTPVersionNotSupported_str ;
- }
-
- std::string status_text;
- if (_status_map.find(status) == _status_map.end()) {
- status_text = "Status Unknown";
- } else {
- status_text = _status_map[status];
- }
-
- return status_text;
-}
-
-// bodyAllowedForStatus reports whether a given response status code
-// permits a body. See RFC2616, section 4.4.
-bool srs_go_http_body_allowd(int status)
-{
- if (status >= 100 && status <= 199) {
- return false;
- } else if (status == 204 || status == 304) {
- return false;
- }
-
- return true;
-}
-
-// DetectContentType implements the algorithm described
-// at http://mimesniff.spec.whatwg.org/ to determine the
-// Content-Type of the given data. It considers at most the
-// first 512 bytes of data. DetectContentType always returns
-// a valid MIME type: if it cannot determine a more specific one, it
-// returns "application/octet-stream".
-string srs_go_http_detect(char* data, int size)
-{
- // detect only when data specified.
- if (data) {
- }
- return "application/octet-stream"; // fallback
-}
-
-// Error replies to the request with the specified error message and HTTP code.
-// The error message should be plain text.
-int srs_go_http_error(ISrsHttpResponseWriter* w, int code, string error)
-{
- int ret = ERROR_SUCCESS;
-
- w->header()->set_content_type("text/plain; charset=utf-8");
- w->header()->set_content_length(error.length());
- w->write_header(code);
- w->write((char*)error.data(), (int)error.length());
-
- return ret;
-}
-
-SrsHttpHeader::SrsHttpHeader()
-{
-}
-
-SrsHttpHeader::~SrsHttpHeader()
-{
-}
-
-void SrsHttpHeader::set(string key, string value)
-{
- headers[key] = value;
-}
-
-string SrsHttpHeader::get(string key)
-{
- std::string v;
-
- if (headers.find(key) != headers.end()) {
- v = headers[key];
- }
-
- return v;
-}
-
-int64_t SrsHttpHeader::content_length()
-{
- std::string cl = get("Content-Length");
-
- if (cl.empty()) {
- return -1;
- }
-
- return (int64_t)::atof(cl.c_str());
-}
-
-void SrsHttpHeader::set_content_length(int64_t size)
-{
- char buf[64];
- snprintf(buf, sizeof(buf), "%"PRId64, size);
- set("Content-Length", buf);
-}
-
-string SrsHttpHeader::content_type()
-{
- return get("Content-Type");
-}
-
-void SrsHttpHeader::set_content_type(string ct)
-{
- set("Content-Type", ct);
-}
-
-void SrsHttpHeader::write(stringstream& ss)
-{
- std::map::iterator it;
- for (it = headers.begin(); it != headers.end(); ++it) {
- ss << it->first << ": " << it->second << SRS_HTTP_CRLF;
- }
-}
-
-ISrsHttpResponseWriter::ISrsHttpResponseWriter()
-{
-}
-
-ISrsHttpResponseWriter::~ISrsHttpResponseWriter()
-{
-}
-
-ISrsHttpResponseReader::ISrsHttpResponseReader()
-{
-}
-
-ISrsHttpResponseReader::~ISrsHttpResponseReader()
-{
-}
-
-ISrsHttpHandler::ISrsHttpHandler()
-{
- entry = NULL;
-}
-
-ISrsHttpHandler::~ISrsHttpHandler()
-{
-}
-
-SrsHttpRedirectHandler::SrsHttpRedirectHandler(string u, int c)
-{
- url = u;
- code = c;
-}
-
-SrsHttpRedirectHandler::~SrsHttpRedirectHandler()
-{
-}
-
-int SrsHttpRedirectHandler::serve_http(ISrsHttpResponseWriter* w, SrsHttpMessage* r)
-{
- int ret = ERROR_SUCCESS;
- // TODO: FIXME: implements it.
- return ret;
-}
-
-SrsHttpNotFoundHandler::SrsHttpNotFoundHandler()
-{
-}
-
-SrsHttpNotFoundHandler::~SrsHttpNotFoundHandler()
-{
-}
-
-int SrsHttpNotFoundHandler::serve_http(ISrsHttpResponseWriter* w, SrsHttpMessage* r)
-{
- return srs_go_http_error(w,
- SRS_CONSTS_HTTP_NotFound, SRS_CONSTS_HTTP_NotFound_str);
-}
-
-SrsHttpFileServer::SrsHttpFileServer(string root_dir)
-{
- dir = root_dir;
-}
-
-SrsHttpFileServer::~SrsHttpFileServer()
-{
-}
-
-int SrsHttpFileServer::serve_http(ISrsHttpResponseWriter* w, SrsHttpMessage* r)
-{
- string upath = r->path();
-
- // add default pages.
- if (srs_string_ends_with(upath, "/")) {
- upath += SRS_HTTP_DEFAULT_PAGE;
- }
-
- string fullpath = dir + "/";
-
- // remove the virtual directory.
- srs_assert(entry);
- size_t pos = entry->pattern.find("/");
- if (upath.length() > entry->pattern.length() && pos != string::npos) {
- fullpath += upath.substr(entry->pattern.length() - pos);
- } else {
- fullpath += upath;
- }
-
- // stat current dir, if exists, return error.
- if (!srs_path_exists(fullpath)) {
- srs_warn("http miss file=%s, pattern=%s, upath=%s",
- fullpath.c_str(), entry->pattern.c_str(), upath.c_str());
- return SrsHttpNotFoundHandler().serve_http(w, r);
- }
- srs_trace("http match file=%s, pattern=%s, upath=%s",
- fullpath.c_str(), entry->pattern.c_str(), upath.c_str());
-
- // handle file according to its extension.
- // use vod stream for .flv/.fhv
- if (srs_string_ends_with(fullpath, ".flv") || srs_string_ends_with(fullpath, ".fhv")) {
- return serve_flv_file(w, r, fullpath);
- } else if (srs_string_ends_with(fullpath, ".mp4")) {
- return serve_mp4_file(w, r, fullpath);
- }
-
- // serve common static file.
- return serve_file(w, r, fullpath);
-}
-
-int SrsHttpFileServer::serve_file(ISrsHttpResponseWriter* w, SrsHttpMessage* r, string fullpath)
-{
- int ret = ERROR_SUCCESS;
-
- // open the target file.
- SrsFileReader fs;
-
- if ((ret = fs.open(fullpath)) != ERROR_SUCCESS) {
- srs_warn("open file %s failed, ret=%d", fullpath.c_str(), ret);
- return ret;
- }
-
- int64_t length = fs.filesize();
-
- // unset the content length to encode in chunked encoding.
- w->header()->set_content_length(length);
-
- static std::map _mime;
- if (_mime.empty()) {
- _mime[".ts"] = "video/MP2T";
- _mime[".flv"] = "video/x-flv";
- _mime[".m4v"] = "video/x-m4v";
- _mime[".3gpp"] = "video/3gpp";
- _mime[".3gp"] = "video/3gpp";
- _mime[".mp4"] = "video/mp4";
- _mime[".aac"] = "audio/x-aac";
- _mime[".mp3"] = "audio/mpeg";
- _mime[".m4a"] = "audio/x-m4a";
- _mime[".ogg"] = "audio/ogg";
- // @see hls-m3u8-draft-pantos-http-live-streaming-12.pdf, page 5.
- _mime[".m3u8"] = "application/vnd.apple.mpegurl"; // application/x-mpegURL
- _mime[".rss"] = "application/rss+xml";
- _mime[".json"] = "application/json";
- _mime[".swf"] = "application/x-shockwave-flash";
- _mime[".doc"] = "application/msword";
- _mime[".zip"] = "application/zip";
- _mime[".rar"] = "application/x-rar-compressed";
- _mime[".xml"] = "text/xml";
- _mime[".html"] = "text/html";
- _mime[".js"] = "text/javascript";
- _mime[".css"] = "text/css";
- _mime[".ico"] = "image/x-icon";
- _mime[".png"] = "image/png";
- _mime[".jpeg"] = "image/jpeg";
- _mime[".jpg"] = "image/jpeg";
- _mime[".gif"] = "image/gif";
- }
-
- if (true) {
- size_t pos;
- std::string ext = fullpath;
- if ((pos = ext.rfind(".")) != string::npos) {
- ext = ext.substr(pos);
- }
-
- if (_mime.find(ext) == _mime.end()) {
- w->header()->set_content_type("application/octet-stream");
- } else {
- w->header()->set_content_type(_mime[ext]);
- }
- }
-
- // write body.
- int64_t left = length;
- if ((ret = copy(w, &fs, r, (int)left)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("read file=%s size=%d failed, ret=%d", fullpath.c_str(), left, ret);
- }
- return ret;
- }
-
- return w->final_request();
-}
-
-int SrsHttpFileServer::serve_flv_file(ISrsHttpResponseWriter* w, SrsHttpMessage* r, string fullpath)
-{
- std::string start = r->query_get("start");
- if (start.empty()) {
- return serve_file(w, r, fullpath);
- }
-
- int offset = ::atoi(start.c_str());
- if (offset <= 0) {
- return serve_file(w, r, fullpath);
- }
-
- return serve_flv_stream(w, r, fullpath, offset);
-}
-
-int SrsHttpFileServer::serve_mp4_file(ISrsHttpResponseWriter* w, SrsHttpMessage* r, string fullpath)
-{
- // for flash to request mp4 range in query string.
- // for example, http://digitalprimates.net/dash/DashTest.html?url=http://dashdemo.edgesuite.net/digitalprimates/nexus/oops-20120802-manifest.mpd
- std::string range = r->query_get("range");
- // or, use bytes to request range,
- // for example, http://dashas.castlabs.com/demo/try.html
- if (range.empty()) {
- range = r->query_get("bytes");
- }
-
- // rollback to serve whole file.
- size_t pos = string::npos;
- if (range.empty() || (pos = range.find("-")) == string::npos) {
- return serve_file(w, r, fullpath);
- }
-
- // parse the start in query string
- int start = 0;
- if (pos > 0) {
- start = ::atoi(range.substr(0, pos).c_str());
- }
-
- // parse end in query string.
- int end = -1;
- if (pos < range.length() - 1) {
- end = ::atoi(range.substr(pos + 1).c_str());
- }
-
- // invalid param, serve as whole mp4 file.
- if (start < 0 || (end != -1 && start > end)) {
- return serve_file(w, r, fullpath);
- }
-
- return serve_mp4_stream(w, r, fullpath, start, end);
-}
-
-int SrsHttpFileServer::serve_flv_stream(ISrsHttpResponseWriter* w, SrsHttpMessage* r, string fullpath, int offset)
-{
- return serve_file(w, r, fullpath);
-}
-
-int SrsHttpFileServer::serve_mp4_stream(ISrsHttpResponseWriter* w, SrsHttpMessage* r, string fullpath, int start, int end)
-{
- return serve_file(w, r, fullpath);
-}
-
-int SrsHttpFileServer::copy(ISrsHttpResponseWriter* w, SrsFileReader* fs, SrsHttpMessage* r, int size)
-{
- int ret = ERROR_SUCCESS;
-
- int left = size;
- char* buf = r->http_ts_send_buffer();
-
- while (left > 0) {
- ssize_t nread = -1;
- int max_read = srs_min(left, SRS_HTTP_TS_SEND_BUFFER_SIZE);
- if ((ret = fs->read(buf, max_read, &nread)) != ERROR_SUCCESS) {
- break;
- }
-
- left -= nread;
- if ((ret = w->write(buf, (int)nread)) != ERROR_SUCCESS) {
- break;
- }
- }
-
- return ret;
-}
-
-SrsHttpMuxEntry::SrsHttpMuxEntry()
-{
- enabled = true;
- explicit_match = false;
- handler = NULL;
-}
-
-SrsHttpMuxEntry::~SrsHttpMuxEntry()
-{
- srs_freep(handler);
-}
-
-ISrsHttpMatchHijacker::ISrsHttpMatchHijacker()
-{
-}
-
-ISrsHttpMatchHijacker::~ISrsHttpMatchHijacker()
-{
-}
-
-SrsHttpServeMux::SrsHttpServeMux()
-{
-}
-
-SrsHttpServeMux::~SrsHttpServeMux()
-{
- std::map::iterator it;
- for (it = entries.begin(); it != entries.end(); ++it) {
- SrsHttpMuxEntry* entry = it->second;
- srs_freep(entry);
- }
- entries.clear();
-
- vhosts.clear();
- hijackers.clear();
-}
-
-int SrsHttpServeMux::initialize()
-{
- int ret = ERROR_SUCCESS;
- // TODO: FIXME: implements it.
- return ret;
-}
-
-void SrsHttpServeMux::hijack(ISrsHttpMatchHijacker* h)
-{
- std::vector::iterator it = ::find(hijackers.begin(), hijackers.end(), h);
- if (it != hijackers.end()) {
- return;
- }
- hijackers.push_back(h);
-}
-
-void SrsHttpServeMux::unhijack(ISrsHttpMatchHijacker* h)
-{
- std::vector::iterator it = ::find(hijackers.begin(), hijackers.end(), h);
- if (it == hijackers.end()) {
- return;
- }
- hijackers.erase(it);
-}
-
-int SrsHttpServeMux::handle(std::string pattern, ISrsHttpHandler* handler)
-{
- int ret = ERROR_SUCCESS;
-
- srs_assert(handler);
-
- if (pattern.empty()) {
- ret = ERROR_HTTP_PATTERN_EMPTY;
- srs_error("http: empty pattern. ret=%d", ret);
- return ret;
- }
-
- if (entries.find(pattern) != entries.end()) {
- SrsHttpMuxEntry* exists = entries[pattern];
- if (exists->explicit_match) {
- ret = ERROR_HTTP_PATTERN_DUPLICATED;
- srs_error("http: multiple registrations for %s. ret=%d", pattern.c_str(), ret);
- return ret;
- }
- }
-
- std::string vhost = pattern;
- if (pattern.at(0) != '/') {
- if (pattern.find("/") != string::npos) {
- vhost = pattern.substr(0, pattern.find("/"));
- }
- vhosts[vhost] = handler;
- }
-
- if (true) {
- SrsHttpMuxEntry* entry = new SrsHttpMuxEntry();
- entry->explicit_match = true;
- entry->handler = handler;
- entry->pattern = pattern;
- entry->handler->entry = entry;
-
- if (entries.find(pattern) != entries.end()) {
- SrsHttpMuxEntry* exists = entries[pattern];
- srs_freep(exists);
- }
- entries[pattern] = entry;
- }
-
- // Helpful behavior:
- // If pattern is /tree/, insert an implicit permanent redirect for /tree.
- // It can be overridden by an explicit registration.
- if (pattern != "/" && !pattern.empty() && pattern.at(pattern.length() - 1) == '/') {
- std::string rpattern = pattern.substr(0, pattern.length() - 1);
- SrsHttpMuxEntry* entry = NULL;
-
- // free the exists not explicit entry
- if (entries.find(rpattern) != entries.end()) {
- SrsHttpMuxEntry* exists = entries[rpattern];
- if (!exists->explicit_match) {
- entry = exists;
- }
- }
-
- // create implicit redirect.
- if (!entry || entry->explicit_match) {
- srs_freep(entry);
-
- entry = new SrsHttpMuxEntry();
- entry->explicit_match = false;
- entry->handler = new SrsHttpRedirectHandler(pattern, SRS_CONSTS_HTTP_MovedPermanently);
- entry->pattern = pattern;
- entry->handler->entry = entry;
-
- entries[rpattern] = entry;
- }
- }
-
- return ret;
-}
-
-int SrsHttpServeMux::serve_http(ISrsHttpResponseWriter* w, SrsHttpMessage* r)
-{
- int ret = ERROR_SUCCESS;
-
- ISrsHttpHandler* h = NULL;
- if ((ret = find_handler(r, &h)) != ERROR_SUCCESS) {
- srs_error("find handler failed. ret=%d", ret);
- return ret;
- }
-
- srs_assert(h);
- if ((ret = h->serve_http(w, r)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("handler serve http failed. ret=%d", ret);
- }
- return ret;
- }
-
- return ret;
-}
-
-int SrsHttpServeMux::find_handler(SrsHttpMessage* r, ISrsHttpHandler** ph)
-{
- int ret = ERROR_SUCCESS;
-
- // TODO: FIXME: support the path . and ..
- if (r->url().find("..") != std::string::npos) {
- ret = ERROR_HTTP_URL_NOT_CLEAN;
- srs_error("htt url not canonical, url=%s. ret=%d", r->url().c_str(), ret);
- return ret;
- }
-
- if ((ret = match(r, ph)) != ERROR_SUCCESS) {
- srs_error("http match handler failed. ret=%d", ret);
- return ret;
- }
-
- // always hijack.
- if (!hijackers.empty()) {
- // notice all hijacker the match failed.
- std::vector::iterator it;
- for (it = hijackers.begin(); it != hijackers.end(); ++it) {
- ISrsHttpMatchHijacker* hijacker = *it;
- if ((ret = hijacker->hijack(r, ph)) != ERROR_SUCCESS) {
- srs_error("hijacker match failed. ret=%d", ret);
- return ret;
- }
- }
- }
-
- if (*ph == NULL) {
- // TODO: FIXME: memory leak.
- *ph = new SrsHttpNotFoundHandler();
- }
-
- return ret;
-}
-
-int SrsHttpServeMux::match(SrsHttpMessage* r, ISrsHttpHandler** ph)
-{
- int ret = ERROR_SUCCESS;
-
- std::string path = r->path();
-
- // Host-specific pattern takes precedence over generic ones
- if (!vhosts.empty() && vhosts.find(r->host()) != vhosts.end()) {
- path = r->host() + path;
- }
-
- int nb_matched = 0;
- ISrsHttpHandler* h = NULL;
-
- std::map::iterator it;
- for (it = entries.begin(); it != entries.end(); ++it) {
- std::string pattern = it->first;
- SrsHttpMuxEntry* entry = it->second;
-
- if (!entry->enabled) {
- continue;
- }
-
- if (!path_match(pattern, path)) {
- continue;
- }
-
- if (!h || (int)pattern.length() > nb_matched) {
- nb_matched = (int)pattern.length();
- h = entry->handler;
- }
- }
-
- *ph = h;
-
- return ret;
-}
-
-bool SrsHttpServeMux::path_match(string pattern, string path)
-{
- if (pattern.empty()) {
- return false;
- }
-
- int n = (int)pattern.length();
-
- // not endswith '/', exactly match.
- if (pattern.at(n - 1) != '/') {
- return pattern == path;
- }
-
- // endswith '/', match any,
- // for example, '/api/' match '/api/[N]'
- if ((int)path.length() >= n) {
- if (memcmp(pattern.data(), path.data(), n) == 0) {
- return true;
- }
- }
-
- return false;
-}
-
-SrsHttpResponseWriter::SrsHttpResponseWriter(SrsStSocket* io)
-{
- skt = io;
- hdr = new SrsHttpHeader();
- header_wrote = false;
- status = SRS_CONSTS_HTTP_OK;
- content_length = -1;
- written = 0;
- header_sent = false;
-}
-
-SrsHttpResponseWriter::~SrsHttpResponseWriter()
-{
- srs_freep(hdr);
-}
-
-int SrsHttpResponseWriter::final_request()
-{
- // complete the chunked encoding.
- if (content_length == -1) {
- std::stringstream ss;
- ss << 0 << SRS_HTTP_CRLF << SRS_HTTP_CRLF;
- std::string ch = ss.str();
- return skt->write((void*)ch.data(), (int)ch.length(), NULL);
- }
-
- // flush when send with content length
- return write(NULL, 0);
-}
-
-SrsHttpHeader* SrsHttpResponseWriter::header()
-{
- return hdr;
-}
-
-int SrsHttpResponseWriter::write(char* data, int size)
-{
- int ret = ERROR_SUCCESS;
-
- if (!header_wrote) {
- write_header(SRS_CONSTS_HTTP_OK);
- }
-
- written += size;
- if (content_length != -1 && written > content_length) {
- ret = ERROR_HTTP_CONTENT_LENGTH;
- srs_error("http: exceed content length. ret=%d", ret);
- return ret;
- }
-
- if ((ret = send_header(data, size)) != ERROR_SUCCESS) {
- srs_error("http: send header failed. ret=%d", ret);
- return ret;
- }
-
- // ignore NULL content.
- if (!data) {
- return ret;
- }
-
- // directly send with content length
- if (content_length != -1) {
- return skt->write((void*)data, size, NULL);
- }
-
- // send in chunked encoding.
- std::stringstream ss;
- ss << hex << size << SRS_HTTP_CRLF;
- std::string ch = ss.str();
- if ((ret = skt->write((void*)ch.data(), (int)ch.length(), NULL)) != ERROR_SUCCESS) {
- return ret;
- }
- if ((ret = skt->write((void*)data, size, NULL)) != ERROR_SUCCESS) {
- return ret;
- }
- if ((ret = skt->write((void*)SRS_HTTP_CRLF, 2, NULL)) != ERROR_SUCCESS) {
- return ret;
- }
-
- return ret;
-}
-
-void SrsHttpResponseWriter::write_header(int code)
-{
- if (header_wrote) {
- srs_warn("http: multiple write_header calls, code=%d", code);
- return;
- }
-
- header_wrote = true;
- status = code;
-
- // parse the content length from header.
- content_length = hdr->content_length();
-}
-
-int SrsHttpResponseWriter::send_header(char* data, int size)
-{
- int ret = ERROR_SUCCESS;
-
- if (header_sent) {
- return ret;
- }
- header_sent = true;
-
- std::stringstream ss;
-
- // status_line
- ss << "HTTP/1.1 " << status << " "
- << srs_generate_http_status_text(status) << SRS_HTTP_CRLF;
-
- // detect content type
- if (srs_go_http_body_allowd(status)) {
- if (hdr->content_type().empty()) {
- hdr->set_content_type(srs_go_http_detect(data, size));
- }
- }
-
- // set server if not set.
- if (hdr->get("Server").empty()) {
- hdr->set("Server", RTMP_SIG_SRS_KEY"/"RTMP_SIG_SRS_VERSION);
- }
-
- // chunked encoding
- if (content_length == -1) {
- hdr->set("Transfer-Encoding", "chunked");
- }
-
- // keep alive to make vlc happy.
- hdr->set("Connection", "Keep-Alive");
-
- // write headers
- hdr->write(ss);
-
- // header_eof
- ss << SRS_HTTP_CRLF;
-
- std::string buf = ss.str();
- return skt->write((void*)buf.c_str(), buf.length(), NULL);
-}
-
-SrsHttpResponseReader::SrsHttpResponseReader(SrsHttpMessage* msg, SrsStSocket* io)
-{
- skt = io;
- owner = msg;
- is_eof = false;
- nb_total_read = 0;
- nb_left_chunk = 0;
- buffer = NULL;
-}
-
-SrsHttpResponseReader::~SrsHttpResponseReader()
-{
-}
-
-int SrsHttpResponseReader::initialize(SrsFastBuffer* body)
-{
- int ret = ERROR_SUCCESS;
-
- nb_chunk = 0;
- nb_left_chunk = 0;
- nb_total_read = 0;
- buffer = body;
-
- return ret;
-}
-
-bool SrsHttpResponseReader::eof()
-{
- return is_eof;
-}
-
-int SrsHttpResponseReader::read(char* data, int nb_data, int* nb_read)
-{
- int ret = ERROR_SUCCESS;
-
- if (is_eof) {
- ret = ERROR_HTTP_RESPONSE_EOF;
- srs_error("http: response EOF. ret=%d", ret);
- return ret;
- }
-
- // chunked encoding.
- if (owner->is_chunked()) {
- return read_chunked(data, nb_data, nb_read);
- }
-
- // read by specified content-length
- int max = (int)owner->content_length() - (int)nb_total_read;
- if (max <= 0) {
- is_eof = true;
- return ret;
- }
-
- // change the max to read.
- nb_data = srs_min(nb_data, max);
- return read_specified(data, nb_data, nb_read);
-}
-
-int SrsHttpResponseReader::read_chunked(char* data, int nb_data, int* nb_read)
-{
- int ret = ERROR_SUCCESS;
-
- // when no bytes left in chunk,
- // parse the chunk length first.
- if (nb_left_chunk <= 0) {
- char* at = NULL;
- int length = 0;
- while (!at) {
- // find the CRLF of chunk header end.
- char* start = buffer->bytes();
- char* end = start + buffer->size();
- for (char* p = start; p < end - 1; p++) {
- if (p[0] == SRS_HTTP_CR && p[1] == SRS_HTTP_LF) {
- // invalid chunk, ignore.
- if (p == start) {
- ret = ERROR_HTTP_INVALID_CHUNK_HEADER;
- srs_error("chunk header start with CRLF. ret=%d", ret);
- return ret;
- }
- length = (int)(p - start + 2);
- at = buffer->read_slice(length);
- break;
- }
- }
-
- // got at, ok.
- if (at) {
- break;
- }
-
- // when empty, only grow 1bytes, but the buffer will cache more.
- if ((ret = buffer->grow(skt, buffer->size() + 1)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("read body from server failed. ret=%d", ret);
- }
- return ret;
- }
- }
- srs_assert(length >= 3);
-
- // it's ok to set the pos and pos+1 to NULL.
- at[length - 1] = 0;
- at[length - 2] = 0;
-
- // size is the bytes size, excludes the chunk header and end CRLF.
- int ilength = (int)::strtol(at, NULL, 16);
- if (ilength < 0) {
- ret = ERROR_HTTP_INVALID_CHUNK_HEADER;
- srs_error("chunk header negative, length=%d. ret=%d", ilength, ret);
- return ret;
- }
-
- // all bytes in chunk is left now.
- nb_chunk = nb_left_chunk = ilength;
- }
-
- if (nb_chunk <= 0) {
- // for the last chunk, eof.
- is_eof = true;
- } else {
- // for not the last chunk, there must always exists bytes.
- // left bytes in chunk, read some.
- srs_assert(nb_left_chunk);
-
- int nb_bytes = srs_min(nb_left_chunk, nb_data);
- ret = read_specified(data, nb_bytes, &nb_bytes);
-
- // the nb_bytes used for output already read size of bytes.
- if (nb_read) {
- *nb_read = nb_bytes;
- }
- nb_left_chunk -= nb_bytes;
- srs_info("http: read %d bytes of chunk", nb_bytes);
-
- // error or still left bytes in chunk, ignore and read in future.
- if (nb_left_chunk > 0 || (ret != ERROR_SUCCESS)) {
- return ret;
- }
- srs_info("http: read total chunk %dB", nb_chunk);
- }
-
- // for both the last or not, the CRLF of chunk payload end.
- if ((ret = buffer->grow(skt, 2)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("read EOF of chunk from server failed. ret=%d", ret);
- }
- return ret;
- }
- buffer->read_slice(2);
-
- return ret;
-}
-
-int SrsHttpResponseReader::read_specified(char* data, int nb_data, int* nb_read)
-{
- int ret = ERROR_SUCCESS;
-
- if (buffer->size() <= 0) {
- // when empty, only grow 1bytes, but the buffer will cache more.
- if ((ret = buffer->grow(skt, 1)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("read body from server failed. ret=%d", ret);
- }
- return ret;
- }
- }
-
- int nb_bytes = srs_min(nb_data, buffer->size());
-
- // read data to buffer.
- srs_assert(nb_bytes);
- char* p = buffer->read_slice(nb_bytes);
- memcpy(data, p, nb_bytes);
- if (nb_read) {
- *nb_read = nb_bytes;
- }
-
- // increase the total read to determine whether EOF.
- nb_total_read += nb_bytes;
-
- // for not chunked
- if (!owner->is_chunked()) {
- // when read completed, eof.
- if (nb_total_read >= (int)owner->content_length()) {
- is_eof = true;
- }
- }
-
- return ret;
-}
-
-SrsHttpMessage::SrsHttpMessage(SrsStSocket* io, SrsConnection* c)
-{
- conn = c;
- chunked = false;
- keep_alive = true;
- _uri = new SrsHttpUri();
- _body = new SrsHttpResponseReader(this, io);
- _http_ts_send_buffer = new char[SRS_HTTP_TS_SEND_BUFFER_SIZE];
-}
-
-SrsHttpMessage::~SrsHttpMessage()
-{
- srs_freep(_body);
- srs_freep(_uri);
- srs_freep(_http_ts_send_buffer);
-}
-
-int SrsHttpMessage::update(string url, http_parser* header, SrsFastBuffer* body, vector& headers)
-{
- int ret = ERROR_SUCCESS;
-
- _url = url;
- _header = *header;
- _headers = headers;
-
- // whether chunked.
- std::string transfer_encoding = get_request_header("Transfer-Encoding");
- chunked = (transfer_encoding == "chunked");
-
- // whether keep alive.
- keep_alive = http_should_keep_alive(header);
-
- // set the buffer.
- if ((ret = _body->initialize(body)) != ERROR_SUCCESS) {
- return ret;
- }
-
- // parse uri from url.
- std::string host = get_request_header("Host");
-
- // donot parse the empty host for uri,
- // for example, the response contains no host,
- // ignore it is ok.
- if (host.empty()) {
- return ret;
- }
-
- // parse uri to schema/server:port/path?query
- std::string uri = "http://" + host + _url;
- if ((ret = _uri->initialize(uri)) != ERROR_SUCCESS) {
- return ret;
- }
-
- // must format as key=value&...&keyN=valueN
- std::string q = _uri->get_query();
- size_t pos = string::npos;
- while (!q.empty()) {
- std::string k = q;
- if ((pos = q.find("=")) != string::npos) {
- k = q.substr(0, pos);
- q = q.substr(pos + 1);
- } else {
- q = "";
- }
-
- std::string v = q;
- if ((pos = q.find("&")) != string::npos) {
- v = q.substr(0, pos);
- q = q.substr(pos + 1);
- } else {
- q = "";
- }
-
- _query[k] = v;
- }
-
- // parse ext.
- _ext = _uri->get_path();
- if ((pos = _ext.rfind(".")) != string::npos) {
- _ext = _ext.substr(pos);
- } else {
- _ext = "";
- }
-
- return ret;
-}
-
-char* SrsHttpMessage::http_ts_send_buffer()
-{
- return _http_ts_send_buffer;
-}
-
-SrsConnection* SrsHttpMessage::connection()
-{
- return conn;
-}
-
-u_int8_t SrsHttpMessage::method()
-{
- return (u_int8_t)_header.method;
-}
-
-u_int16_t SrsHttpMessage::status_code()
-{
- return (u_int16_t)_header.status_code;
-}
-
-string SrsHttpMessage::method_str()
-{
- if (is_http_get()) {
- return "GET";
- }
- if (is_http_put()) {
- return "PUT";
- }
- if (is_http_post()) {
- return "POST";
- }
- if (is_http_delete()) {
- return "DELETE";
- }
- if (is_http_options()) {
- return "OPTIONS";
- }
-
- return "OTHER";
-}
-
-bool SrsHttpMessage::is_http_get()
-{
- return _header.method == SRS_CONSTS_HTTP_GET;
-}
-
-bool SrsHttpMessage::is_http_put()
-{
- return _header.method == SRS_CONSTS_HTTP_PUT;
-}
-
-bool SrsHttpMessage::is_http_post()
-{
- return _header.method == SRS_CONSTS_HTTP_POST;
-}
-
-bool SrsHttpMessage::is_http_delete()
-{
- return _header.method == SRS_CONSTS_HTTP_DELETE;
-}
-
-bool SrsHttpMessage::is_http_options()
-{
- return _header.method == SRS_CONSTS_HTTP_OPTIONS;
-}
-
-bool SrsHttpMessage::is_chunked()
-{
- return chunked;
-}
-
-bool SrsHttpMessage::is_keep_alive()
-{
- return keep_alive;
-}
-
-string SrsHttpMessage::uri()
-{
- std::string uri = _uri->get_schema();
- if (uri.empty()) {
- uri += "http";
- }
- uri += "://";
-
- uri += host();
- uri += path();
- return uri;
-}
-
-string SrsHttpMessage::url()
-{
- return _uri->get_url();
-}
-
-string SrsHttpMessage::host()
-{
- return _uri->get_host();
-}
-
-string SrsHttpMessage::path()
-{
- return _uri->get_path();
-}
-
-string SrsHttpMessage::ext()
-{
- return _ext;
-}
-
-int SrsHttpMessage::body_read_all(string& body)
-{
- int ret = ERROR_SUCCESS;
-
- // cache to read.
- char* buf = new char[SRS_HTTP_READ_CACHE_BYTES];
- SrsAutoFree(char, buf);
-
- // whatever, read util EOF.
- while (!_body->eof()) {
- int nb_read = 0;
- if ((ret = _body->read(buf, SRS_HTTP_READ_CACHE_BYTES, &nb_read)) != ERROR_SUCCESS) {
- return ret;
- }
-
- if (nb_read > 0) {
- body.append(buf, nb_read);
- }
- }
-
- return ret;
-}
-
-ISrsHttpResponseReader* SrsHttpMessage::body_reader()
-{
- return _body;
-}
-
-int64_t SrsHttpMessage::content_length()
-{
- return _header.content_length;
-}
-
-string SrsHttpMessage::query_get(string key)
-{
- std::string v;
-
- if (_query.find(key) != _query.end()) {
- v = _query[key];
- }
-
- return v;
-}
-
-int SrsHttpMessage::request_header_count()
-{
- return (int)_headers.size();
-}
-
-string SrsHttpMessage::request_header_key_at(int index)
-{
- srs_assert(index < request_header_count());
- SrsHttpHeaderField item = _headers[index];
- return item.first;
-}
-
-string SrsHttpMessage::request_header_value_at(int index)
-{
- srs_assert(index < request_header_count());
- SrsHttpHeaderField item = _headers[index];
- return item.second;
-}
-
-string SrsHttpMessage::get_request_header(string name)
-{
- std::vector::iterator it;
-
- for (it = _headers.begin(); it != _headers.end(); ++it) {
- SrsHttpHeaderField& elem = *it;
- std::string key = elem.first;
- std::string value = elem.second;
- if (key == name) {
- return value;
- }
- }
-
- return "";
-}
-
-SrsRequest* SrsHttpMessage::to_request(string vhost)
-{
- SrsRequest* req = new SrsRequest();
-
- req->app = _uri->get_path();
- size_t pos = string::npos;
- if ((pos = req->app.rfind("/")) != string::npos) {
- req->stream = req->app.substr(pos + 1);
- req->app = req->app.substr(0, pos);
- }
- if ((pos = req->stream.rfind(".")) != string::npos) {
- req->stream = req->stream.substr(0, pos);
- }
-
- req->tcUrl = "rtmp://" + vhost + req->app;
- req->pageUrl = get_request_header("Referer");
- req->objectEncoding = 0;
-
- srs_discovery_tc_url(req->tcUrl,
- req->schema, req->host, req->vhost, req->app, req->port,
- req->param);
- req->strip();
-
- return req;
-}
-
-SrsHttpParser::SrsHttpParser()
-{
- buffer = new SrsFastBuffer();
-}
-
-SrsHttpParser::~SrsHttpParser()
-{
- srs_freep(buffer);
-}
-
-int SrsHttpParser::initialize(enum http_parser_type type)
-{
- int ret = ERROR_SUCCESS;
-
- 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_init(&parser, type);
- // callback object ptr.
- parser.data = (void*)this;
-
- return ret;
-}
-
-int SrsHttpParser::parse_message(SrsStSocket* skt, SrsConnection* conn, SrsHttpMessage** ppmsg)
-{
- *ppmsg = NULL;
-
- int ret = ERROR_SUCCESS;
-
- // reset request data.
- field_name = "";
- field_value = "";
- expect_field_name = true;
- state = SrsHttpParseStateInit;
- header = http_parser();
- url = "";
- headers.clear();
- header_parsed = 0;
-
- // do parse
- if ((ret = parse_message_imp(skt)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("parse http msg failed. ret=%d", ret);
- }
- return ret;
- }
-
- // create msg
- SrsHttpMessage* msg = new SrsHttpMessage(skt, conn);
-
- // initalize http msg, parse url.
- if ((ret = msg->update(url, &header, buffer, headers)) != ERROR_SUCCESS) {
- srs_error("initialize http msg failed. ret=%d", ret);
- srs_freep(msg);
- return ret;
- }
-
- // parse ok, return the msg.
- *ppmsg = msg;
-
- return ret;
-}
-
-int SrsHttpParser::parse_message_imp(SrsStSocket* skt)
-{
- int ret = ERROR_SUCCESS;
-
- while (true) {
- ssize_t nparsed = 0;
-
- // when got entire http header, parse it.
- // @see https://github.com/simple-rtmp-server/srs/issues/400
- char* start = buffer->bytes();
- char* end = start + buffer->size();
- for (char* p = start; p <= end - 4; p++) {
- // SRS_HTTP_CRLFCRLF "\r\n\r\n" // 0x0D0A0D0A
- if (p[0] == SRS_CONSTS_CR && p[1] == SRS_CONSTS_LF && p[2] == SRS_CONSTS_CR && p[3] == SRS_CONSTS_LF) {
- nparsed = http_parser_execute(&parser, &settings, buffer->bytes(), buffer->size());
- srs_info("buffer=%d, nparsed=%d, header=%d", buffer->size(), (int)nparsed, header_parsed);
- break;
- }
- }
-
- // consume the parsed bytes.
- if (nparsed && header_parsed) {
- buffer->read_slice(header_parsed);
- }
-
- // ok atleast header completed,
- // never wait for body completed, for maybe chunked.
- if (state == SrsHttpParseStateHeaderComplete || state == SrsHttpParseStateMessageComplete) {
- break;
- }
-
- // when nothing parsed, read more to parse.
- if (nparsed == 0) {
- // when requires more, only grow 1bytes, but the buffer will cache more.
- if ((ret = buffer->grow(skt, buffer->size() + 1)) != ERROR_SUCCESS) {
- if (!srs_is_client_gracefully_close(ret)) {
- srs_error("read body from server failed. ret=%d", ret);
- }
- return ret;
- }
- }
- }
-
- // parse last header.
- if (!field_name.empty() && !field_value.empty()) {
- headers.push_back(std::make_pair(field_name, field_value));
- }
-
- return ret;
-}
-
-int SrsHttpParser::on_message_begin(http_parser* parser)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- obj->state = SrsHttpParseStateStart;
-
- srs_info("***MESSAGE BEGIN***");
-
- return 0;
-}
-
-int SrsHttpParser::on_headers_complete(http_parser* parser)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- obj->header = *parser;
- // save the parser when header parse completed.
- obj->state = SrsHttpParseStateHeaderComplete;
- obj->header_parsed = (int)parser->nread;
-
- srs_info("***HEADERS COMPLETE***");
-
- // see http_parser.c:1570, return 1 to skip body.
- return 0;
-}
-
-int SrsHttpParser::on_message_complete(http_parser* parser)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- // save the parser when body parse completed.
- obj->state = SrsHttpParseStateMessageComplete;
-
- srs_info("***MESSAGE COMPLETE***\n");
-
- return 0;
-}
-
-int SrsHttpParser::on_url(http_parser* parser, const char* at, size_t length)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- if (length > 0) {
- obj->url.append(at, (int)length);
- }
-
- srs_info("Method: %d, Url: %.*s", parser->method, (int)length, at);
-
- return 0;
-}
-
-int SrsHttpParser::on_header_field(http_parser* parser, const char* at, size_t length)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- // field value=>name, reap the field.
- if (!obj->expect_field_name) {
- obj->headers.push_back(std::make_pair(obj->field_name, obj->field_value));
-
- // reset the field name when parsed.
- obj->field_name = "";
- obj->field_value = "";
- }
- obj->expect_field_name = true;
-
- if (length > 0) {
- obj->field_name.append(at, (int)length);
- }
-
- srs_info("Header field(%d bytes): %.*s", (int)length, (int)length, at);
- return 0;
-}
-
-int SrsHttpParser::on_header_value(http_parser* parser, const char* at, size_t length)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- if (length > 0) {
- obj->field_value.append(at, (int)length);
- }
- obj->expect_field_name = false;
-
- srs_info("Header value(%d bytes): %.*s", (int)length, (int)length, at);
- return 0;
-}
-
-int SrsHttpParser::on_body(http_parser* parser, const char* at, size_t length)
-{
- SrsHttpParser* obj = (SrsHttpParser*)parser->data;
- srs_assert(obj);
-
- srs_info("Body: %.*s", (int)length, at);
-
- return 0;
-}
-
-SrsHttpUri::SrsHttpUri()
-{
- port = SRS_DEFAULT_HTTP_PORT;
-}
-
-SrsHttpUri::~SrsHttpUri()
-{
-}
-
-int SrsHttpUri::initialize(string _url)
-{
- int ret = ERROR_SUCCESS;
-
- url = _url;
- const char* purl = url.c_str();
-
- http_parser_url hp_u;
- if((ret = http_parser_parse_url(purl, url.length(), 0, &hp_u)) != 0){
- int code = ret;
- ret = ERROR_HTTP_PARSE_URI;
-
- srs_error("parse url %s failed, code=%d, ret=%d", purl, code, ret);
- return ret;
- }
-
- std::string field = get_uri_field(url, &hp_u, UF_SCHEMA);
- if(!field.empty()){
- schema = field;
- }
-
- host = get_uri_field(url, &hp_u, UF_HOST);
-
- field = get_uri_field(url, &hp_u, UF_PORT);
- if(!field.empty()){
- port = atoi(field.c_str());
- }
-
- path = get_uri_field(url, &hp_u, UF_PATH);
- srs_info("parse url %s success", purl);
-
- query = get_uri_field(url, &hp_u, UF_QUERY);
- srs_info("parse query %s success", query.c_str());
-
- return ret;
-}
-
-const char* SrsHttpUri::get_url()
-{
- return url.data();
-}
-
-const char* SrsHttpUri::get_schema()
-{
- return schema.data();
-}
-
-const char* SrsHttpUri::get_host()
-{
- return host.data();
-}
-
-int SrsHttpUri::get_port()
-{
- return port;
-}
-
-const char* SrsHttpUri::get_path()
-{
- return path.data();
-}
-
-const char* SrsHttpUri::get_query()
-{
- return query.data();
-}
-
-string SrsHttpUri::get_uri_field(string uri, http_parser_url* hp_u, http_parser_url_fields field)
-{
- if((hp_u->field_set & (1 << field)) == 0){
- return "";
- }
-
- srs_verbose("uri field matched, off=%d, len=%d, value=%.*s",
- hp_u->field_data[field].off,
- hp_u->field_data[field].len,
- hp_u->field_data[field].len,
- uri.c_str() + hp_u->field_data[field].off);
-
- int offset = hp_u->field_data[field].off;
- int len = hp_u->field_data[field].len;
-
- return uri.substr(offset, len);
-}
-
-#endif
-
diff --git a/trunk/src/app/srs_app_http.hpp b/trunk/src/app/srs_app_http.hpp
deleted file mode 100644
index 07b9fee08..000000000
--- a/trunk/src/app/srs_app_http.hpp
+++ /dev/null
@@ -1,686 +0,0 @@
-/*
-The MIT License (MIT)
-
-Copyright (c) 2013-2015 SRS(simple-rtmp-server)
-
-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.
-*/
-
-#ifndef SRS_APP_HTTP_HPP
-#define SRS_APP_HTTP_HPP
-
-/*
-#include
-*/
-#include
-
-#ifdef SRS_AUTO_HTTP_PARSER
-
-#include