2024-01-18 04:18:07 +00:00
|
|
|
use std::time::Duration;
|
2023-06-12 23:46:39 +00:00
|
|
|
|
2022-10-18 05:24:48 +00:00
|
|
|
use futures::StreamExt;
|
2024-01-10 00:14:54 +00:00
|
|
|
use gpui::{actions, KeyBinding, Menu, MenuItem};
|
2024-01-10 21:11:59 +00:00
|
|
|
use live_kit_client::{LocalAudioTrack, LocalVideoTrack, Room, RoomUpdate};
|
2022-10-19 17:20:31 +00:00
|
|
|
use live_kit_server::token::{self, VideoGrant};
|
2022-10-18 05:24:48 +00:00
|
|
|
use log::LevelFilter;
|
|
|
|
use simplelog::SimpleLogger;
|
|
|
|
|
2024-01-03 20:49:33 +00:00
|
|
|
actions!(live_kit_client, [Quit]);
|
2022-10-18 05:24:48 +00:00
|
|
|
|
|
|
|
fn main() {
|
|
|
|
SimpleLogger::init(LevelFilter::Info, Default::default()).expect("could not initialize logger");
|
|
|
|
|
2024-01-18 04:18:07 +00:00
|
|
|
gpui::App::new().run(|cx| {
|
2023-06-12 21:02:36 +00:00
|
|
|
#[cfg(any(test, feature = "test-support"))]
|
|
|
|
println!("USING TEST LIVEKIT");
|
|
|
|
|
|
|
|
#[cfg(not(any(test, feature = "test-support")))]
|
|
|
|
println!("USING REAL LIVEKIT");
|
|
|
|
|
2024-01-03 20:49:33 +00:00
|
|
|
cx.activate(true);
|
2022-10-18 05:24:48 +00:00
|
|
|
|
2024-01-03 20:49:33 +00:00
|
|
|
cx.on_action(quit);
|
|
|
|
cx.bind_keys([KeyBinding::new("cmd-q", Quit, None)]);
|
|
|
|
|
2024-01-10 00:14:54 +00:00
|
|
|
cx.set_menus(vec![Menu {
|
|
|
|
name: "Zed",
|
|
|
|
items: vec![MenuItem::Action {
|
|
|
|
name: "Quit",
|
|
|
|
action: Box::new(Quit),
|
|
|
|
os_action: None,
|
|
|
|
}],
|
|
|
|
}]);
|
2022-10-18 05:24:48 +00:00
|
|
|
|
|
|
|
let live_kit_url = std::env::var("LIVE_KIT_URL").unwrap_or("http://localhost:7880".into());
|
|
|
|
let live_kit_key = std::env::var("LIVE_KIT_KEY").unwrap_or("devkey".into());
|
|
|
|
let live_kit_secret = std::env::var("LIVE_KIT_SECRET").unwrap_or("secret".into());
|
|
|
|
|
|
|
|
cx.spawn(|cx| async move {
|
|
|
|
let user_a_token = token::create(
|
|
|
|
&live_kit_key,
|
|
|
|
&live_kit_secret,
|
|
|
|
Some("test-participant-1"),
|
|
|
|
VideoGrant::to_join("test-room"),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let room_a = Room::new();
|
|
|
|
room_a.connect(&live_kit_url, &user_a_token).await.unwrap();
|
|
|
|
|
|
|
|
let user2_token = token::create(
|
|
|
|
&live_kit_key,
|
|
|
|
&live_kit_secret,
|
|
|
|
Some("test-participant-2"),
|
|
|
|
VideoGrant::to_join("test-room"),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let room_b = Room::new();
|
|
|
|
room_b.connect(&live_kit_url, &user2_token).await.unwrap();
|
|
|
|
|
2024-01-10 21:11:59 +00:00
|
|
|
let mut room_updates = room_b.updates();
|
2023-06-12 21:02:36 +00:00
|
|
|
let audio_track = LocalAudioTrack::create();
|
2023-10-26 14:40:44 +00:00
|
|
|
let audio_track_publication = room_a.publish_audio_track(audio_track).await.unwrap();
|
2023-06-12 21:02:36 +00:00
|
|
|
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::SubscribedToRemoteAudioTrack(track, _) =
|
|
|
|
room_updates.next().await.unwrap()
|
2023-06-16 02:28:59 +00:00
|
|
|
{
|
2023-06-12 21:02:36 +00:00
|
|
|
let remote_tracks = room_b.remote_audio_tracks("test-participant-1");
|
|
|
|
assert_eq!(remote_tracks.len(), 1);
|
|
|
|
assert_eq!(remote_tracks[0].publisher_id(), "test-participant-1");
|
|
|
|
assert_eq!(track.publisher_id(), "test-participant-1");
|
|
|
|
} else {
|
|
|
|
panic!("unexpected message");
|
|
|
|
}
|
|
|
|
|
2023-06-28 02:19:08 +00:00
|
|
|
audio_track_publication.set_mute(true).await.unwrap();
|
|
|
|
|
|
|
|
println!("waiting for mute changed!");
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::RemoteAudioTrackMuteChanged { track_id, muted } =
|
|
|
|
room_updates.next().await.unwrap()
|
2023-06-28 02:19:08 +00:00
|
|
|
{
|
|
|
|
let remote_tracks = room_b.remote_audio_tracks("test-participant-1");
|
|
|
|
assert_eq!(remote_tracks[0].sid(), track_id);
|
|
|
|
assert_eq!(muted, true);
|
|
|
|
} else {
|
|
|
|
panic!("unexpected message");
|
|
|
|
}
|
|
|
|
|
|
|
|
audio_track_publication.set_mute(false).await.unwrap();
|
|
|
|
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::RemoteAudioTrackMuteChanged { track_id, muted } =
|
|
|
|
room_updates.next().await.unwrap()
|
2023-06-28 02:19:08 +00:00
|
|
|
{
|
|
|
|
let remote_tracks = room_b.remote_audio_tracks("test-participant-1");
|
|
|
|
assert_eq!(remote_tracks[0].sid(), track_id);
|
|
|
|
assert_eq!(muted, false);
|
|
|
|
} else {
|
|
|
|
panic!("unexpected message");
|
|
|
|
}
|
|
|
|
|
2023-06-15 01:40:21 +00:00
|
|
|
println!("Pausing for 5 seconds to test audio, make some noise!");
|
2024-01-03 20:49:33 +00:00
|
|
|
let timer = cx.background_executor().timer(Duration::from_secs(5));
|
2023-06-12 23:46:39 +00:00
|
|
|
timer.await;
|
2023-06-12 21:02:36 +00:00
|
|
|
let remote_audio_track = room_b
|
|
|
|
.remote_audio_tracks("test-participant-1")
|
|
|
|
.pop()
|
|
|
|
.unwrap();
|
|
|
|
room_a.unpublish_track(audio_track_publication);
|
2023-06-28 02:19:08 +00:00
|
|
|
|
|
|
|
// Clear out any active speakers changed messages
|
2024-01-10 21:11:59 +00:00
|
|
|
let mut next = room_updates.next().await.unwrap();
|
|
|
|
while let RoomUpdate::ActiveSpeakersChanged { speakers } = next {
|
2023-06-28 02:19:08 +00:00
|
|
|
println!("Speakers changed: {:?}", speakers);
|
2024-01-10 21:11:59 +00:00
|
|
|
next = room_updates.next().await.unwrap();
|
2023-06-28 02:19:08 +00:00
|
|
|
}
|
|
|
|
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::UnsubscribedFromRemoteAudioTrack {
|
2023-06-12 21:02:36 +00:00
|
|
|
publisher_id,
|
|
|
|
track_id,
|
2023-06-28 02:19:08 +00:00
|
|
|
} = next
|
2023-06-12 21:02:36 +00:00
|
|
|
{
|
|
|
|
assert_eq!(publisher_id, "test-participant-1");
|
|
|
|
assert_eq!(remote_audio_track.sid(), track_id);
|
|
|
|
assert_eq!(room_b.remote_audio_tracks("test-participant-1").len(), 0);
|
|
|
|
} else {
|
|
|
|
panic!("unexpected message");
|
|
|
|
}
|
2022-10-18 05:24:48 +00:00
|
|
|
|
2022-10-19 17:20:31 +00:00
|
|
|
let displays = room_a.display_sources().await.unwrap();
|
2022-10-18 05:38:43 +00:00
|
|
|
let display = displays.into_iter().next().unwrap();
|
2022-10-18 05:24:48 +00:00
|
|
|
|
2023-06-12 21:02:36 +00:00
|
|
|
let local_video_track = LocalVideoTrack::screen_share_for_display(&display);
|
2023-10-26 14:40:44 +00:00
|
|
|
let local_video_track_publication =
|
|
|
|
room_a.publish_video_track(local_video_track).await.unwrap();
|
2022-10-18 05:24:48 +00:00
|
|
|
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::SubscribedToRemoteVideoTrack(track) =
|
|
|
|
room_updates.next().await.unwrap()
|
2023-06-16 02:28:59 +00:00
|
|
|
{
|
2023-06-12 21:02:36 +00:00
|
|
|
let remote_video_tracks = room_b.remote_video_tracks("test-participant-1");
|
|
|
|
assert_eq!(remote_video_tracks.len(), 1);
|
|
|
|
assert_eq!(remote_video_tracks[0].publisher_id(), "test-participant-1");
|
2022-10-18 05:38:43 +00:00
|
|
|
assert_eq!(track.publisher_id(), "test-participant-1");
|
2022-10-18 05:24:48 +00:00
|
|
|
} else {
|
2022-10-19 08:29:03 +00:00
|
|
|
panic!("unexpected message");
|
|
|
|
}
|
|
|
|
|
2023-06-12 21:02:36 +00:00
|
|
|
let remote_video_track = room_b
|
2022-10-19 08:29:03 +00:00
|
|
|
.remote_video_tracks("test-participant-1")
|
|
|
|
.pop()
|
|
|
|
.unwrap();
|
2023-06-12 21:02:36 +00:00
|
|
|
room_a.unpublish_track(local_video_track_publication);
|
2024-01-10 21:11:59 +00:00
|
|
|
if let RoomUpdate::UnsubscribedFromRemoteVideoTrack {
|
2022-10-19 08:29:03 +00:00
|
|
|
publisher_id,
|
|
|
|
track_id,
|
2024-01-10 21:11:59 +00:00
|
|
|
} = room_updates.next().await.unwrap()
|
2022-10-19 08:29:03 +00:00
|
|
|
{
|
|
|
|
assert_eq!(publisher_id, "test-participant-1");
|
2023-06-12 21:02:36 +00:00
|
|
|
assert_eq!(remote_video_track.sid(), track_id);
|
2022-10-19 08:29:03 +00:00
|
|
|
assert_eq!(room_b.remote_video_tracks("test-participant-1").len(), 0);
|
|
|
|
} else {
|
|
|
|
panic!("unexpected message");
|
2022-10-18 05:24:48 +00:00
|
|
|
}
|
|
|
|
|
2024-01-03 20:49:33 +00:00
|
|
|
cx.update(|cx| cx.shutdown()).ok();
|
2022-10-18 05:24:48 +00:00
|
|
|
})
|
|
|
|
.detach();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-04-06 21:49:03 +00:00
|
|
|
fn quit(_: &Quit, cx: &mut gpui::AppContext) {
|
2024-01-03 20:49:33 +00:00
|
|
|
cx.quit();
|
2022-10-18 05:24:48 +00:00
|
|
|
}
|