1
0
Fork 0
mirror of https://github.com/ossrs/srs.git synced 2025-02-13 20:01:56 +00:00
srs/trunk/src/app/srs_app_edge.hpp

259 lines
7.2 KiB
C++
Raw Normal View History

//
2024-01-01 02:51:24 +00:00
// Copyright (c) 2013-2024 The SRS Authors
//
2023-10-23 06:33:19 +00:00
// SPDX-License-Identifier: MIT
//
2014-04-25 08:35:03 +00:00
#ifndef SRS_APP_EDGE_HPP
#define SRS_APP_EDGE_HPP
#include <srs_core.hpp>
#include <srs_app_st.hpp>
#include <srs_core_autofree.hpp>
2014-04-26 10:08:21 +00:00
#include <string>
class SrsStSocket;
class SrsRtmpServer;
2021-05-16 08:14:00 +00:00
class SrsLiveSource;
2014-04-26 09:16:18 +00:00
class SrsRequest;
2014-04-27 01:29:37 +00:00
class SrsPlayEdge;
class SrsPublishEdge;
class SrsRtmpClient;
class SrsCommonMessage;
2014-04-28 09:20:35 +00:00
class SrsMessageQueue;
class ISrsProtocolReadWriter;
2014-05-12 09:27:50 +00:00
class SrsKbps;
class SrsLbRoundRobin;
2015-10-14 02:48:08 +00:00
class SrsTcpClient;
2015-10-14 06:37:24 +00:00
class SrsSimpleRtmpClient;
class SrsPacket;
class SrsHttpClient;
class ISrsHttpMessage;
class SrsHttpFileReader;
class SrsFlvDecoder;
class ISrsApmSpan;
2014-04-26 09:16:18 +00:00
2019-04-30 00:24:52 +00:00
// The state of edge, auto machine
enum SrsEdgeState
{
SrsEdgeStateInit = 0,
2017-03-25 09:21:39 +00:00
2019-04-30 00:24:52 +00:00
// For play edge
SrsEdgeStatePlay = 100,
// play stream from origin, ingest stream
SrsEdgeStateIngestConnected = 101,
2019-04-30 00:24:52 +00:00
// For publish edge
2014-04-27 01:29:37 +00:00
SrsEdgeStatePublish = 200,
// We are stopping edge ingesting.
SrsEdgeStateIngestStopping = 300,
};
2019-04-30 00:24:52 +00:00
// The state of edge from user, manual machine
enum SrsEdgeUserState
{
SrsEdgeUserStateInit = 0,
SrsEdgeUserStateReloading = 100,
};
2019-04-30 00:24:52 +00:00
// The upstream of edge, can be rtmp or http.
class SrsEdgeUpstream
{
public:
SrsEdgeUpstream();
virtual ~SrsEdgeUpstream();
public:
2018-01-01 11:39:57 +00:00
virtual srs_error_t connect(SrsRequest* r, SrsLbRoundRobin* lb) = 0;
virtual srs_error_t recv_message(SrsCommonMessage** pmsg) = 0;
virtual srs_error_t decode_message(SrsCommonMessage* msg, SrsPacket** ppacket) = 0;
virtual void close() = 0;
public:
2019-12-01 11:24:17 +00:00
virtual void selected(std::string& server, int& port) = 0;
2019-04-17 00:31:53 +00:00
virtual void set_recv_timeout(srs_utime_t tm) = 0;
virtual void kbps_sample(const char* label, srs_utime_t age) = 0;
};
class SrsEdgeRtmpUpstream : public SrsEdgeUpstream
{
private:
2019-04-30 00:24:52 +00:00
// For RTMP 302, if not empty,
2016-01-11 07:46:23 +00:00
// use this <ip[:port]> as upstream.
std::string redirect;
SrsSimpleRtmpClient* sdk;
2019-12-01 11:24:17 +00:00
private:
// Current selected server, the ip:port.
std::string selected_ip;
int selected_port;
public:
2016-01-11 07:46:23 +00:00
// @param rediect, override the server. ignore if empty.
SrsEdgeRtmpUpstream(std::string r);
virtual ~SrsEdgeRtmpUpstream();
public:
2018-01-01 11:39:57 +00:00
virtual srs_error_t connect(SrsRequest* r, SrsLbRoundRobin* lb);
virtual srs_error_t recv_message(SrsCommonMessage** pmsg);
virtual srs_error_t decode_message(SrsCommonMessage* msg, SrsPacket** ppacket);
virtual void close();
public:
2019-12-01 11:24:17 +00:00
virtual void selected(std::string& server, int& port);
2019-04-17 00:31:53 +00:00
virtual void set_recv_timeout(srs_utime_t tm);
virtual void kbps_sample(const char* label, srs_utime_t age);
};
class SrsEdgeFlvUpstream : public SrsEdgeUpstream
{
private:
std::string schema_;
SrsHttpClient* sdk_;
ISrsHttpMessage* hr_;
private:
SrsHttpFileReader* reader_;
SrsFlvDecoder* decoder_;
private:
// We might modify the request by HTTP redirect.
SrsRequest* req_;
// Current selected server, the ip:port.
std::string selected_ip;
int selected_port;
public:
SrsEdgeFlvUpstream(std::string schema);
virtual ~SrsEdgeFlvUpstream();
public:
virtual srs_error_t connect(SrsRequest* r, SrsLbRoundRobin* lb);
private:
virtual srs_error_t do_connect(SrsRequest* r, SrsLbRoundRobin* lb, int redirect_depth);
public:
virtual srs_error_t recv_message(SrsCommonMessage** pmsg);
virtual srs_error_t decode_message(SrsCommonMessage* msg, SrsPacket** ppacket);
virtual void close();
public:
virtual void selected(std::string& server, int& port);
virtual void set_recv_timeout(srs_utime_t tm);
virtual void kbps_sample(const char* label, srs_utime_t age);
};
2019-04-30 00:24:52 +00:00
// The edge used to ingest stream from origin.
class SrsEdgeIngester : public ISrsCoroutineHandler
2014-04-26 10:08:21 +00:00
{
private:
// Because source references to this object, so we should directly use the source ptr.
SrsLiveSource* source_;
private:
2015-10-14 06:37:24 +00:00
SrsPlayEdge* edge;
SrsRequest* req;
SrsCoroutine* trd;
SrsLbRoundRobin* lb;
SrsEdgeUpstream* upstream;
#ifdef SRS_APM
ISrsApmSpan* span_main_;
#endif
2014-04-26 10:08:21 +00:00
public:
SrsEdgeIngester();
virtual ~SrsEdgeIngester();
public:
virtual srs_error_t initialize(SrsSharedPtr<SrsLiveSource> s, SrsPlayEdge* e, SrsRequest* r);
virtual srs_error_t start();
virtual void stop();
2015-09-17 05:36:02 +00:00
virtual std::string get_curr_origin();
#ifdef SRS_APM
// Get the current main span. Note that it might be NULL.
ISrsApmSpan* span();
#endif
2019-04-30 00:30:13 +00:00
// Interface ISrsReusableThread2Handler
2014-04-26 10:08:21 +00:00
public:
virtual srs_error_t cycle();
private:
virtual srs_error_t do_cycle();
private:
virtual srs_error_t ingest(std::string& redirect);
virtual srs_error_t process_publish_message(SrsCommonMessage* msg, std::string& redirect);
2014-04-26 10:08:21 +00:00
};
2019-04-30 00:24:52 +00:00
// The edge used to forward stream to origin.
class SrsEdgeForwarder : public ISrsCoroutineHandler
2014-04-27 01:29:37 +00:00
{
private:
// Because source references to this object, so we should directly use the source ptr.
SrsLiveSource* source_;
private:
2015-10-14 07:08:55 +00:00
SrsPublishEdge* edge;
SrsRequest* req;
SrsCoroutine* trd;
2015-10-14 07:08:55 +00:00
SrsSimpleRtmpClient* sdk;
2015-09-24 10:33:07 +00:00
SrsLbRoundRobin* lb;
2019-04-30 00:24:52 +00:00
// we must ensure one thread one fd principle,
// that is, a fd must be write/read by the one thread.
// The publish service thread will proxy(msg), and the edge forward thread
// will cycle(), so we use queue for cycle to send the msg of proxy.
2014-04-28 09:20:35 +00:00
SrsMessageQueue* queue;
2019-04-30 00:24:52 +00:00
// error code of send, for edge proxy thread to query.
2014-04-28 09:20:35 +00:00
int send_error_code;
2014-04-27 01:29:37 +00:00
public:
SrsEdgeForwarder();
virtual ~SrsEdgeForwarder();
2014-04-28 09:20:35 +00:00
public:
virtual void set_queue_size(srs_utime_t queue_size);
2014-04-27 01:29:37 +00:00
public:
virtual srs_error_t initialize(SrsSharedPtr<SrsLiveSource> s, SrsPublishEdge* e, SrsRequest* r);
virtual srs_error_t start();
2014-04-27 01:29:37 +00:00
virtual void stop();
2019-04-30 00:30:13 +00:00
// Interface ISrsReusableThread2Handler
public:
virtual srs_error_t cycle();
private:
virtual srs_error_t do_cycle();
2014-04-27 01:29:37 +00:00
public:
2017-09-23 14:12:33 +00:00
virtual srs_error_t proxy(SrsCommonMessage* msg);
2014-04-27 01:29:37 +00:00
};
2019-04-30 00:24:52 +00:00
// The play edge control service.
2014-04-27 01:29:37 +00:00
class SrsPlayEdge
{
private:
SrsEdgeState state;
2014-04-26 10:08:21 +00:00
SrsEdgeIngester* ingester;
public:
2014-04-27 01:29:37 +00:00
SrsPlayEdge();
virtual ~SrsPlayEdge();
public:
2019-04-30 00:24:52 +00:00
// Always use the req of source,
// For we assume all client to edge is invalid,
// if auth open, edge must valid it from origin, then service it.
virtual srs_error_t initialize(SrsSharedPtr<SrsLiveSource> source, SrsRequest* req);
2019-04-30 00:24:52 +00:00
// When client play stream on edge.
2018-01-01 11:39:57 +00:00
virtual srs_error_t on_client_play();
2019-04-30 00:24:52 +00:00
// When all client stopped play, disconnect to origin.
virtual void on_all_client_stop();
2015-09-17 05:36:02 +00:00
virtual std::string get_curr_origin();
public:
2019-04-30 00:24:52 +00:00
// When ingester start to play stream.
2018-01-01 11:39:57 +00:00
virtual srs_error_t on_ingest_play();
};
2019-04-30 00:24:52 +00:00
// The publish edge control service.
2014-04-27 01:29:37 +00:00
class SrsPublishEdge
{
private:
SrsEdgeState state;
SrsEdgeForwarder* forwarder;
public:
SrsPublishEdge();
virtual ~SrsPublishEdge();
2014-04-28 09:20:35 +00:00
public:
virtual void set_queue_size(srs_utime_t queue_size);
2014-04-27 01:29:37 +00:00
public:
virtual srs_error_t initialize(SrsSharedPtr<SrsLiveSource> source, SrsRequest* req);
virtual bool can_publish();
2019-04-30 00:24:52 +00:00
// When client publish stream on edge.
2017-09-23 14:12:33 +00:00
virtual srs_error_t on_client_publish();
2019-04-30 00:24:52 +00:00
// Proxy publish stream to edge
2017-09-23 14:12:33 +00:00
virtual srs_error_t on_proxy_publish(SrsCommonMessage* msg);
2019-04-30 00:24:52 +00:00
// Proxy unpublish stream to edge.
2014-04-27 06:57:28 +00:00
virtual void on_proxy_unpublish();
2014-04-27 01:29:37 +00:00
};
#endif
2014-08-02 14:18:39 +00:00