capi_track.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. /**
  2. * Copyright (c) 2020 Paul-Louis Ageneau
  3. *
  4. * This Source Code Form is subject to the terms of the Mozilla Public
  5. * License, v. 2.0. If a copy of the MPL was not distributed with this
  6. * file, You can obtain one at https://mozilla.org/MPL/2.0/.
  7. */
  8. #include "test.hpp"
  9. #include <rtc/rtc.h>
  10. #include <cstdio>
  11. #include <cstdlib>
  12. #include <cstring>
  13. #ifdef _WIN32
  14. #include <windows.h>
  15. static void sleep(unsigned int secs) { Sleep(secs * 1000); }
  16. #else
  17. #include <unistd.h> // for sleep
  18. #endif
  19. #define BUFFER_SIZE 4096
  20. typedef struct {
  21. rtcState state;
  22. rtcGatheringState gatheringState;
  23. int pc;
  24. int tr;
  25. bool connected;
  26. } Peer;
  27. static Peer *peer1 = NULL;
  28. static Peer *peer2 = NULL;
  29. static const char *mediaDescription = "video 9 UDP/TLS/RTP/SAVPF\r\n"
  30. "a=mid:video\r\n"
  31. "a=sendonly\r\n";
  32. static void RTC_API descriptionCallback(int pc, const char *sdp, const char *type, void *ptr) {
  33. Peer *peer = (Peer *)ptr;
  34. printf("Description %d:\n%s\n", peer == peer1 ? 1 : 2, sdp);
  35. Peer *other = peer == peer1 ? peer2 : peer1;
  36. rtcSetRemoteDescription(other->pc, sdp, type);
  37. }
  38. static void RTC_API candidateCallback(int pc, const char *cand, const char *mid, void *ptr) {
  39. Peer *peer = (Peer *)ptr;
  40. printf("Candidate %d: %s\n", peer == peer1 ? 1 : 2, cand);
  41. Peer *other = peer == peer1 ? peer2 : peer1;
  42. rtcAddRemoteCandidate(other->pc, cand, mid);
  43. }
  44. static void RTC_API stateChangeCallback(int pc, rtcState state, void *ptr) {
  45. Peer *peer = (Peer *)ptr;
  46. peer->state = state;
  47. printf("State %d: %d\n", peer == peer1 ? 1 : 2, (int)state);
  48. }
  49. static void RTC_API gatheringStateCallback(int pc, rtcGatheringState state, void *ptr) {
  50. Peer *peer = (Peer *)ptr;
  51. peer->gatheringState = state;
  52. printf("Gathering state %d: %d\n", peer == peer1 ? 1 : 2, (int)state);
  53. }
  54. static void RTC_API openCallback(int id, void *ptr) {
  55. Peer *peer = (Peer *)ptr;
  56. peer->connected = true;
  57. printf("Track %d: Open\n", peer == peer1 ? 1 : 2);
  58. }
  59. static void RTC_API closedCallback(int id, void *ptr) {
  60. Peer *peer = (Peer *)ptr;
  61. peer->connected = false;
  62. printf("Track %d: Closed\n", peer == peer1 ? 1 : 2);
  63. }
  64. static void RTC_API trackCallback(int pc, int tr, void *ptr) {
  65. Peer *peer = (Peer *)ptr;
  66. char buffer[1024];
  67. if (rtcGetTrackDescription(tr, buffer, 1024) < 0) {
  68. fprintf(stderr, "rtcGetTrackDescription failed\n");
  69. return;
  70. }
  71. printf("Track %d: Received with media description: \n%s\n", peer == peer1 ? 1 : 2, buffer);
  72. char mid[256];
  73. if (rtcGetTrackMid(tr, mid, 256) < 0 || strcmp(mid, "video") != 0) {
  74. fprintf(stderr, "rtcGetTrackMid failed\n");
  75. return;
  76. }
  77. // Description is reversed here
  78. rtcDirection direction;
  79. if (rtcGetTrackDirection(tr, &direction) < 0 || direction != RTC_DIRECTION_RECVONLY) {
  80. fprintf(stderr, "rtcGetTrackDirection failed\n");
  81. return;
  82. }
  83. peer->tr = tr;
  84. rtcSetOpenCallback(tr, openCallback);
  85. rtcSetClosedCallback(tr, closedCallback);
  86. }
  87. static Peer *createPeer(const rtcConfiguration *config) {
  88. Peer *peer = (Peer *)malloc(sizeof(Peer));
  89. if (!peer)
  90. return nullptr;
  91. memset(peer, 0, sizeof(Peer));
  92. // Create peer connection
  93. peer->pc = rtcCreatePeerConnection(config);
  94. if (peer->pc < 0) {
  95. fprintf(stderr, "PeerConnection creation failed\n");
  96. free(peer);
  97. return nullptr;
  98. }
  99. rtcSetUserPointer(peer->pc, peer);
  100. rtcSetTrackCallback(peer->pc, trackCallback);
  101. rtcSetLocalDescriptionCallback(peer->pc, descriptionCallback);
  102. rtcSetLocalCandidateCallback(peer->pc, candidateCallback);
  103. rtcSetStateChangeCallback(peer->pc, stateChangeCallback);
  104. rtcSetGatheringStateChangeCallback(peer->pc, gatheringStateCallback);
  105. peer->tr = -1;
  106. return peer;
  107. }
  108. static void deletePeer(Peer *peer) {
  109. if (peer) {
  110. if (peer->tr >= 0)
  111. rtcDeleteTrack(peer->tr);
  112. if (peer->pc >= 0)
  113. rtcDeletePeerConnection(peer->pc);
  114. free(peer);
  115. }
  116. }
  117. int test_capi_track_main() {
  118. int attempts;
  119. rtcInitLogger(RTC_LOG_DEBUG, nullptr);
  120. // Create peer 1
  121. rtcConfiguration config1;
  122. memset(&config1, 0, sizeof(config1));
  123. // STUN server example
  124. // const char *iceServers[1] = {"stun:stun.l.google.com:19302"};
  125. // config1.iceServers = iceServers;
  126. // config1.iceServersCount = 1;
  127. peer1 = createPeer(&config1);
  128. if (!peer1)
  129. goto error;
  130. // Create peer 2
  131. rtcConfiguration config2;
  132. memset(&config2, 0, sizeof(config2));
  133. // STUN server example
  134. // config2.iceServers = iceServers;
  135. // config2.iceServersCount = 1;
  136. // Port range example
  137. config2.portRangeBegin = 5000;
  138. config2.portRangeEnd = 6000;
  139. peer2 = createPeer(&config2);
  140. if (!peer2)
  141. goto error;
  142. // Peer 1: Create track
  143. peer1->tr = rtcAddTrack(peer1->pc, mediaDescription);
  144. rtcSetOpenCallback(peer1->tr, openCallback);
  145. rtcSetClosedCallback(peer1->tr, closedCallback);
  146. char mid[256];
  147. if (rtcGetTrackMid(peer1->tr, mid, 256) < 0 || strcmp(mid, "video") != 0) {
  148. fprintf(stderr, "rtcGetTrackMid failed\n");
  149. goto error;
  150. }
  151. rtcDirection direction;
  152. if (rtcGetTrackDirection(peer1->tr, &direction) < 0 || direction != RTC_DIRECTION_SENDONLY) {
  153. fprintf(stderr, "rtcGetTrackDirection failed\n");
  154. goto error;
  155. }
  156. // Test createOffer
  157. char buffer[BUFFER_SIZE];
  158. if (rtcCreateOffer(peer1->pc, buffer, BUFFER_SIZE) < 0) {
  159. fprintf(stderr, "rtcCreateOffer failed\n");
  160. goto error;
  161. }
  162. if (rtcGetLocalDescription(peer1->pc, buffer, BUFFER_SIZE) >= 0) {
  163. fprintf(stderr, "rtcCreateOffer has set the local description\n");
  164. goto error;
  165. }
  166. // Initiate the handshake
  167. rtcSetLocalDescription(peer1->pc, NULL);
  168. if (rtcGetLocalDescription(peer1->pc, buffer, BUFFER_SIZE) < 0) {
  169. fprintf(stderr, "rtcGetLocalDescription failed\n");
  170. goto error;
  171. }
  172. attempts = 10;
  173. while ((!peer2->connected || !peer1->connected) && attempts--)
  174. sleep(1);
  175. if (peer1->state != RTC_CONNECTED || peer2->state != RTC_CONNECTED) {
  176. fprintf(stderr, "PeerConnection is not connected\n");
  177. goto error;
  178. }
  179. if (!peer1->connected || !peer2->connected) {
  180. fprintf(stderr, "Track is not connected\n");
  181. goto error;
  182. }
  183. deletePeer(peer1);
  184. sleep(1);
  185. deletePeer(peer2);
  186. sleep(1);
  187. printf("Success\n");
  188. return 0;
  189. error:
  190. deletePeer(peer1);
  191. deletePeer(peer2);
  192. return -1;
  193. }
  194. #include <stdexcept>
  195. TestResult test_capi_track() {
  196. if (test_capi_track_main())
  197. return TestResult(false, "Connection failed");
  198. return TestResult(true);
  199. }