Skip to content

Commit 8fde036

Browse files
industriousonesoftwinlinvipchundonglinlin
authored
Kickoff publisher when stream is idle, which means no players. v6.0.31, v5.0.144 (#3105)
For some use scenario, the publisher is invited when player want to view the stream: 1. Publisher connect to system, but does not publish any stream to SRS yet. 2. Player connect to system and start to request the stream. 3. System notifies publisher to publish stream to SRS. 4. Player play the stream from SRS. Please notice that `system` means your business system, not SRS. This is what we called `on-demand-live-streaming`, so when the last player stop to view the stream, what happends? 1. System needs to notify publisher to stop publish. 2. Or, SRS disconnect the publisher when idle(the last player stops playing). This PR is for the solution 2, so that the cleanup is very simple, your system does not need to notify publisher to stop publish, because SRS has already disconnected the publihser. --------- Co-authored-by: winlin <[email protected]> Co-authored-by: chundonglinlin <[email protected]>
1 parent dc7be76 commit 8fde036

14 files changed

+125
-19
lines changed

trunk/conf/full.conf

+6
Original file line numberDiff line numberDiff line change
@@ -1341,6 +1341,12 @@ vhost publish.srs.com {
13411341
# Overwrite by env SRS_VHOST_PUBLISH_TRY_ANNEXB_FIRST for all vhosts.
13421342
# default: on
13431343
try_annexb_first on;
1344+
# The timeout in seconds to disconnect publisher when idle, which means no players.
1345+
# Note that 0 means no timeout or this feature is disabled.
1346+
# Note that this feature conflicts with forward, because it disconnect the publisher stream.
1347+
# Overwrite by env SRS_VHOST_PUBLISH_KICKOFF_FOR_IDLE for all vhosts.
1348+
# default: 0
1349+
kickoff_for_idle 0;
13441350
}
13451351
}
13461352

trunk/conf/rtmp.kickoff.conf

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
# the config for srs to delivery RTMP with kicking off publish as no one watching.
2+
# @see https://github.com/ossrs/srs/wiki/v1_CN_SampleRTMP
3+
# @see full.conf for detail config.
4+
5+
listen 1935;
6+
max_connections 1000;
7+
daemon off;
8+
srs_log_tank console;
9+
vhost __defaultVhost__ {
10+
publish {
11+
kickoff_for_idle 60000;
12+
}
13+
}

trunk/doc/CHANGELOG.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ The changelog for SRS.
88

99
## SRS 6.0 Changelog
1010

