Files
media-server/librtsp/test/rtsp-client-push-test.cpp

300 lines
8.3 KiB
C++

#if defined(_DEBUG) || defined(DEBUG)
#include "sockutil.h"
#include "rtsp-client.h"
#include <assert.h>
#include <stdlib.h>
#include "sockpair.h"
#include "cstringext.h"
#include "sys/system.h"
#include "cpm/unuse.h"
#include "sdp.h"
#include "ntp-time.h"
#include "rtp-profile.h"
#include "media/ps-file-source.h"
#include "media/h264-file-source.h"
#include "media/h265-file-source.h"
#include "media/mp4-file-source.h"
#include "rtp-udp-transport.h"
#include <map>
#include <memory>
#include <string>
#include "cpm/shared_ptr.h"
#if defined(_HAVE_FFMPEG_)
#include "media/ffmpeg-file-source.h"
#include "media/ffmpeg-live-source.h"
#endif
//#define UDP_MULTICAST_ADDR "239.0.0.2"
extern "C" int rtsp_addr_is_multicast(const char* ip);
struct rtsp_client_push_test_t
{
std::shared_ptr<IMediaSource> source;
rtsp_client_t* rtsp;
socket_t socket;
char sdp[2 * 1024];
std::string peer;
int mode;
int status;
socket_t rtp[5][2];
unsigned short port[5][2];
std::shared_ptr<IRTPTransport> transport[5];
};
static int rtsp_client_sdp(struct rtsp_client_push_test_t* ctx, const char* file)
{
static const char* pattern_vod =
"v=0\n"
"o=- %llu %llu IN IP4 %s\n"
"s=rtsp-client-push-test\n"
"c=IN IP4 0.0.0.0\n"
"t=0 0\n"
"a=range:npt=0-%.1f\n"
"a=recvonly\n"
"a=control:*\n"; // aggregate control
static const char* pattern_live =
"v=0\n"
"o=- %llu %llu IN IP4 %s\n"
"s=rtsp-client-push-test\n"
"c=IN IP4 0.0.0.0\n"
"t=0 0\n"
"a=range:npt=now-\n" // live
"a=recvonly\n"
"a=control:*\n"; // aggregate control
int offset = 0;
if (0 == strcmp(file, "camera"))
{
#if defined(_HAVE_FFMPEG_)
ctx->source.reset(new FFLiveSource("video=Integrated Webcam"));
#endif
offset = snprintf(ctx->sdp, sizeof(ctx->sdp), pattern_live, ntp64_now(), ntp64_now(), "0.0.0.0");
assert(offset > 0 && offset + 1 < sizeof(ctx->sdp));
}
else
{
if (strendswith(file, ".ps"))
ctx->source.reset(new PSFileSource(file));
else if (strendswith(file, ".h264"))
ctx->source.reset(new H264FileSource(file));
else if (strendswith(file, ".h265"))
ctx->source.reset(new H265FileSource(file));
else
{
#if defined(_HAVE_FFMPEG_)
ctx->source.reset(new FFFileSource(file));
#else
ctx->source.reset(new MP4FileSource(file));
#endif
}
int64_t duration;
ctx->source->GetDuration(duration);
offset = snprintf(ctx->sdp, sizeof(ctx->sdp), pattern_vod, ntp64_now(), ntp64_now(), "0.0.0.0", duration / 1000.0);
assert(offset > 0 && offset + 1 < sizeof(ctx->sdp));
}
std::string sdpmedia;
ctx->source->GetSDPMedia(sdpmedia);
return offset + snprintf(ctx->sdp + offset, sizeof(ctx->sdp) - offset, "%s", sdpmedia.c_str());
}
static int rtsp_client_send(void* param, const char* uri, const void* req, size_t bytes)
{
//TODO: check uri and make socket
//1. uri != rtsp describe uri(user input)
//2. multi-uri if media_count > 1
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
return socket_send_all_by_time(ctx->socket, req, bytes, 0, 2000);
}
static int rtpport(void* param, int media, const char* source, unsigned short rtp[2], char* ip, int len)
{
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
int m = rtsp_client_get_media_type(ctx->rtsp, media);
if (SDP_M_MEDIA_AUDIO != m && SDP_M_MEDIA_VIDEO != m)
return 0; // ignore
switch (ctx->mode)
{
case RTSP_TRANSPORT_RTP_UDP:
// TODO: ipv6
assert(0 == sockpair_create("0.0.0.0", ctx->rtp[media], ctx->port[media]));
rtp[0] = ctx->port[media][0];
rtp[1] = ctx->port[media][1];
if (rtsp_addr_is_multicast(ip))
{
if (0 != socket_udp_multicast(ctx->rtp[media][0], ip, source, 16) || 0 != socket_udp_multicast(ctx->rtp[media][1], ip, source, 16))
return -1;
}
#if defined(UDP_MULTICAST_ADDR)
else
{
if (0 != socket_udp_multicast(ctx->rtp[media][0], UDP_MULTICAST_ADDR, source, 16) || 0 != socket_udp_multicast(ctx->rtp[media][1], UDP_MULTICAST_ADDR, source, 16))
return -1;
snprintf(ip, len, "%s", UDP_MULTICAST_ADDR);
}
#endif
break;
case RTSP_TRANSPORT_RTP_TCP:
rtp[0] = 2 * media;
rtp[1] = 2 * media + 1;
break;
default:
assert(0);
return -1;
}
return ctx->mode;
}
int rtsp_client_options(rtsp_client_t* rtsp, const char* commands);
static void onrtp(void* param, uint8_t channel, const void* data, uint16_t bytes)
{
static int keepalive = 0;
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
//rtp_receiver_tcp_input(channel, data, bytes);
//if (++keepalive % 1000 == 0)
//{
// rtsp_client_play(ctx->rtsp, NULL, NULL);
//}
}
static int onannounce(void* param)
{
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
return rtsp_client_setup(ctx->rtsp, ctx->sdp, strlen(ctx->sdp));
}
static int onsetup(void* param, int timeout, int64_t duration)
{
int i;
uint64_t npt = 0;
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
for (i = 0; i < rtsp_client_media_count(ctx->rtsp); i++)
{
int payload;
unsigned short port[2];
const char* encoding;
const struct rtsp_header_transport_t* transport;
char track[16] = { 0 };
#if defined(_HAVE_FFMPEG_)
snprintf(track, sizeof(track) - 1, "track%d", i);
#else
snprintf(track, sizeof(track) - 1, "track%d", i + 1); // mp4 track base 1
#endif
transport = rtsp_client_get_media_transport(ctx->rtsp, i);
encoding = rtsp_client_get_media_encoding(ctx->rtsp, i);
payload = rtsp_client_get_media_payload(ctx->rtsp, i);
if (RTSP_TRANSPORT_RTP_UDP == transport->transport)
{
//assert(RTSP_TRANSPORT_RTP_UDP == transport->transport); // udp only
assert(0 == transport->multicast); // unicast only
assert(transport->rtp.u.client_port1 == ctx->port[i][0]);
assert(transport->rtp.u.client_port2 == ctx->port[i][1]);
port[0] = transport->rtp.u.server_port1;
port[1] = transport->rtp.u.server_port2;
ctx->transport[i] = std::make_shared<RTPUdpTransport>();
assert(transport->rtp.u.server_port1 && transport->rtp.u.server_port2);
const char* ip = transport->destination[0] ? transport->destination : ctx->peer.c_str();
assert(0 == ((RTPUdpTransport*)ctx->transport[i].get())->Init(ctx->rtp[i], ip, port));
ctx->source->SetTransport(track, ctx->transport[i]);
}
else if (RTSP_TRANSPORT_RTP_TCP == transport->transport)
{
//assert(transport->rtp.u.client_port1 == transport->interleaved1);
//assert(transport->rtp.u.client_port2 == transport->interleaved2);
assert(0); // todo
}
else
{
assert(0); // TODO
}
}
assert(0 == rtsp_client_record(ctx->rtsp, &npt, NULL));
return 0;
}
static int onteardown(void* param)
{
return 0;
}
static int onrecord(void* param, int media, const uint64_t* nptbegin, const uint64_t* nptend, const double* scale, const struct rtsp_rtp_info_t* rtpinfo, int count)
{
struct rtsp_client_push_test_t* ctx = (struct rtsp_client_push_test_t*)param;
ctx->status = 1;
return 0;
}
void rtsp_client_push_test(const char* host, const char* file)
{
int r;
struct rtsp_client_push_test_t ctx;
struct rtsp_client_handler_t handler;
static char packet[2 * 1024 * 1024];
memset(&ctx, 0, sizeof(ctx));
handler.send = rtsp_client_send;
handler.rtpport = rtpport;
handler.onannounce = onannounce;
handler.onsetup = onsetup;
handler.onrecord = onrecord;
handler.onteardown = onteardown;
handler.onrtp = onrtp;
ctx.status = 0;
ctx.peer = std::string(host);
ctx.mode = RTSP_TRANSPORT_RTP_UDP; // RTSP_TRANSPORT_RTP_TCP
snprintf(packet, sizeof(packet), "rtsp://%s/live/push", host); // url
socket_init();
ctx.socket = socket_connect_host(host, 554, 2000);
assert(socket_invalid != ctx.socket);
//ctx.rtsp = rtsp_client_create(NULL, NULL, &handler, &ctx);
ctx.rtsp = rtsp_client_create(packet, "username1", "password1", &handler, &ctx);
assert(ctx.rtsp);
assert(rtsp_client_sdp(&ctx, file) > 0);
assert(0 == rtsp_client_announce(ctx.rtsp, ctx.sdp));
socket_setnonblock(ctx.socket, 0);
r = socket_recv(ctx.socket, packet, sizeof(packet), 0);
while (r > 0)
{
assert(0 == rtsp_client_input(ctx.rtsp, packet, r));
if (ctx.status == 1)
break;
r = socket_recv(ctx.socket, packet, sizeof(packet), 0);
}
while (r > 0 && ctx.status == 1)
{
system_sleep(5);
if (1 == ctx.status)
ctx.source->Play();
// TODO: check rtsp session activity
}
assert(0 == rtsp_client_teardown(ctx.rtsp));
rtsp_client_destroy(ctx.rtsp);
socket_close(ctx.socket);
socket_cleanup();
}
#endif