ConsultationLiveVideoProviderV2.cs 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using Vinno.FIS.TRTCClient.Common.Enum;
  5. using Vinno.FIS.TRTCClient.Common.Models;
  6. using Vinno.IUS.Common.Log;
  7. using Vinno.vCloud.Common.FIS.Helper;
  8. using Vinno.vCloud.FIS.CrossPlatform.Common;
  9. using Vinno.vCloud.FIS.CrossPlatform.Common.Consultation;
  10. using Vinno.vCloud.FIS.CrossPlatform.Common.Consultation.Interface;
  11. using Vinno.vCloud.FIS.CrossPlatform.Common.Enum;
  12. using Vinno.vCloud.FIS.CrossPlatform.Common.LiveVideo;
  13. using Vinno.vCloud.Protocol.Infrastructures;
  14. using WingInterfaceLibrary.Interface;
  15. using WingInterfaceLibrary.LiveConsultation;
  16. using WingInterfaceLibrary.Request;
  17. namespace Vinno.vCloud.Common.FIS.Consultation
  18. {
  19. internal class ConsultationLiveVideoProviderV2 : IDisposable
  20. {
  21. private readonly object _lock = new object();
  22. private readonly object _startConsultationLocker = new object();//Felix之所以加这个lock,会有Start没跑完就Stop的情况。
  23. private readonly List<VideoProviderV2> _videoProviders;
  24. private IRtcRoom _rtcRoom;
  25. private bool _isConsulting;
  26. private bool _isOnlyRtmpPushing;
  27. private LiveProtocol _liveProtocol;
  28. private string _accountId;
  29. private ConsultationInfo _consultationInfo;
  30. private int _imagePreviewErrorLogCount;
  31. private ILiveConsultationService _liveConsultationService;
  32. private IOrganizationService _organizationService;
  33. public event EventHandler<EnumTRTCRoomError> OnTRTCRoomEnterError;
  34. /// <summary>
  35. /// 本地的摄像头数据
  36. /// </summary>
  37. public event EventHandler<ConsultationVideoFrameData> ConsultationLocalVideoFrameArrived;
  38. /// <summary>
  39. /// 其余所有用户的摄像头数据
  40. /// </summary>
  41. public event EventHandler<ConsultationVideoFrameData> ConsultationRemoteVideoFrameArrived;
  42. /// <summary>
  43. /// 超声机摄像头画面
  44. /// </summary>
  45. public event EventHandler<CPVideoFrameData> DeviceCameraFrameArrived;
  46. public event EventHandler<RemoteUserLeaveRoomArgs> RemoteUserLeaveRoomArrived;
  47. public event EventHandler CloseConsultation;
  48. public event EventHandler TryToReconnect;
  49. public ConsultationLiveVideoProviderV2(ILiveConsultationService liveConsultationService, IOrganizationService organizationService)
  50. {
  51. _liveConsultationService = liveConsultationService;
  52. _organizationService = organizationService;
  53. _videoProviders = new List<VideoProviderV2>();
  54. ScanCameraCacheHelper.Instance.ScanCameraVideoFrameDataArrived += OnTerminalCameraFrameArrived;
  55. }
  56. public void ChangeConsultationCodeForConsultationConnectionKeeper(string consultationCode)
  57. {
  58. if (string.IsNullOrEmpty(consultationCode))
  59. {
  60. return;
  61. }
  62. var existVideoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == _accountId);
  63. if (existVideoProvider != null)
  64. {
  65. existVideoProvider.ChangeConsultationCode(consultationCode);
  66. }
  67. }
  68. /// <summary>
  69. /// 开始会诊
  70. /// </summary>
  71. /// <param name="consultationInfo"></param>
  72. public void StartConsultationVideo(ConsultationInfo consultationInfo, string accountId, string accountName, string accountToken, string videoDeviceId, string micDeviceId, string speakerDeviceId, bool isOnlyRtmpPushing)
  73. {
  74. if (_isConsulting)
  75. {
  76. return;
  77. }
  78. lock (_startConsultationLocker)
  79. {
  80. _isConsulting = true;
  81. _isOnlyRtmpPushing = isOnlyRtmpPushing;
  82. _liveProtocol = consultationInfo.LiveProtocol;
  83. _accountId = accountId;
  84. _consultationInfo = consultationInfo;
  85. var terminalInfo = consultationInfo.TerminalInfo;
  86. var consultationMembers = consultationInfo.ConsultationMemberInfos;
  87. if (consultationInfo.LiveTalkingMode == Protocol.Messages.Client.LiveTalking.LiveTalkingMode.Speech)
  88. {
  89. videoDeviceId = string.Empty;
  90. }
  91. if (_liveProtocol == LiveProtocol.RTC)
  92. {
  93. if (consultationInfo.State == LiveStates.OK)
  94. {
  95. foreach (var member in consultationMembers)
  96. {
  97. if (member.Id == _accountId)
  98. {
  99. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == member.Id);
  100. if (videoProvider == null)
  101. {
  102. videoProvider = new RtcVideoProviderV2(member);
  103. Logger.WriteLineInfo($"Start To Create RTC Video Provider, Name:{member.Name},Id:{member.Id}");
  104. _videoProviders.Add(videoProvider);
  105. }
  106. else
  107. {
  108. Logger.WriteLineInfo($"RTC Video Provider Already exist, Name:{member.Name},Id:{member.Id}");
  109. videoProvider.ConsultationMemberInfo = member;
  110. }
  111. var userSign = member.UserSign;
  112. var integerRoomId = consultationInfo.IntegerRoomId;
  113. var appId = consultationInfo.AppId;
  114. var mode = consultationInfo.LiveTalkingMode;
  115. var roomInfo = new TRTCRoomInfo
  116. {
  117. AppId = (uint)appId,
  118. UserSig = userSign,
  119. UserId = accountId,
  120. RoomId = (uint)integerRoomId,
  121. VideoDeviceId = videoDeviceId,
  122. MicDeviceId = micDeviceId,
  123. SpeakerDeviceId = speakerDeviceId,
  124. TerminalIsPushing = _consultationInfo.TerminalInfo?.State == LiveStates.OK,
  125. TerminalId = _consultationInfo.TerminalInfo?.Id,
  126. CameraId = _consultationInfo.TerminalInfo?.CameraId,
  127. IsMultiChannel = _consultationInfo.TerminalInfo == null ? false : _consultationInfo.TerminalInfo.IsMultiChannels,
  128. TerminalRoomId = (uint)_consultationInfo.TerminalInfo?.TerminalIntegerRoomId,
  129. IsLiveMode = false,
  130. IsOldServerMode = false,
  131. Category = EnumLiveChannelCategory.Main,
  132. };
  133. if (LiveVideoStatusChecker.Instance.IsConsultationInLiveVideo)
  134. {
  135. LiveVideoStatusChecker.Instance.TRTCRoomInfo = roomInfo;
  136. LiveVideoStatusChecker.Instance.RemoteVideoFrameDataEvent += OnRtcRemoteVideoFrameArrived;
  137. LiveVideoStatusChecker.Instance.IsConsultationLivingInCurrentMachine = true;
  138. }
  139. else
  140. {
  141. _rtcRoom = CrossPlatformHelper.Instance.RtcRoomCreator.CreateRtcRoom();
  142. _rtcRoom.LocalVideoFrameArrived += OnRtcLocalVideoFrameArrived;
  143. _rtcRoom.RemoteVideoFrameArrived += OnRtcRemoteVideoFrameArrived;
  144. _rtcRoom.OnTRTCRoomEnterError += OnTRTCRoomEnterErrorHappened;
  145. _rtcRoom.RemoteUserLeaveRoomArrived += OnRemoteUserLeaveRoomArrived;
  146. _rtcRoom.TryToReconnect += OnTryToReconnect;
  147. _rtcRoom.Enter(roomInfo);
  148. Logger.WriteLineInfo($"{accountName} enterred room {integerRoomId}");
  149. }
  150. if (member.State == LiveStates.OK)
  151. {
  152. var tokenRequest = new TokenRequest
  153. {
  154. Token = accountToken,
  155. };
  156. int interval = 10;
  157. var serverSettingResult = JsonRpcHelper.GetServerSetting(_organizationService, tokenRequest);
  158. if (serverSettingResult == null)
  159. {
  160. Logger.WriteLineError($"GetServerSetting Error:result is null");
  161. }
  162. else
  163. {
  164. interval = serverSettingResult.LiveConsultationRateSeconds;
  165. }
  166. videoProvider.StartConsultationConnectionKeeper(accountToken, consultationInfo.ConsultationId, interval, _liveConsultationService);
  167. videoProvider.Offlined += OnOfflined;
  168. }
  169. }
  170. else
  171. {
  172. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == member.Id);
  173. if (videoProvider == null)
  174. {
  175. Logger.WriteLineInfo($"Start To Create RTC Video Provider, Name:{member.Name},Id:{member.Id}");
  176. _videoProviders.Add(new RtcVideoProviderV2(member));
  177. }
  178. else
  179. {
  180. Logger.WriteLineInfo($"RTC Video Provider Already exist, Name:{member.Name},Id:{member.Id}");
  181. videoProvider.ConsultationMemberInfo = member;
  182. }
  183. }
  184. }
  185. }
  186. }
  187. else if (_liveProtocol == LiveProtocol.Rtmp)
  188. {
  189. foreach (var member in consultationMembers)
  190. {
  191. if (member.Id == _accountId)
  192. {
  193. try
  194. {
  195. var pusherUrl = member.PushUrl;
  196. VideoProviderV2 videoProvider;
  197. if (!string.IsNullOrEmpty(pusherUrl))
  198. {
  199. Logger.WriteLineInfo($"Start push with rtmp address {pusherUrl}");
  200. videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == member.Id);
  201. if (videoProvider == null)
  202. {
  203. Logger.WriteLineInfo($"Start To Create Rtmp Video Pusher, Name:{member.Name},Id:{member.Id}");
  204. if (LiveVideoStatusChecker.Instance.IsConsultationInLiveVideo)
  205. {
  206. LiveVideoStatusChecker.Instance.IsConsultationLivingInCurrentMachine = true;
  207. videoProvider = new RtmpVideoProviderV2(member);
  208. }
  209. else
  210. {
  211. videoProvider = new RtmpVideoPusherV2(member, videoDeviceId, micDeviceId, _isOnlyRtmpPushing);
  212. if (!_isOnlyRtmpPushing)
  213. {
  214. videoProvider.VideoFrameArrived += OnLocalVideoFrameArrived;
  215. }
  216. }
  217. _videoProviders.Add(videoProvider);
  218. }
  219. else
  220. {
  221. Logger.WriteLineInfo($"Rtmp video Pusher already exists,Name:{member.Name},Id:{member.Id}");
  222. videoProvider.ConsultationMemberInfo = member;
  223. }
  224. if (member.State == LiveStates.OK)
  225. {
  226. var tokenRequest = new TokenRequest
  227. {
  228. Token = accountToken,
  229. };
  230. int interval = 10;
  231. var serverSettingResult = JsonRpcHelper.GetServerSetting(_organizationService, tokenRequest);
  232. if (serverSettingResult == null)
  233. {
  234. Logger.WriteLineError($"GetServerSetting Error:result is null");
  235. }
  236. else
  237. {
  238. interval = serverSettingResult.LiveConsultationRateSeconds;
  239. }
  240. videoProvider.StartConsultationConnectionKeeper(accountToken, consultationInfo.ConsultationId, interval, _liveConsultationService);
  241. videoProvider.Offlined += OnOfflined;
  242. }
  243. }
  244. }
  245. catch (Exception ex)
  246. {
  247. Logger.WriteLineError($"Exception happened while play rtmp{ex}");
  248. }
  249. }
  250. else if (!_isOnlyRtmpPushing)
  251. {
  252. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == member.Id);
  253. if (videoProvider == null)
  254. {
  255. Logger.WriteLineInfo($"Start To Create Rtmp Player,Name:{member.Name},Id:{member.Id}");
  256. CreateRtmpPlayer(member);
  257. }
  258. else
  259. {
  260. Logger.WriteLineInfo($"Rtmp Player already exists,Name:{member.Name},Id:{member.Id}");
  261. videoProvider.ConsultationMemberInfo = member;
  262. }
  263. }
  264. }
  265. }
  266. }
  267. }
  268. private void OnTerminalCameraFrameArrived(object sender, CPVideoFrameData e)
  269. {
  270. if (_isConsulting && _consultationInfo != null)
  271. {
  272. DeviceCameraFrameArrived?.Invoke(this, e);
  273. }
  274. }
  275. private void OnRtcLocalVideoFrameArrived(object sender, CPVideoFrameData e)
  276. {
  277. var consultationMembers = _consultationInfo.ConsultationMemberInfos;
  278. var consultationMember = consultationMembers.FirstOrDefault(x => x.Id == _accountId);
  279. if (consultationMember != null)
  280. {
  281. OnLocalVideoFrameArrived(this, new ConsultationVideoFrameData(e, consultationMember));
  282. }
  283. }
  284. private void OnLocalVideoFrameArrived(object sender, ConsultationVideoFrameData e)
  285. {
  286. ConsultationLocalVideoFrameArrived?.Invoke(this, e);
  287. }
  288. private void OnRtcRemoteVideoFrameArrived(object sender, CPRemoteVideoFrameData e)
  289. {
  290. try
  291. {
  292. if (_consultationInfo != null)
  293. {
  294. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == e.UserId);
  295. if (videoProvider != null)
  296. {
  297. OnVideoFrameArrived(videoProvider, e.Data);
  298. }
  299. _imagePreviewErrorLogCount = 0;
  300. }
  301. }
  302. catch (Exception ex)
  303. {
  304. if (_imagePreviewErrorLogCount++ < 1)
  305. {
  306. Logger.WriteLineError($"OnRtcRemoteVideoFrameArrived ex:{ex}");
  307. }
  308. }
  309. }
  310. private void OnVideoFrameArrived(VideoProviderV2 videoProvider, CPVideoFrameData e)
  311. {
  312. OnRemoteVideoFrameArrived(this, new ConsultationVideoFrameData(e, videoProvider.ConsultationMemberInfo));
  313. }
  314. private void OnTRTCRoomEnterErrorHappened(object sender, EnumTRTCRoomError e)
  315. {
  316. OnTRTCRoomEnterError?.Invoke(sender, e);
  317. }
  318. private void OnTryToReconnect(object sender, EventArgs e)
  319. {
  320. TryToReconnect?.Invoke(this, EventArgs.Empty);
  321. }
  322. private void OnOfflined(object sender, EventArgs e)
  323. {
  324. CloseConsultation?.Invoke(sender, e);
  325. }
  326. private void CreateRtmpPlayer(ConsultationMemberInfo consultationMember)
  327. {
  328. var rtmpVideoProvider = new RtmpVideoPlayerV2(consultationMember);
  329. rtmpVideoProvider.VideoFrameArrived += OnRemoteVideoFrameArrived;
  330. _videoProviders.Add(rtmpVideoProvider);
  331. }
  332. private void OnRemoteVideoFrameArrived(object sender, ConsultationVideoFrameData e)
  333. {
  334. ConsultationRemoteVideoFrameArrived?.Invoke(this, e);
  335. }
  336. private void OnRemoteUserLeaveRoomArrived(object sender, RemoteUserLeaveRoomArgs e)
  337. {
  338. RemoteUserLeaveRoomArrived?.Invoke(this, e);
  339. }
  340. /// <summary>
  341. /// 会诊人员发生变化
  342. /// </summary>
  343. public void ConsultationMemberChange(ConsultationMemberChangeDTO meetingChangeMember)
  344. {
  345. if (_liveProtocol == LiveProtocol.RTC)
  346. {
  347. var memberInfo = meetingChangeMember.MemberInfo;
  348. if (memberInfo.OperationType == ClientMessageOperationType.Add)
  349. {
  350. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == memberInfo.Id);
  351. if (videoProvider == null)
  352. {
  353. _videoProviders.Add(new RtcVideoProviderV2(memberInfo));
  354. }
  355. else
  356. {
  357. videoProvider.ConsultationMemberInfo = memberInfo;
  358. }
  359. }
  360. else if (memberInfo.OperationType == ClientMessageOperationType.Delete)
  361. {
  362. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == memberInfo.Id);
  363. if (videoProvider != null)
  364. {
  365. videoProvider.Offlined -= OnOfflined;
  366. videoProvider.Dispose();
  367. _videoProviders.Remove(videoProvider);
  368. }
  369. }
  370. }
  371. else if (_liveProtocol == LiveProtocol.Rtmp && !_isOnlyRtmpPushing)
  372. {
  373. var memberInfo = meetingChangeMember.MemberInfo;
  374. if (memberInfo.OperationType == ClientMessageOperationType.Add)
  375. {
  376. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == memberInfo.Id);
  377. if (videoProvider == null)
  378. {
  379. CreateRtmpPlayer(memberInfo);
  380. }
  381. else
  382. {
  383. videoProvider.ConsultationMemberInfo = memberInfo;
  384. }
  385. }
  386. else if (memberInfo.OperationType == ClientMessageOperationType.Delete)
  387. {
  388. var videoProvider = _videoProviders.FirstOrDefault(f => f.ConsultationMemberInfo.Id == memberInfo.Id);
  389. if (videoProvider != null)
  390. {
  391. videoProvider.Dispose();
  392. _videoProviders.Remove(videoProvider);
  393. }
  394. }
  395. }
  396. }
  397. public void RemoveVideoProvider(string userId)
  398. {
  399. lock (_startConsultationLocker)
  400. {
  401. if (_videoProviders.Count > 0)
  402. {
  403. var video = _videoProviders.FirstOrDefault(v => v.ConsultationMemberInfo.Id == userId);
  404. if (video != null)
  405. {
  406. video.Dispose();
  407. _videoProviders.Remove(video);
  408. }
  409. }
  410. }
  411. }
  412. /// <summary>
  413. /// 挂断
  414. /// </summary>
  415. public void Hangup()
  416. {
  417. if (!_isConsulting)
  418. {
  419. return;
  420. }
  421. lock (_startConsultationLocker)
  422. {
  423. try
  424. {
  425. if (_isConsulting)
  426. {
  427. if (_videoProviders.Count > 0)
  428. {
  429. for (int i = _videoProviders.Count - 1; i >= 0; i--)
  430. {
  431. var videoProvider = _videoProviders[i];
  432. UnregisterReference(videoProvider);
  433. videoProvider.Dispose();
  434. }
  435. _videoProviders.Clear();
  436. LiveVideoStatusChecker.Instance.RemoteVideoFrameDataEvent -= OnRtcRemoteVideoFrameArrived;
  437. LiveVideoStatusChecker.Instance.IsConsultationLivingInCurrentMachine = false;
  438. if (_liveProtocol == LiveProtocol.RTC)
  439. {
  440. if (_rtcRoom != null)
  441. {
  442. _rtcRoom.LocalVideoFrameArrived -= OnRtcLocalVideoFrameArrived;
  443. _rtcRoom.RemoteVideoFrameArrived -= OnRtcRemoteVideoFrameArrived;
  444. _rtcRoom.TryToReconnect -= OnTryToReconnect;
  445. _rtcRoom.RemoteUserLeaveRoomArrived -= OnRemoteUserLeaveRoomArrived;
  446. _rtcRoom.Exit();
  447. }
  448. }
  449. }
  450. }
  451. }
  452. catch (Exception ex)
  453. {
  454. Logger.WriteLineInfo($"Hangup failed ex: {ex}");
  455. }
  456. finally
  457. {
  458. _isConsulting = false;
  459. _isOnlyRtmpPushing = false;
  460. }
  461. }
  462. }
  463. private void UnregisterReference(VideoProviderV2 videoProvider)
  464. {
  465. if (videoProvider.ConsultationMemberInfo.Id == _accountId)
  466. {
  467. videoProvider.VideoFrameArrived -= OnLocalVideoFrameArrived;
  468. videoProvider.Offlined -= OnOfflined;
  469. }
  470. else
  471. {
  472. videoProvider.VideoFrameArrived -= OnRemoteVideoFrameArrived;
  473. videoProvider.Offlined -= OnOfflined;
  474. }
  475. }
  476. public void SwitchCamera()
  477. {
  478. lock (_lock)
  479. {
  480. if (_isConsulting && _consultationInfo != null)
  481. {
  482. if (_liveProtocol == LiveProtocol.RTC)
  483. {
  484. _rtcRoom?.SwitchCamera();
  485. }
  486. else
  487. {
  488. var rtmpPusher = (RtmpVideoPusherV2)_videoProviders.FirstOrDefault(x => x is RtmpVideoPusherV2);
  489. if (rtmpPusher != null)
  490. {
  491. rtmpPusher.SwitchCamera();
  492. }
  493. }
  494. }
  495. }
  496. }
  497. public void Mute(bool isMute)
  498. {
  499. lock (_lock)
  500. {
  501. if (_isConsulting && _consultationInfo != null)
  502. {
  503. if (_liveProtocol == LiveProtocol.RTC)
  504. {
  505. _rtcRoom?.Mute(isMute);
  506. }
  507. else
  508. {
  509. var rtmpPusher = (RtmpVideoPusherV2)_videoProviders.FirstOrDefault(x => x is RtmpVideoPusherV2);
  510. if (rtmpPusher != null)
  511. {
  512. rtmpPusher.SetMute(isMute);
  513. }
  514. }
  515. }
  516. }
  517. }
  518. public void Dispose()
  519. {
  520. ScanCameraCacheHelper.Instance.ScanCameraVideoFrameDataArrived -= OnTerminalCameraFrameArrived;
  521. Hangup();
  522. }
  523. internal void UpdateTerminalInfo(TerminalInfo terminalInfo)
  524. {
  525. if (_consultationInfo != null && _isConsulting)
  526. {
  527. _consultationInfo.TerminalInfo = terminalInfo;
  528. }
  529. if (_liveProtocol == LiveProtocol.RTC)
  530. {
  531. _rtcRoom?.UpdateCameraId(terminalInfo.CameraId);
  532. }
  533. }
  534. }
  535. }