11+
* v6.0, 2023-03-04, Merge [#3105](https://github.com/ossrs/srs/pull/3105): Kickoff publisher when stream is idle, which means no players. v6.0.31 (#3105)
1112
* v6.0, 2023-02-25, Merge [#3438](https://github.com/ossrs/srs/pull/3438): Forward add question mark to the end. v6.0.30 (#3438)
1213
* v6.0, 2023-02-25, Merge [#3416](https://github.com/ossrs/srs/pull/3416): GB: Support HEVC for regression test and load tool for GB. v6.0.29 (#3416)
1314
* v6.0, 2023-02-25, Merge [#3424](https://github.com/ossrs/srs/pull/3424): API: Add service_id for http_hooks, which identify the process. v6.0.28 (#3424)
@@ -44,7 +45,8 @@ The changelog for SRS.
4445

4546
## SRS 5.0 Changelog
4647

47-
* v5.0, 2023-02-25, Merge [#3424](https://github.com/ossrs/srs/pull/3424): API: Add service_id for http_hooks, which identify the process. v5.0.142 (#3424)
48+
* v5.0, 2023-03-04, Merge [#3105](https://github.com/ossrs/srs/pull/3105): Kickoff publisher when stream is idle, which means no players. v5.0.144 (#3105)
49+
* v5.0, 2023-02-25, Merge [#3424](https://github.com/ossrs/srs/pull/3424): API: Add service_id for http_hooks, which identify the process. v5.0.143 (#3424)
4850
* v5.0, 2023-02-22, Compatible with legacy RTMP URL. v5.0.142
4951
* v5.0, 2023-02-12, Merge [#3409](https://github.com/ossrs/srs/pull/3409): SRT: Reduce latency to 200ms of srt2rtc.conf. v5.0.141 (#3409)
5052
* v5.0, 2023-02-08, Merge [#3391](https://github.com/ossrs/srs/pull/3391): Config: Error when both HLS and HTTP-TS enabled. v5.0.140 (#3391)

trunk/doc/Features.md

+1
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@ The features of SRS.
4747
- [x] Live: Support origin cluster, please read [#464](https://github.com/ossrs/srs/issues/464), [RTMP 302](https://github.com/ossrs/srs/issues/92). v3.0.0+
4848
- [x] Live: Support NGINX HLS Cluster, see [CN](https://ossrs.net/lts/zh-cn/docs/v4/doc/sample-hls-cluster) or [EN](https://ossrs.io/lts/en-us/docs/v4/doc/sample-hls-cluster). v5.0.28+
4949
- [x] Live: SRT: Support PUSH SRT by IP and optional port, see [#3198](https://github.com/ossrs/srs/issues/3198). v5.0.76+
50+
- [x] Live: Kickoff publisher when stream is idle, which means no players. v5.0.144+
5051
- [x] Live: [Experimental] Support SRT server, read [#1147](https://github.com/ossrs/srs/issues/1147). v4.0.143+
5152
- [x] Live: [Experimental] Support Coroutine Native SRT over ST, [#3010](https://github.com/ossrs/srs/pull/3010). v5.0.30+
5253
- [x] Live: [Experimental] Support MPEG-DASH, Dynamic Adaptive Streaming over HTTP, read [#299](https://github.com/ossrs/srs/issues/299). v5.0.96+

trunk/src/app/srs_app_config.cpp

+38-1
Original file line numberDiff line numberDiff line change
@@ -2559,7 +2559,7 @@ srs_error_t SrsConfig::check_normal_config()
25592559
for (int j = 0; j < (int)conf->directives.size(); j++) {
25602560
string m = conf->at(j)->name;
25612561
if (m != "mr" && m != "mr_latency" && m != "firstpkt_timeout" && m != "normal_timeout"
2562-
&& m != "parse_sps" && m != "try_annexb_first") {
2562+
&& m != "parse_sps" && m != "try_annexb_first" && m != "kickoff_for_idle") {
25632563
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "illegal vhost.publish.%s of %s", m.c_str(), vhost->arg0().c_str());
25642564
}
25652565
}
@@ -2727,6 +2727,14 @@ srs_error_t SrsConfig::check_normal_config()
27272727
}
27282728
}
27292729

2730+
// Check forward dnd kickoff for publsher idle.
2731+
for (int n = 0; n < (int)vhosts.size(); n++) {
2732+
SrsConfDirective* vhost = vhosts[n];
2733+
if (get_forward_enabled(vhost) && get_publish_kickoff_for_idle(vhost)) {
2734+
return srs_error_new(ERROR_SYSTEM_CONFIG_INVALID, "vhost.forward conflicts with vhost.publish.kickoff_for_idle");
2735+
}
2736+
}
2737+
27302738
// check ingest id unique.
27312739
for (int i = 0; i < (int)vhosts.size(); i++) {
27322740
SrsConfDirective* vhost = vhosts[i];
@@ -5340,6 +5348,35 @@ srs_utime_t SrsConfig::get_publish_normal_timeout(string vhost)
53405348
return (srs_utime_t)(::atoi(conf->arg0().c_str()) * SRS_UTIME_MILLISECONDS);
53415349
}
53425350

5351+
srs_utime_t SrsConfig::get_publish_kickoff_for_idle(std::string vhost)
5352+
{
5353+
return get_publish_kickoff_for_idle(get_vhost(vhost));
5354+
}
5355+
5356+
srs_utime_t SrsConfig::get_publish_kickoff_for_idle(SrsConfDirective* vhost)
5357+
{
5358+
SRS_OVERWRITE_BY_ENV_FLOAT_SECONDS("srs.vhost.publish.kickoff_for_idle"); // SRS_VHOST_PUBLISH_KICKOFF_FOR_IDLE
5359+
5360+
static srs_utime_t DEFAULT = 0 * SRS_UTIME_SECONDS;
5361+
5362+
SrsConfDirective* conf = vhost;
5363+
if (!conf) {
5364+
return DEFAULT;
5365+
}
5366+
5367+
conf = conf->get("publish");
5368+
if (!conf) {
5369+
return DEFAULT;
5370+
}
5371+
5372+
conf = conf->get("kickoff_for_idle");
5373+
if (!conf || conf->arg0().empty()) {
5374+
return DEFAULT;
5375+
}
5376+
5377+
return (srs_utime_t)(::atof(conf->arg0().c_str()) * SRS_UTIME_SECONDS);
5378+
}
5379+
53435380
int SrsConfig::get_global_chunk_size()
53445381
{
53455382
SRS_OVERWRITE_BY_ENV_INT("srs.vhost.chunk_size"); // SRS_VHOST_CHUNK_SIZE

trunk/src/app/srs_app_config.hpp

+3
Original file line numberDiff line numberDiff line change
@@ -636,6 +636,9 @@ class SrsConfig
636636
virtual srs_utime_t get_publish_1stpkt_timeout(std::string vhost);
637637
// The normal packet timeout in srs_utime_t for encoder.
638638
virtual srs_utime_t get_publish_normal_timeout(std::string vhost);
639+
// The kickoff timeout in srs_utime_t for publisher.
640+
virtual srs_utime_t get_publish_kickoff_for_idle(std::string vhost);
641+
virtual srs_utime_t get_publish_kickoff_for_idle(SrsConfDirective* vhost);
639642
private:
640643
// Get the global chunk size.
641644
virtual int get_global_chunk_size();

trunk/src/app/srs_app_gb28181.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -255,7 +255,7 @@ srs_error_t SrsLazyGbSession::cycle()
255255

256256
// It maybe success with message.
257257
if (srs_error_code(err) == ERROR_SUCCESS) {
258-
srs_trace("client finished%s.", srs_error_summary(err).c_str());
258+
srs_trace("client finished %s.", srs_error_summary(err).c_str());
259259
srs_freep(err);
260260
return err;
261261
}

trunk/src/app/srs_app_rtmp_conn.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -985,6 +985,7 @@ srs_error_t SrsRtmpConn::do_publishing(SrsLiveSource* source, SrsPublishRecvThre
985985
// initialize the publish timeout.
986986
publish_1stpkt_timeout = _srs_config->get_publish_1stpkt_timeout(req->vhost);
987987
publish_normal_timeout = _srs_config->get_publish_normal_timeout(req->vhost);
988+
srs_utime_t publish_kickoff_for_idle = _srs_config->get_publish_kickoff_for_idle(req->vhost);
988989

989990
// set the sock options.
990991
set_sock_options();
@@ -1008,6 +1009,11 @@ srs_error_t SrsRtmpConn::do_publishing(SrsLiveSource* source, SrsPublishRecvThre
10081009
return srs_error_wrap(err, "rtmp: thread quit");
10091010
}
10101011

1012+
// Kick off the publisher when idle for a period of timeout.
1013+
if (source->publisher_is_idle_for(publish_kickoff_for_idle)) {
1014+
return srs_error_new(ERROR_KICKOFF_FOR_IDLE, "kicked for idle, url=%s, timeout=%ds", req->tcUrl.c_str(), srsu2si(publish_kickoff_for_idle));
1015+
}
1016+
10111017
pprint->elapse();
10121018

10131019
// cond wait for timeout.

trunk/src/app/srs_app_source.cpp

+40-9
Original file line numberDiff line numberDiff line change
@@ -1877,7 +1877,7 @@ srs_error_t SrsLiveSourceManager::notify(int event, srs_utime_t interval, srs_ut
18771877
// @see https://github.com/ossrs/srs/issues/714
18781878
#if 0
18791879
// When source expired, remove it.
1880-
if (source->expired()) {
1880+
if (source->stream_is_dead()) {
18811881
int cid = source->source_id();
18821882
if (cid == -1 && source->pre_source_id() > 0) {
18831883
cid = source->pre_source_id();
@@ -1926,7 +1926,8 @@ SrsLiveSource::SrsLiveSource()
19261926
mix_queue = new SrsMixQueue();
19271927

19281928
_can_publish = true;
1929-
die_at = 0;
1929+
stream_die_at_ = 0;
1930+
publisher_idle_at_ = 0;
19301931

19311932
handler = NULL;
19321933
bridge_ = NULL;
@@ -1983,10 +1984,10 @@ srs_error_t SrsLiveSource::cycle()
19831984
return srs_success;
19841985
}
19851986

1986-
bool SrsLiveSource::expired()
1987+
bool SrsLiveSource::stream_is_dead()
19871988
{
19881989
// unknown state?
1989-
if (die_at == 0) {
1990+
if (stream_die_at_ == 0) {
19901991
return false;
19911992
}
19921993

@@ -2001,13 +2002,26 @@ bool SrsLiveSource::expired()
20012002
}
20022003

20032004
srs_utime_t now = srs_get_system_time();
2004-
if (now > die_at + SRS_SOURCE_CLEANUP) {
2005+
if (now > stream_die_at_ + SRS_SOURCE_CLEANUP) {
20052006
return true;
20062007
}
20072008

20082009
return false;
20092010
}
20102011

2012+
bool SrsLiveSource::publisher_is_idle_for(srs_utime_t timeout)
2013+
{
2014+
if (!publisher_idle_at_ || !timeout) {
2015+
return false;
2016+
}
2017+
2018+
srs_utime_t now = srs_get_system_time();
2019+
if (now > publisher_idle_at_ + timeout) {
2020+
return true;
2021+
}
2022+
return false;
2023+
}
2024+
20112025
srs_error_t SrsLiveSource::initialize(SrsRequest* r, ISrsLiveSourceHandler* h)
20122026
{
20132027
srs_error_t err = srs_success;
@@ -2634,6 +2648,11 @@ srs_error_t SrsLiveSource::on_publish()
26342648

26352649
SrsStatistic* stat = SrsStatistic::instance();
26362650
stat->on_stream_publish(req, _source_id.c_str());
2651+
2652+
// When no players, the publisher is idle now.
2653+
if (consumers.empty()) {
2654+
publisher_idle_at_ = srs_get_system_time();
2655+
}
26372656

26382657
return err;
26392658
}
@@ -2680,7 +2699,7 @@ void SrsLiveSource::on_unpublish()
26802699

26812700
// no consumer, stream is die.
26822701
if (consumers.empty()) {
2683-
die_at = srs_get_system_time();
2702+
stream_die_at_ = srs_get_system_time();
26842703
}
26852704
}
26862705

@@ -2690,7 +2709,11 @@ srs_error_t SrsLiveSource::create_consumer(SrsLiveConsumer*& consumer)
26902709

26912710
consumer = new SrsLiveConsumer(this);
26922711
consumers.push_back(consumer);
2693-
2712+
2713+
// There should be one consumer, so reset the timeout.
2714+
stream_die_at_ = 0;
2715+
publisher_idle_at_ = 0;
2716+
26942717
// for edge, when play edge stream, check the state
26952718
if (_srs_config->get_vhost_is_edge(req->vhost)) {
26962719
// notice edge to start for the first client.
@@ -2752,10 +2775,18 @@ void SrsLiveSource::on_consumer_destroy(SrsLiveConsumer* consumer)
27522775
if (it != consumers.end()) {
27532776
it = consumers.erase(it);
27542777
}
2755-
2778+
27562779
if (consumers.empty()) {
27572780
play_edge->on_all_client_stop();
2758-
die_at = srs_get_system_time();
2781+
2782+
// For edge server, the stream die when the last player quit, because the edge stream is created by player
2783+
// activities, so it should die when all players quit.
2784+
if (_srs_config->get_vhost_is_edge(req->vhost)) {
2785+
stream_die_at_ = srs_get_system_time();
2786+
}
2787+
2788+
// When no players, the publisher is idle now.
2789+
publisher_idle_at_ = srs_get_system_time();
27592790
}
27602791
}
27612792

trunk/src/app/srs_app_source.hpp

+8-5
Original file line numberDiff line numberDiff line change
@@ -533,17 +533,20 @@ class SrsLiveSource : public ISrsReloadHandler
533533
private:
534534
// Whether source is avaiable for publishing.
535535
bool _can_publish;
536-
// The last die time, when all consumers quit and no publisher,
537-
// We will remove the source when source die.
538-
srs_utime_t die_at;
536+
// The last die time, while die means neither publishers nor players.
537+
srs_utime_t stream_die_at_;
538+
// The last idle time, while idle means no players.
539+
srs_utime_t publisher_idle_at_;
539540
public:
540541
SrsLiveSource();
541542
virtual ~SrsLiveSource();
542543
public:
543544
virtual void dispose();
544545
virtual srs_error_t cycle();
545-
// Remove source when expired.
546-
virtual bool expired();
546+
// Whether stream is dead, which is no publisher or player.
547+
virtual bool stream_is_dead();
548+
// Whether publisher is idle for a period of timeout.
549+
bool publisher_is_idle_for(srs_utime_t timeout);
547550
public:
548551
// Initialize the hls with handlers.
549552
virtual srs_error_t initialize(SrsRequest* r, ISrsLiveSourceHandler* h);

trunk/src/core/srs_core_version5.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,6 @@
99

1010
#define VERSION_MAJOR 5
1111
#define VERSION_MINOR 0
12-
#define VERSION_REVISION 143
12+
#define VERSION_REVISION 144
1313

1414
#endif

trunk/src/core/srs_core_version6.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,6 @@
99

1010
#define VERSION_MAJOR 6
1111
#define VERSION_MINOR 0
12-
#define VERSION_REVISION 30
12+
#define VERSION_REVISION 31
1313

1414
#endif

trunk/src/kernel/srs_kernel_error.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,7 @@
167167
XX(ERROR_RTMP_MESSAGE_CREATE , 2053, "MessageCreate", "Failed to create shared pointer message") \
168168
XX(ERROR_RTMP_PROXY_EXCEED , 2054, "RtmpProxy", "Failed to decode message of RTMP proxy") \
169169
XX(ERROR_RTMP_CREATE_STREAM_DEPTH , 2055, "RtmpIdentify", "Failed to identify RTMP client") \
170+
XX(ERROR_KICKOFF_FOR_IDLE , 2056, "KickoffForIdle", "Kickoff for publisher is idle") \
170171
XX(ERROR_CONTROL_REDIRECT , 2997, "RtmpRedirect", "RTMP 302 redirection") \
171172
XX(ERROR_CONTROL_RTMP_CLOSE , 2998, "RtmpClose", "RTMP connection is closed") \
172173
XX(ERROR_CONTROL_REPUBLISH , 2999, "RtmpRepublish", "RTMP stream is republished")

trunk/src/utest/srs_utest_config.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -4461,6 +4461,9 @@ VOID TEST(ConfigEnvTest, CheckEnvValuesVhostPublish)
44614461

44624462
SrsSetEnvConfig(try_annexb_first, "SRS_VHOST_PUBLISH_TRY_ANNEXB_FIRST", "off");
44634463
EXPECT_FALSE(conf.try_annexb_first("__defaultVhost__"));
4464+
4465+
SrsSetEnvConfig(kickoff_for_idle, "SRS_VHOST_PUBLISH_KICKOFF_FOR_IDLE", "30");
4466+
EXPECT_EQ(30 * SRS_UTIME_SECONDS, conf.get_publish_kickoff_for_idle("__defaultVhost__"));
44644467
}
44654468
}
44664469

0 commit comments

Comments
 (0)