mirror of
https://github.com/ossrs/srs.git
synced 2025-03-09 15:49:59 +00:00
The heartbeat of SRS is a timer that requests an HTTP URL. We can use this heartbeat to report the necessary information for registering the backend server with the proxy server. ```text SRS(backend) --heartbeat---> Proxy server ``` A proxy server is a specialized load balancer for media servers. It operates at the application level rather than the TCP level. For more information about the proxy server, see issue #4158. Note that we will merge this PR into SRS 5.0+, allowing the use of SRS 5.0+ as the backend server, not limited to SRS 7.0. However, the proxy server is introduced in SRS 7.0. It's also possible to implement a registration service, allowing you to use other media servers as backend servers. For example, if you gather information about an nginx-rtmp server and register it with the proxy server, the proxy will forward RTMP streams to nginx-rtmp. The backend server is not limited to SRS. --------- Co-authored-by: Jacob Su <suzp1984@gmail.com>
157 lines
4.8 KiB
C++
157 lines
4.8 KiB
C++
//
|
|
// Copyright (c) 2013-2024 The SRS Authors
|
|
//
|
|
// SPDX-License-Identifier: MIT or MulanPSL-2.0
|
|
//
|
|
|
|
#include <srs_app_heartbeat.hpp>
|
|
|
|
#include <sstream>
|
|
using namespace std;
|
|
|
|
#include <srs_kernel_error.hpp>
|
|
#include <srs_kernel_log.hpp>
|
|
#include <srs_app_config.hpp>
|
|
#include <srs_app_http_client.hpp>
|
|
#include <srs_protocol_json.hpp>
|
|
#include <srs_app_utility.hpp>
|
|
#include <srs_core_autofree.hpp>
|
|
#include <srs_app_http_conn.hpp>
|
|
#include <srs_protocol_amf0.hpp>
|
|
#include <srs_kernel_utility.hpp>
|
|
#include <srs_app_statistic.hpp>
|
|
|
|
SrsHttpHeartbeat::SrsHttpHeartbeat()
|
|
{
|
|
}
|
|
|
|
SrsHttpHeartbeat::~SrsHttpHeartbeat()
|
|
{
|
|
}
|
|
|
|
void SrsHttpHeartbeat::heartbeat()
|
|
{
|
|
srs_error_t err = do_heartbeat();
|
|
if (err != srs_success) {
|
|
srs_warn("heartbeat err=%s", srs_error_desc(err).c_str());
|
|
}
|
|
srs_freep(err);
|
|
return;
|
|
}
|
|
|
|
srs_error_t SrsHttpHeartbeat::do_heartbeat()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
std::string url = _srs_config->get_heartbeat_url();
|
|
|
|
SrsHttpUri uri;
|
|
if ((err = uri.initialize(url)) != srs_success) {
|
|
return srs_error_wrap(err, "http uri parse hartbeart url failed. url=%s", url.c_str());
|
|
}
|
|
|
|
string ip;
|
|
std::string device_id = _srs_config->get_heartbeat_device_id();
|
|
|
|
// Try to load the ip from the environment variable.
|
|
ip = srs_getenv("srs.device.ip"); // SRS_DEVICE_IP
|
|
if (ip.empty()) {
|
|
// Use the local ip address specified by the stats.network config.
|
|
vector<SrsIPAddress*>& ips = srs_get_local_ips();
|
|
if (!ips.empty()) {
|
|
ip = ips[_srs_config->get_stats_network() % (int) ips.size()]->ip;
|
|
}
|
|
}
|
|
|
|
SrsJsonObject* obj = SrsJsonAny::object();
|
|
SrsAutoFree(SrsJsonObject, obj);
|
|
|
|
obj->set("device_id", SrsJsonAny::str(device_id.c_str()));
|
|
obj->set("ip", SrsJsonAny::str(ip.c_str()));
|
|
|
|
SrsStatistic* stat = SrsStatistic::instance();
|
|
obj->set("server", SrsJsonAny::str(stat->server_id().c_str()));
|
|
obj->set("service", SrsJsonAny::str(stat->service_id().c_str()));
|
|
obj->set("pid", SrsJsonAny::str(stat->service_pid().c_str()));
|
|
|
|
if (_srs_config->get_heartbeat_summaries()) {
|
|
SrsJsonObject* summaries = SrsJsonAny::object();
|
|
obj->set("summaries", summaries);
|
|
|
|
srs_api_dump_summaries(summaries);
|
|
}
|
|
|
|
if (_srs_config->get_heartbeat_ports()) {
|
|
// For RTMP listen endpoints.
|
|
if (true) {
|
|
SrsJsonArray* o = SrsJsonAny::array();
|
|
obj->set("rtmp", o);
|
|
|
|
vector<string> endpoints = _srs_config->get_listens();
|
|
for (int i = 0; i < (int) endpoints.size(); i++) {
|
|
o->append(SrsJsonAny::str(endpoints.at(i).c_str()));
|
|
}
|
|
}
|
|
|
|
// For HTTP Stream listen endpoints.
|
|
if (_srs_config->get_http_stream_enabled()) {
|
|
SrsJsonArray* o = SrsJsonAny::array();
|
|
obj->set("http", o);
|
|
|
|
string endpoint = _srs_config->get_http_stream_listen();
|
|
o->append(SrsJsonAny::str(endpoint.c_str()));
|
|
}
|
|
|
|
// For HTTP API listen endpoints.
|
|
if (_srs_config->get_http_api_enabled()) {
|
|
SrsJsonArray* o = SrsJsonAny::array();
|
|
obj->set("api", o);
|
|
|
|
string endpoint = _srs_config->get_http_api_listen();
|
|
o->append(SrsJsonAny::str(endpoint.c_str()));
|
|
}
|
|
|
|
// For SRT listen endpoints.
|
|
if (_srs_config->get_srt_enabled()) {
|
|
SrsJsonArray* o = SrsJsonAny::array();
|
|
obj->set("srt", o);
|
|
|
|
uint16_t endpoint = _srs_config->get_srt_listen_port();
|
|
o->append(SrsJsonAny::str(srs_fmt("udp://0.0.0.0:%d", endpoint).c_str()));
|
|
}
|
|
|
|
// For WebRTC listen endpoints.
|
|
if (_srs_config->get_rtc_server_enabled()) {
|
|
SrsJsonArray* o = SrsJsonAny::array();
|
|
obj->set("rtc", o);
|
|
|
|
int endpoint = _srs_config->get_rtc_server_listen();
|
|
o->append(SrsJsonAny::str(srs_fmt("udp://0.0.0.0:%d", endpoint).c_str()));
|
|
|
|
if (_srs_config->get_rtc_server_tcp_enabled()) {
|
|
endpoint = _srs_config->get_rtc_server_tcp_listen();
|
|
o->append(SrsJsonAny::str(srs_fmt("tcp://0.0.0.0:%d", endpoint).c_str()));
|
|
}
|
|
}
|
|
}
|
|
|
|
SrsHttpClient http;
|
|
if ((err = http.initialize(uri.get_schema(), uri.get_host(), uri.get_port())) != srs_success) {
|
|
return srs_error_wrap(err, "init uri=%s", uri.get_url().c_str());
|
|
}
|
|
|
|
std::string req = obj->dumps();
|
|
ISrsHttpMessage* msg = NULL;
|
|
if ((err = http.post(uri.get_path(), req, &msg)) != srs_success) {
|
|
return srs_error_wrap(err, "http post hartbeart uri failed. url=%s, request=%s", url.c_str(), req.c_str());
|
|
}
|
|
SrsAutoFree(ISrsHttpMessage, msg);
|
|
|
|
std::string res;
|
|
if ((err = msg->body_read_all(res)) != srs_success) {
|
|
return srs_error_wrap(err, "read body");
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|