mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 01:45:33 +00:00
webrtcstats: Improve selected candidate pair stats by adding ICE candidate info
The implementation follows w3.org specs: * https://www.w3.org/TR/webrtc-stats/#icecandidate-dict* * https://www.w3.org/TR/webrtc-stats/#candidatepair-dict* Corresponding unit tests are also added. Rebased and updated from https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1462 Fixes #1207 Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1998>
This commit is contained in:
parent
2042c2d4f0
commit
3e7fb83393
5 changed files with 400 additions and 68 deletions
|
@ -57,6 +57,8 @@ typedef struct _TransportReceiveBinClass TransportReceiveBinClass;
|
||||||
typedef struct _WebRTCTransceiver WebRTCTransceiver;
|
typedef struct _WebRTCTransceiver WebRTCTransceiver;
|
||||||
typedef struct _WebRTCTransceiverClass WebRTCTransceiverClass;
|
typedef struct _WebRTCTransceiverClass WebRTCTransceiverClass;
|
||||||
|
|
||||||
|
typedef struct _GstWebRTCICECandidateStats GstWebRTCICECandidateStats;
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __WEBRTC_FWD_H__ */
|
#endif /* __WEBRTC_FWD_H__ */
|
||||||
|
|
|
@ -848,6 +848,164 @@ gst_webrtc_ice_set_tos (GstWebRTCICE * ice, GstWebRTCICEStream * stream,
|
||||||
nice_agent_set_stream_tos (ice->priv->nice_agent, item->nice_stream_id, tos);
|
nice_agent_set_stream_tos (ice->priv->nice_agent, item->nice_stream_id, tos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const gchar *
|
||||||
|
_relay_type_to_string (GstUri * turn_server)
|
||||||
|
{
|
||||||
|
const gchar *scheme;
|
||||||
|
const gchar *transport;
|
||||||
|
|
||||||
|
if (!turn_server)
|
||||||
|
return "none";
|
||||||
|
|
||||||
|
scheme = gst_uri_get_scheme (turn_server);
|
||||||
|
transport = gst_uri_get_query_value (turn_server, "transport");
|
||||||
|
|
||||||
|
if (g_strcmp0 (scheme, "turns") == 0) {
|
||||||
|
return "tls";
|
||||||
|
} else if (g_strcmp0 (scheme, "turn") == 0) {
|
||||||
|
if (!transport || g_strcmp0 (transport, "udp") == 0)
|
||||||
|
return "udp";
|
||||||
|
if (!transport || g_strcmp0 (transport, "tcp") == 0)
|
||||||
|
return "tcp";
|
||||||
|
}
|
||||||
|
|
||||||
|
return "none";
|
||||||
|
}
|
||||||
|
|
||||||
|
static gchar *
|
||||||
|
_get_server_url (GstWebRTCICE * ice, NiceCandidate * cand)
|
||||||
|
{
|
||||||
|
switch (cand->type) {
|
||||||
|
case NICE_CANDIDATE_TYPE_RELAYED:
|
||||||
|
return g_strdup (gst_uri_get_host (ice->turn_server));
|
||||||
|
case NICE_CANDIDATE_TYPE_SERVER_REFLEXIVE:
|
||||||
|
return g_strdup (gst_uri_get_host (ice->stun_server));
|
||||||
|
default:
|
||||||
|
return g_strdup ("");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* TODO: replace it with nice_candidate_type_to_string()
|
||||||
|
* when it's ready for use
|
||||||
|
* https://libnice.freedesktop.org/libnice/NiceCandidate.html#nice-candidate-type-to-string
|
||||||
|
*/
|
||||||
|
static const gchar *
|
||||||
|
_candidate_type_to_string (NiceCandidateType type)
|
||||||
|
{
|
||||||
|
switch (type) {
|
||||||
|
case NICE_CANDIDATE_TYPE_HOST:
|
||||||
|
return "host";
|
||||||
|
case NICE_CANDIDATE_TYPE_SERVER_REFLEXIVE:
|
||||||
|
return "srflx";
|
||||||
|
case NICE_CANDIDATE_TYPE_PEER_REFLEXIVE:
|
||||||
|
return "prflx";
|
||||||
|
case NICE_CANDIDATE_TYPE_RELAYED:
|
||||||
|
return "relay";
|
||||||
|
default:
|
||||||
|
g_assert_not_reached ();
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_populate_candidate_stats (GstWebRTCICE * ice, NiceCandidate * cand,
|
||||||
|
GstWebRTCICEStream * stream, GstWebRTCICECandidateStats * stats,
|
||||||
|
gboolean is_local)
|
||||||
|
{
|
||||||
|
gchar ipaddr[INET6_ADDRSTRLEN];
|
||||||
|
|
||||||
|
g_assert (cand != NULL);
|
||||||
|
|
||||||
|
nice_address_to_string (&cand->addr, ipaddr);
|
||||||
|
stats->port = nice_address_get_port (&cand->addr);
|
||||||
|
stats->ipaddr = g_strdup (ipaddr);
|
||||||
|
stats->stream_id = stream->stream_id;
|
||||||
|
stats->type = _candidate_type_to_string (cand->type);
|
||||||
|
stats->prio = cand->priority;
|
||||||
|
stats->proto =
|
||||||
|
cand->transport == NICE_CANDIDATE_TRANSPORT_UDP ? "udp" : "tcp";
|
||||||
|
if (is_local) {
|
||||||
|
if (cand->type == NICE_CANDIDATE_TYPE_RELAYED)
|
||||||
|
stats->relay_proto = _relay_type_to_string (ice->turn_server);
|
||||||
|
stats->url = _get_server_url (ice, cand);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_populate_candidate_list_stats (GstWebRTCICE * ice, GSList * cands,
|
||||||
|
GstWebRTCICEStream * stream, GArray * result, gboolean is_local)
|
||||||
|
{
|
||||||
|
GSList *item;
|
||||||
|
|
||||||
|
for (item = cands; item != NULL; item = item->next) {
|
||||||
|
GstWebRTCICECandidateStats stats;
|
||||||
|
NiceCandidate *c = item->data;
|
||||||
|
_populate_candidate_stats (ice, c, stream, &stats, is_local);
|
||||||
|
g_array_append_val (result, stats);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GArray *
|
||||||
|
gst_webrtc_ice_get_local_candidates (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream)
|
||||||
|
{
|
||||||
|
GSList *cands = NULL;
|
||||||
|
|
||||||
|
GArray *result =
|
||||||
|
g_array_new (FALSE, TRUE, sizeof (GstWebRTCICECandidateStats));
|
||||||
|
|
||||||
|
cands = nice_agent_get_local_candidates (ice->priv->nice_agent,
|
||||||
|
stream->stream_id, NICE_COMPONENT_TYPE_RTP);
|
||||||
|
|
||||||
|
_populate_candidate_list_stats (ice, cands, stream, result, TRUE);
|
||||||
|
g_slist_free_full (cands, (GDestroyNotify) nice_candidate_free);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
GArray *
|
||||||
|
gst_webrtc_ice_get_remote_candidates (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream)
|
||||||
|
{
|
||||||
|
GSList *cands = NULL;
|
||||||
|
|
||||||
|
GArray *result =
|
||||||
|
g_array_new (FALSE, TRUE, sizeof (GstWebRTCICECandidateStats));
|
||||||
|
|
||||||
|
cands = nice_agent_get_remote_candidates (ice->priv->nice_agent,
|
||||||
|
stream->stream_id, NICE_COMPONENT_TYPE_RTP);
|
||||||
|
|
||||||
|
_populate_candidate_list_stats (ice, cands, stream, result, FALSE);
|
||||||
|
g_slist_free_full (cands, (GDestroyNotify) nice_candidate_free);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
gst_webrtc_ice_get_selected_pair (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream, GstWebRTCICECandidateStats ** local_stats,
|
||||||
|
GstWebRTCICECandidateStats ** remote_stats)
|
||||||
|
{
|
||||||
|
NiceCandidate *local_cand = NULL;
|
||||||
|
NiceCandidate *remote_cand = NULL;
|
||||||
|
|
||||||
|
if (stream) {
|
||||||
|
if (nice_agent_get_selected_pair (ice->priv->nice_agent, stream->stream_id,
|
||||||
|
NICE_COMPONENT_TYPE_RTP, &local_cand, &remote_cand)) {
|
||||||
|
*local_stats = g_new0 (GstWebRTCICECandidateStats, 1);
|
||||||
|
_populate_candidate_stats (ice, local_cand, stream, *local_stats, TRUE);
|
||||||
|
|
||||||
|
*remote_stats = g_new0 (GstWebRTCICECandidateStats, 1);
|
||||||
|
_populate_candidate_stats (ice, remote_cand, stream, *remote_stats,
|
||||||
|
FALSE);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
_clear_ice_stream (struct NiceStreamItem *item)
|
_clear_ice_stream (struct NiceStreamItem *item)
|
||||||
{
|
{
|
||||||
|
|
|
@ -56,6 +56,18 @@ struct _GstWebRTCICE
|
||||||
guint max_rtp_port;
|
guint max_rtp_port;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct _GstWebRTCICECandidateStats
|
||||||
|
{
|
||||||
|
gchar *ipaddr;
|
||||||
|
guint port;
|
||||||
|
guint stream_id;
|
||||||
|
const gchar *type;
|
||||||
|
const gchar *proto;
|
||||||
|
const gchar *relay_proto;
|
||||||
|
guint prio;
|
||||||
|
gchar *url;
|
||||||
|
};
|
||||||
|
|
||||||
struct _GstWebRTCICEClass
|
struct _GstWebRTCICEClass
|
||||||
{
|
{
|
||||||
GstObjectClass parent_class;
|
GstObjectClass parent_class;
|
||||||
|
@ -107,6 +119,15 @@ void gst_webrtc_ice_set_on_ice_candidate (GstWebRTCIC
|
||||||
void gst_webrtc_ice_set_tos (GstWebRTCICE * ice,
|
void gst_webrtc_ice_set_tos (GstWebRTCICE * ice,
|
||||||
GstWebRTCICEStream * stream,
|
GstWebRTCICEStream * stream,
|
||||||
guint tos);
|
guint tos);
|
||||||
|
|
||||||
|
GArray * gst_webrtc_ice_get_local_candidates (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream);
|
||||||
|
GArray * gst_webrtc_ice_get_remote_candidates (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream);
|
||||||
|
gboolean gst_webrtc_ice_get_selected_pair (GstWebRTCICE * ice,
|
||||||
|
GstWebRTCICEStream * stream,
|
||||||
|
GstWebRTCICECandidateStats ** local_stats,
|
||||||
|
GstWebRTCICECandidateStats ** remote_stats);
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __GST_WEBRTC_ICE_H__ */
|
#endif /* __GST_WEBRTC_ICE_H__ */
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
|
|
||||||
#include "gstwebrtcstats.h"
|
#include "gstwebrtcstats.h"
|
||||||
#include "gstwebrtcbin.h"
|
#include "gstwebrtcbin.h"
|
||||||
|
#include "icestream.h"
|
||||||
#include "transportstream.h"
|
#include "transportstream.h"
|
||||||
#include "transportreceivebin.h"
|
#include "transportreceivebin.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
|
@ -564,67 +565,138 @@ _get_stats_from_rtp_source_stats (GstWebRTCBin * webrtc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* https://www.w3.org/TR/webrtc-stats/#candidatepair-dict* */
|
/* https://www.w3.org/TR/webrtc-stats/#icecandidate-dict* */
|
||||||
static gchar *
|
static gchar *
|
||||||
_get_stats_from_ice_transport (GstWebRTCBin * webrtc,
|
_get_stats_from_ice_candidates (GstWebRTCBin * webrtc,
|
||||||
GstWebRTCICETransport * transport, const GstStructure * twcc_stats,
|
GstWebRTCICECandidateStats * can, const gchar * transport_id,
|
||||||
GstStructure * s)
|
const gchar * candidate_tag, GstStructure * s)
|
||||||
{
|
{
|
||||||
GstStructure *stats;
|
GstStructure *stats;
|
||||||
|
GstWebRTCStatsType type;
|
||||||
gchar *id;
|
gchar *id;
|
||||||
double ts;
|
double ts;
|
||||||
|
|
||||||
gst_structure_get_double (s, "timestamp", &ts);
|
gst_structure_get_double (s, "timestamp", &ts);
|
||||||
|
|
||||||
|
id = g_strdup_printf ("ice-candidate-%s_%u_%s_%u", candidate_tag,
|
||||||
|
can->stream_id, can->ipaddr, can->port);
|
||||||
|
stats = gst_structure_new_empty (id);
|
||||||
|
|
||||||
|
if (strcmp (candidate_tag, "local")) {
|
||||||
|
type = GST_WEBRTC_STATS_LOCAL_CANDIDATE;
|
||||||
|
} else if (strcmp (candidate_tag, "remote")) {
|
||||||
|
type = GST_WEBRTC_STATS_REMOTE_CANDIDATE;
|
||||||
|
} else {
|
||||||
|
GST_WARNING_OBJECT (webrtc, "Invalid ice candidate tag: %s", candidate_tag);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
_set_base_stats (stats, type, ts, id);
|
||||||
|
|
||||||
|
/* RTCIceCandidateStats
|
||||||
|
DOMString transportId;
|
||||||
|
DOMString address;
|
||||||
|
long port;
|
||||||
|
DOMString protocol;
|
||||||
|
RTCIceCandidateType candidateType;
|
||||||
|
long priority;
|
||||||
|
DOMString url;
|
||||||
|
DOMString relayProtocol;
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (transport_id)
|
||||||
|
gst_structure_set (stats, "transport-id", G_TYPE_STRING, transport_id,
|
||||||
|
NULL);
|
||||||
|
gst_structure_set (stats, "address", G_TYPE_STRING, can->ipaddr, NULL);
|
||||||
|
gst_structure_set (stats, "port", G_TYPE_UINT, can->port, NULL);
|
||||||
|
gst_structure_set (stats, "candidate-type", G_TYPE_STRING, can->type, NULL);
|
||||||
|
gst_structure_set (stats, "priority", G_TYPE_UINT, can->prio, NULL);
|
||||||
|
gst_structure_set (stats, "protocol", G_TYPE_STRING, can->proto, NULL);
|
||||||
|
if (can->relay_proto)
|
||||||
|
gst_structure_set (stats, "relay-protocol", G_TYPE_STRING, can->relay_proto,
|
||||||
|
NULL);
|
||||||
|
if (can->url)
|
||||||
|
gst_structure_set (stats, "url", G_TYPE_STRING, can->url, NULL);
|
||||||
|
|
||||||
|
gst_structure_set (s, id, GST_TYPE_STRUCTURE, stats, NULL);
|
||||||
|
gst_structure_free (stats);
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* https://www.w3.org/TR/webrtc-stats/#candidatepair-dict* */
|
||||||
|
static gchar *
|
||||||
|
_get_stats_from_ice_transport (GstWebRTCBin * webrtc,
|
||||||
|
GstWebRTCICETransport * transport, GstWebRTCICEStream * stream,
|
||||||
|
const GstStructure * twcc_stats, const gchar * transport_id,
|
||||||
|
GstStructure * s)
|
||||||
|
{
|
||||||
|
GstStructure *stats;
|
||||||
|
gchar *id;
|
||||||
|
gchar *local_cand_id = NULL, *remote_cand_id = NULL;
|
||||||
|
double ts;
|
||||||
|
GstWebRTCICECandidateStats *local_cand = NULL, *remote_cand = NULL;
|
||||||
|
|
||||||
|
gst_structure_get_double (s, "timestamp", &ts);
|
||||||
|
|
||||||
id = g_strdup_printf ("ice-candidate-pair_%s", GST_OBJECT_NAME (transport));
|
id = g_strdup_printf ("ice-candidate-pair_%s", GST_OBJECT_NAME (transport));
|
||||||
stats = gst_structure_new_empty (id);
|
stats = gst_structure_new_empty (id);
|
||||||
_set_base_stats (stats, GST_WEBRTC_STATS_TRANSPORT, ts, id);
|
_set_base_stats (stats, GST_WEBRTC_STATS_CANDIDATE_PAIR, ts, id);
|
||||||
|
|
||||||
/* XXX: RTCIceCandidatePairStats
|
/* RTCIceCandidatePairStats
|
||||||
DOMString transportId;
|
DOMString transportId;
|
||||||
DOMString localCandidateId;
|
DOMString localCandidateId;
|
||||||
DOMString remoteCandidateId;
|
DOMString remoteCandidateId;
|
||||||
RTCStatsIceCandidatePairState state;
|
|
||||||
unsigned long long priority;
|
|
||||||
boolean nominated;
|
|
||||||
unsigned long packetsSent;
|
|
||||||
unsigned long packetsReceived;
|
|
||||||
unsigned long long bytesSent;
|
|
||||||
unsigned long long bytesReceived;
|
|
||||||
DOMHighResTimeStamp lastPacketSentTimestamp;
|
|
||||||
DOMHighResTimeStamp lastPacketReceivedTimestamp;
|
|
||||||
DOMHighResTimeStamp firstRequestTimestamp;
|
|
||||||
DOMHighResTimeStamp lastRequestTimestamp;
|
|
||||||
DOMHighResTimeStamp lastResponseTimestamp;
|
|
||||||
double totalRoundTripTime;
|
|
||||||
double currentRoundTripTime;
|
|
||||||
double availableOutgoingBitrate;
|
|
||||||
double availableIncomingBitrate;
|
|
||||||
unsigned long circuitBreakerTriggerCount;
|
|
||||||
unsigned long long requestsReceived;
|
|
||||||
unsigned long long requestsSent;
|
|
||||||
unsigned long long responsesReceived;
|
|
||||||
unsigned long long responsesSent;
|
|
||||||
unsigned long long retransmissionsReceived;
|
|
||||||
unsigned long long retransmissionsSent;
|
|
||||||
unsigned long long consentRequestsSent;
|
|
||||||
DOMHighResTimeStamp consentExpiredTimestamp;
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* XXX: RTCIceCandidateStats
|
XXX: To be added:
|
||||||
DOMString transportId;
|
|
||||||
boolean isRemote;
|
RTCStatsIceCandidatePairState state;
|
||||||
RTCNetworkType networkType;
|
boolean nominated;
|
||||||
DOMString ip;
|
unsigned long packetsSent;
|
||||||
long port;
|
unsigned long packetsReceived;
|
||||||
DOMString protocol;
|
unsigned long long bytesSent;
|
||||||
RTCIceCandidateType candidateType;
|
unsigned long long bytesReceived;
|
||||||
long priority;
|
DOMHighResTimeStamp lastPacketSentTimestamp;
|
||||||
DOMString url;
|
DOMHighResTimeStamp lastPacketReceivedTimestamp;
|
||||||
DOMString relayProtocol;
|
DOMHighResTimeStamp firstRequestTimestamp;
|
||||||
boolean deleted = false;
|
DOMHighResTimeStamp lastRequestTimestamp;
|
||||||
};
|
DOMHighResTimeStamp lastResponseTimestamp;
|
||||||
*/
|
double totalRoundTripTime;
|
||||||
|
double currentRoundTripTime;
|
||||||
|
double availableOutgoingBitrate;
|
||||||
|
double availableIncomingBitrate;
|
||||||
|
unsigned long circuitBreakerTriggerCount;
|
||||||
|
unsigned long long requestsReceived;
|
||||||
|
unsigned long long requestsSent;
|
||||||
|
unsigned long long responsesReceived;
|
||||||
|
unsigned long long responsesSent;
|
||||||
|
unsigned long long retransmissionsReceived;
|
||||||
|
unsigned long long retransmissionsSent;
|
||||||
|
unsigned long long consentRequestsSent;
|
||||||
|
DOMHighResTimeStamp consentExpiredTimestamp;
|
||||||
|
unsigned long packetsDiscardedOnSend;
|
||||||
|
unsigned long long bytesDiscardedOnSend;
|
||||||
|
unsigned long long requestBytesSent;
|
||||||
|
unsigned long long consentRequestBytesSent;
|
||||||
|
unsigned long long responseBytesSent;
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (gst_webrtc_ice_get_selected_pair (webrtc->priv->ice, stream,
|
||||||
|
&local_cand, &remote_cand)) {
|
||||||
|
local_cand_id =
|
||||||
|
_get_stats_from_ice_candidates (webrtc, local_cand, transport_id,
|
||||||
|
"local", s);
|
||||||
|
remote_cand_id =
|
||||||
|
_get_stats_from_ice_candidates (webrtc, remote_cand, transport_id,
|
||||||
|
"remote", s);
|
||||||
|
|
||||||
|
gst_structure_set (stats, "local-candidate-id", G_TYPE_STRING,
|
||||||
|
local_cand_id, NULL);
|
||||||
|
gst_structure_set (stats, "remote-candidate-id", G_TYPE_STRING,
|
||||||
|
remote_cand_id, NULL);
|
||||||
|
} else
|
||||||
|
GST_INFO_OBJECT (webrtc,
|
||||||
|
"No selected ICE candidate pair was found for transport %s",
|
||||||
|
GST_OBJECT_NAME (transport));
|
||||||
|
|
||||||
/* XXX: these stats are at the rtp session level but there isn't a specific
|
/* XXX: these stats are at the rtp session level but there isn't a specific
|
||||||
* stats structure for that. The RTCIceCandidatePairStats is the closest with
|
* stats structure for that. The RTCIceCandidatePairStats is the closest with
|
||||||
|
@ -635,6 +707,20 @@ _get_stats_from_ice_transport (GstWebRTCBin * webrtc,
|
||||||
NULL);
|
NULL);
|
||||||
|
|
||||||
gst_structure_set (s, id, GST_TYPE_STRUCTURE, stats, NULL);
|
gst_structure_set (s, id, GST_TYPE_STRUCTURE, stats, NULL);
|
||||||
|
|
||||||
|
g_free (local_cand_id);
|
||||||
|
g_free (remote_cand_id);
|
||||||
|
|
||||||
|
if (local_cand) {
|
||||||
|
g_free (local_cand->ipaddr);
|
||||||
|
g_free (local_cand->url);
|
||||||
|
}
|
||||||
|
if (remote_cand)
|
||||||
|
g_free (remote_cand->ipaddr);
|
||||||
|
|
||||||
|
g_free (local_cand);
|
||||||
|
g_free (remote_cand);
|
||||||
|
|
||||||
gst_structure_free (stats);
|
gst_structure_free (stats);
|
||||||
|
|
||||||
return id;
|
return id;
|
||||||
|
@ -643,8 +729,8 @@ _get_stats_from_ice_transport (GstWebRTCBin * webrtc,
|
||||||
/* https://www.w3.org/TR/webrtc-stats/#dom-rtctransportstats */
|
/* https://www.w3.org/TR/webrtc-stats/#dom-rtctransportstats */
|
||||||
static gchar *
|
static gchar *
|
||||||
_get_stats_from_dtls_transport (GstWebRTCBin * webrtc,
|
_get_stats_from_dtls_transport (GstWebRTCBin * webrtc,
|
||||||
GstWebRTCDTLSTransport * transport, const GstStructure * twcc_stats,
|
GstWebRTCDTLSTransport * transport, GstWebRTCICEStream * stream,
|
||||||
GstStructure * s)
|
const GstStructure * twcc_stats, GstStructure * s)
|
||||||
{
|
{
|
||||||
GstStructure *stats;
|
GstStructure *stats;
|
||||||
gchar *id;
|
gchar *id;
|
||||||
|
@ -677,26 +763,18 @@ _get_stats_from_dtls_transport (GstWebRTCBin * webrtc,
|
||||||
DOMString issuerCertificateId;
|
DOMString issuerCertificateId;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* XXX: RTCIceCandidateStats
|
ice_id =
|
||||||
DOMString transportId;
|
_get_stats_from_ice_transport (webrtc, transport->transport, stream,
|
||||||
boolean isRemote;
|
twcc_stats, id, s);
|
||||||
DOMString ip;
|
if (ice_id) {
|
||||||
long port;
|
gst_structure_set (stats, "selected-candidate-pair-id", G_TYPE_STRING,
|
||||||
DOMString protocol;
|
ice_id, NULL);
|
||||||
RTCIceCandidateType candidateType;
|
g_free (ice_id);
|
||||||
long priority;
|
}
|
||||||
DOMString url;
|
|
||||||
boolean deleted = false;
|
|
||||||
*/
|
|
||||||
|
|
||||||
gst_structure_set (s, id, GST_TYPE_STRUCTURE, stats, NULL);
|
gst_structure_set (s, id, GST_TYPE_STRUCTURE, stats, NULL);
|
||||||
gst_structure_free (stats);
|
gst_structure_free (stats);
|
||||||
|
|
||||||
ice_id =
|
|
||||||
_get_stats_from_ice_transport (webrtc, transport->transport, twcc_stats,
|
|
||||||
s);
|
|
||||||
g_free (ice_id);
|
|
||||||
|
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -879,7 +957,7 @@ _get_stats_from_pad (GstWebRTCBin * webrtc, GstPad * pad, GstStructure * s)
|
||||||
|
|
||||||
ts_stats.transport_id =
|
ts_stats.transport_id =
|
||||||
_get_stats_from_dtls_transport (webrtc, ts_stats.stream->transport,
|
_get_stats_from_dtls_transport (webrtc, ts_stats.stream->transport,
|
||||||
twcc_stats, s);
|
GST_WEBRTC_ICE_STREAM (ts_stats.stream->stream), twcc_stats, s);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (webrtc, "retrieving rtp stream stats from transport %"
|
GST_DEBUG_OBJECT (webrtc, "retrieving rtp stream stats from transport %"
|
||||||
GST_PTR_FORMAT " rtp session %" GST_PTR_FORMAT " with %u rtp sources, "
|
GST_PTR_FORMAT " rtp session %" GST_PTR_FORMAT " with %u rtp sources, "
|
||||||
|
|
|
@ -1559,6 +1559,29 @@ validate_remote_outbound_rtp_stats (const GstStructure * s,
|
||||||
g_free (local_id);
|
g_free (local_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
validate_candidate_stats (const GstStructure * s, const GstStructure * stats)
|
||||||
|
{
|
||||||
|
guint port;
|
||||||
|
guint64 priority;
|
||||||
|
gchar *address, *candidateType, *protocol;
|
||||||
|
|
||||||
|
fail_unless (gst_structure_get (s, "address", G_TYPE_STRING, &address, NULL));
|
||||||
|
fail_unless (gst_structure_get (s, "port", G_TYPE_UINT, &port, NULL));
|
||||||
|
fail_unless (gst_structure_get (s, "candidate-type", G_TYPE_STRING,
|
||||||
|
&candidateType, NULL));
|
||||||
|
fail_unless (gst_structure_get (s, "priority", G_TYPE_UINT64, &priority,
|
||||||
|
NULL));
|
||||||
|
fail_unless (gst_structure_get (s, "protocol", G_TYPE_STRING, &protocol,
|
||||||
|
NULL));
|
||||||
|
|
||||||
|
fail_unless (strcmp (protocol, "udp") || strcmp (protocol, "tcp"));
|
||||||
|
|
||||||
|
g_free (address);
|
||||||
|
g_free (candidateType);
|
||||||
|
g_free (protocol);
|
||||||
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
validate_stats_foreach (GQuark field_id, const GValue * value,
|
validate_stats_foreach (GQuark field_id, const GValue * value,
|
||||||
const GstStructure * stats)
|
const GstStructure * stats)
|
||||||
|
@ -1592,7 +1615,9 @@ validate_stats_foreach (GQuark field_id, const GValue * value,
|
||||||
} else if (type == GST_WEBRTC_STATS_TRANSPORT) {
|
} else if (type == GST_WEBRTC_STATS_TRANSPORT) {
|
||||||
} else if (type == GST_WEBRTC_STATS_CANDIDATE_PAIR) {
|
} else if (type == GST_WEBRTC_STATS_CANDIDATE_PAIR) {
|
||||||
} else if (type == GST_WEBRTC_STATS_LOCAL_CANDIDATE) {
|
} else if (type == GST_WEBRTC_STATS_LOCAL_CANDIDATE) {
|
||||||
|
validate_candidate_stats (s, stats);
|
||||||
} else if (type == GST_WEBRTC_STATS_REMOTE_CANDIDATE) {
|
} else if (type == GST_WEBRTC_STATS_REMOTE_CANDIDATE) {
|
||||||
|
validate_candidate_stats (s, stats);
|
||||||
} else if (type == GST_WEBRTC_STATS_CERTIFICATE) {
|
} else if (type == GST_WEBRTC_STATS_CERTIFICATE) {
|
||||||
} else {
|
} else {
|
||||||
g_assert_not_reached ();
|
g_assert_not_reached ();
|
||||||
|
@ -1646,6 +1671,53 @@ GST_START_TEST (test_session_stats)
|
||||||
|
|
||||||
GST_END_TEST;
|
GST_END_TEST;
|
||||||
|
|
||||||
|
GST_START_TEST (test_stats_with_stream)
|
||||||
|
{
|
||||||
|
struct test_webrtc *t = create_audio_test ();
|
||||||
|
GstPromise *p;
|
||||||
|
GstCaps *caps;
|
||||||
|
GstPad *pad;
|
||||||
|
|
||||||
|
/* test that the stats generated with stream are sane */
|
||||||
|
|
||||||
|
t->on_offer_created = NULL;
|
||||||
|
t->on_answer_created = NULL;
|
||||||
|
t->on_negotiation_needed = NULL;
|
||||||
|
|
||||||
|
fail_if (gst_element_set_state (t->webrtc1,
|
||||||
|
GST_STATE_READY) == GST_STATE_CHANGE_FAILURE);
|
||||||
|
fail_if (gst_element_set_state (t->webrtc2,
|
||||||
|
GST_STATE_READY) == GST_STATE_CHANGE_FAILURE);
|
||||||
|
|
||||||
|
test_webrtc_create_offer (t);
|
||||||
|
|
||||||
|
fail_if (gst_element_set_state (t->webrtc1,
|
||||||
|
GST_STATE_PLAYING) == GST_STATE_CHANGE_FAILURE);
|
||||||
|
fail_if (gst_element_set_state (t->webrtc2,
|
||||||
|
GST_STATE_PLAYING) == GST_STATE_CHANGE_FAILURE);
|
||||||
|
|
||||||
|
/* set caps for webrtcbin sink to validate codec stats */
|
||||||
|
caps = gst_caps_from_string (OPUS_RTP_CAPS (96));
|
||||||
|
pad = gst_element_get_static_pad (t->webrtc1, "sink_0");
|
||||||
|
gst_pad_set_caps (pad, caps);
|
||||||
|
gst_caps_unref (caps);
|
||||||
|
|
||||||
|
test_webrtc_wait_for_answer_error_eos (t);
|
||||||
|
fail_unless (t->state == STATE_ANSWER_SET);
|
||||||
|
|
||||||
|
p = gst_promise_new_with_change_func (_on_stats, t, NULL);
|
||||||
|
g_signal_emit_by_name (t->webrtc1, "get-stats", NULL, p);
|
||||||
|
p = gst_promise_new_with_change_func (_on_stats, t, NULL);
|
||||||
|
g_signal_emit_by_name (t->webrtc2, "get-stats", NULL, p);
|
||||||
|
|
||||||
|
test_webrtc_wait_for_state_mask (t, 1 << STATE_CUSTOM);
|
||||||
|
|
||||||
|
gst_object_unref (pad);
|
||||||
|
test_webrtc_free (t);
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_END_TEST;
|
||||||
|
|
||||||
GST_START_TEST (test_add_transceiver)
|
GST_START_TEST (test_add_transceiver)
|
||||||
{
|
{
|
||||||
struct test_webrtc *t = test_webrtc_new ();
|
struct test_webrtc *t = test_webrtc_new ();
|
||||||
|
@ -5260,6 +5332,7 @@ webrtcbin_suite (void)
|
||||||
if (nicesrc && nicesink && dtlssrtpenc && dtlssrtpdec) {
|
if (nicesrc && nicesink && dtlssrtpenc && dtlssrtpdec) {
|
||||||
tcase_add_test (tc, test_sdp_no_media);
|
tcase_add_test (tc, test_sdp_no_media);
|
||||||
tcase_add_test (tc, test_session_stats);
|
tcase_add_test (tc, test_session_stats);
|
||||||
|
tcase_add_test (tc, test_stats_with_stream);
|
||||||
tcase_add_test (tc, test_audio);
|
tcase_add_test (tc, test_audio);
|
||||||
tcase_add_test (tc, test_ice_port_restriction);
|
tcase_add_test (tc, test_ice_port_restriction);
|
||||||
tcase_add_test (tc, test_audio_video);
|
tcase_add_test (tc, test_audio_video);
|
||||||
|
|
Loading…
Reference in a new issue