Remove 2 suffix from live_kit_client
Co-authored-by: Mikayla <mikayla@zed.dev>
This commit is contained in:
parent
ce4bc994e7
commit
3c81dda8e2
21 changed files with 199 additions and 2725 deletions
37
Cargo.lock
generated
37
Cargo.lock
generated
|
@ -1114,7 +1114,7 @@ dependencies = [
|
||||||
"gpui2",
|
"gpui2",
|
||||||
"image",
|
"image",
|
||||||
"language",
|
"language",
|
||||||
"live_kit_client2",
|
"live_kit_client",
|
||||||
"log",
|
"log",
|
||||||
"media",
|
"media",
|
||||||
"postage",
|
"postage",
|
||||||
|
@ -1476,7 +1476,7 @@ dependencies = [
|
||||||
"language",
|
"language",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"lipsum",
|
"lipsum",
|
||||||
"live_kit_client2",
|
"live_kit_client",
|
||||||
"live_kit_server",
|
"live_kit_server",
|
||||||
"log",
|
"log",
|
||||||
"lsp",
|
"lsp",
|
||||||
|
@ -4168,39 +4168,6 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "live_kit_client"
|
name = "live_kit_client"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
|
||||||
"anyhow",
|
|
||||||
"async-broadcast",
|
|
||||||
"async-trait",
|
|
||||||
"block",
|
|
||||||
"byteorder",
|
|
||||||
"bytes 1.5.0",
|
|
||||||
"cocoa",
|
|
||||||
"collections",
|
|
||||||
"core-foundation",
|
|
||||||
"core-graphics",
|
|
||||||
"foreign-types",
|
|
||||||
"futures 0.3.28",
|
|
||||||
"gpui",
|
|
||||||
"hmac 0.12.1",
|
|
||||||
"jwt",
|
|
||||||
"live_kit_server",
|
|
||||||
"log",
|
|
||||||
"media",
|
|
||||||
"nanoid",
|
|
||||||
"objc",
|
|
||||||
"parking_lot 0.11.2",
|
|
||||||
"postage",
|
|
||||||
"serde",
|
|
||||||
"serde_derive",
|
|
||||||
"serde_json",
|
|
||||||
"sha2 0.10.7",
|
|
||||||
"simplelog",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "live_kit_client2"
|
|
||||||
version = "0.1.0"
|
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-broadcast",
|
"async-broadcast",
|
||||||
|
|
|
@ -24,7 +24,7 @@ client = { path = "../client" }
|
||||||
collections = { path = "../collections" }
|
collections = { path = "../collections" }
|
||||||
gpui = { package = "gpui2", path = "../gpui2" }
|
gpui = { package = "gpui2", path = "../gpui2" }
|
||||||
log.workspace = true
|
log.workspace = true
|
||||||
live_kit_client = { package = "live_kit_client2", path = "../live_kit_client2" }
|
live_kit_client = { path = "../live_kit_client" }
|
||||||
fs = { path = "../fs" }
|
fs = { path = "../fs" }
|
||||||
language = { path = "../language" }
|
language = { path = "../language" }
|
||||||
media = { path = "../media" }
|
media = { path = "../media" }
|
||||||
|
@ -49,6 +49,6 @@ fs = { path = "../fs", features = ["test-support"] }
|
||||||
language = { path = "../language", features = ["test-support"] }
|
language = { path = "../language", features = ["test-support"] }
|
||||||
collections = { path = "../collections", features = ["test-support"] }
|
collections = { path = "../collections", features = ["test-support"] }
|
||||||
gpui = { package = "gpui2", path = "../gpui2", features = ["test-support"] }
|
gpui = { package = "gpui2", path = "../gpui2", features = ["test-support"] }
|
||||||
live_kit_client = { package = "live_kit_client2", path = "../live_kit_client2", features = ["test-support"] }
|
live_kit_client = { path = "../live_kit_client", features = ["test-support"] }
|
||||||
project = { path = "../project", features = ["test-support"] }
|
project = { path = "../project", features = ["test-support"] }
|
||||||
util = { path = "../util", features = ["test-support"] }
|
util = { path = "../util", features = ["test-support"] }
|
||||||
|
|
|
@ -70,7 +70,7 @@ editor = { path = "../editor", features = ["test-support"] }
|
||||||
language = { path = "../language", features = ["test-support"] }
|
language = { path = "../language", features = ["test-support"] }
|
||||||
fs = { path = "../fs", features = ["test-support"] }
|
fs = { path = "../fs", features = ["test-support"] }
|
||||||
git = { path = "../git", features = ["test-support"] }
|
git = { path = "../git", features = ["test-support"] }
|
||||||
live_kit_client = { package = "live_kit_client2", path = "../live_kit_client2", features = ["test-support"] }
|
live_kit_client = { path = "../live_kit_client", features = ["test-support"] }
|
||||||
lsp = { path = "../lsp", features = ["test-support"] }
|
lsp = { path = "../lsp", features = ["test-support"] }
|
||||||
node_runtime = { path = "../node_runtime" }
|
node_runtime = { path = "../node_runtime" }
|
||||||
notifications = { path = "../notifications", features = ["test-support"] }
|
notifications = { path = "../notifications", features = ["test-support"] }
|
||||||
|
|
|
@ -1,99 +0,0 @@
|
||||||
[package]
|
|
||||||
authors = ["Nathan Sobo <nathan@zed.dev>"]
|
|
||||||
default-run = "collab"
|
|
||||||
edition = "2021"
|
|
||||||
name = "collab"
|
|
||||||
version = "0.28.0"
|
|
||||||
publish = false
|
|
||||||
|
|
||||||
[[bin]]
|
|
||||||
name = "collab"
|
|
||||||
|
|
||||||
[[bin]]
|
|
||||||
name = "seed"
|
|
||||||
required-features = ["seed-support"]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
clock = { path = "../clock" }
|
|
||||||
collections = { path = "../collections" }
|
|
||||||
live_kit_server = { path = "../live_kit_server" }
|
|
||||||
text = { path = "../text" }
|
|
||||||
rpc = { path = "../rpc" }
|
|
||||||
util = { path = "../util" }
|
|
||||||
|
|
||||||
anyhow.workspace = true
|
|
||||||
async-tungstenite = "0.16"
|
|
||||||
axum = { version = "0.5", features = ["json", "headers", "ws"] }
|
|
||||||
axum-extra = { version = "0.3", features = ["erased-json"] }
|
|
||||||
base64 = "0.13"
|
|
||||||
clap = { version = "3.1", features = ["derive"], optional = true }
|
|
||||||
dashmap = "5.4"
|
|
||||||
envy = "0.4.2"
|
|
||||||
futures.workspace = true
|
|
||||||
hyper = "0.14"
|
|
||||||
lazy_static.workspace = true
|
|
||||||
lipsum = { version = "0.8", optional = true }
|
|
||||||
log.workspace = true
|
|
||||||
nanoid = "0.4"
|
|
||||||
parking_lot.workspace = true
|
|
||||||
prometheus = "0.13"
|
|
||||||
prost.workspace = true
|
|
||||||
rand.workspace = true
|
|
||||||
reqwest = { version = "0.11", features = ["json"], optional = true }
|
|
||||||
scrypt = "0.7"
|
|
||||||
smallvec.workspace = true
|
|
||||||
sea-orm = { version = "0.12.x", features = ["sqlx-postgres", "postgres-array", "runtime-tokio-rustls", "with-uuid"] }
|
|
||||||
serde.workspace = true
|
|
||||||
serde_derive.workspace = true
|
|
||||||
serde_json.workspace = true
|
|
||||||
sha-1 = "0.9"
|
|
||||||
sqlx = { version = "0.7", features = ["runtime-tokio-rustls", "postgres", "json", "time", "uuid", "any"] }
|
|
||||||
time.workspace = true
|
|
||||||
tokio = { version = "1", features = ["full"] }
|
|
||||||
tokio-tungstenite = "0.17"
|
|
||||||
tonic = "0.6"
|
|
||||||
tower = "0.4"
|
|
||||||
toml.workspace = true
|
|
||||||
tracing = "0.1.34"
|
|
||||||
tracing-log = "0.1.3"
|
|
||||||
tracing-subscriber = { version = "0.3.11", features = ["env-filter", "json"] }
|
|
||||||
uuid.workspace = true
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
audio = { path = "../audio" }
|
|
||||||
collections = { path = "../collections", features = ["test-support"] }
|
|
||||||
gpui = { package = "gpui2", path = "../gpui2", features = ["test-support"] }
|
|
||||||
call = { path = "../call", features = ["test-support"] }
|
|
||||||
client = { path = "../client", features = ["test-support"] }
|
|
||||||
channel = { path = "../channel" }
|
|
||||||
editor = { path = "../editor", features = ["test-support"] }
|
|
||||||
language = { path = "../language", features = ["test-support"] }
|
|
||||||
fs = { path = "../fs", features = ["test-support"] }
|
|
||||||
git = { path = "../git", features = ["test-support"] }
|
|
||||||
live_kit_client = { package = "live_kit_client2", path = "../live_kit_client2", features = ["test-support"] }
|
|
||||||
lsp = { path = "../lsp", features = ["test-support"] }
|
|
||||||
node_runtime = { path = "../node_runtime" }
|
|
||||||
notifications = { path = "../notifications", features = ["test-support"] }
|
|
||||||
|
|
||||||
project = { path = "../project", features = ["test-support"] }
|
|
||||||
rpc = { path = "../rpc", features = ["test-support"] }
|
|
||||||
settings = { path = "../settings", features = ["test-support"] }
|
|
||||||
theme = { path = "../theme" }
|
|
||||||
workspace = { path = "../workspace", features = ["test-support"] }
|
|
||||||
|
|
||||||
collab_ui = { path = "../collab_ui", features = ["test-support"] }
|
|
||||||
|
|
||||||
async-trait.workspace = true
|
|
||||||
pretty_assertions.workspace = true
|
|
||||||
ctor.workspace = true
|
|
||||||
env_logger.workspace = true
|
|
||||||
indoc.workspace = true
|
|
||||||
util = { path = "../util" }
|
|
||||||
lazy_static.workspace = true
|
|
||||||
sea-orm = { version = "0.12.x", features = ["sqlx-sqlite"] }
|
|
||||||
serde_json.workspace = true
|
|
||||||
sqlx = { version = "0.7", features = ["sqlite"] }
|
|
||||||
unindent.workspace = true
|
|
||||||
|
|
||||||
[features]
|
|
||||||
seed-support = ["clap", "lipsum", "reqwest"]
|
|
|
@ -23,7 +23,7 @@ test-support = [
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
collections = { path = "../collections", optional = true }
|
collections = { path = "../collections", optional = true }
|
||||||
gpui = { path = "../gpui", optional = true }
|
gpui = { package = "gpui2", path = "../gpui2", optional = true }
|
||||||
live_kit_server = { path = "../live_kit_server", optional = true }
|
live_kit_server = { path = "../live_kit_server", optional = true }
|
||||||
media = { path = "../media" }
|
media = { path = "../media" }
|
||||||
|
|
||||||
|
@ -41,7 +41,7 @@ nanoid = { version ="0.4", optional = true}
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
collections = { path = "../collections", features = ["test-support"] }
|
collections = { path = "../collections", features = ["test-support"] }
|
||||||
gpui = { path = "../gpui", features = ["test-support"] }
|
gpui = { package = "gpui2", path = "../gpui2", features = ["test-support"] }
|
||||||
live_kit_server = { path = "../live_kit_server" }
|
live_kit_server = { path = "../live_kit_server" }
|
||||||
media = { path = "../media" }
|
media = { path = "../media" }
|
||||||
nanoid = "0.4"
|
nanoid = "0.4"
|
||||||
|
|
|
@ -42,8 +42,8 @@
|
||||||
"repositoryURL": "https://github.com/apple/swift-protobuf.git",
|
"repositoryURL": "https://github.com/apple/swift-protobuf.git",
|
||||||
"state": {
|
"state": {
|
||||||
"branch": null,
|
"branch": null,
|
||||||
"revision": "0af9125c4eae12a4973fb66574c53a54962a9e1e",
|
"revision": "ce20dc083ee485524b802669890291c0d8090170",
|
||||||
"version": "1.21.0"
|
"version": "1.22.1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
|
@ -61,12 +61,14 @@ fn build_bridge(swift_target: &SwiftTarget) {
|
||||||
|
|
||||||
let swift_package_root = swift_package_root();
|
let swift_package_root = swift_package_root();
|
||||||
let swift_target_folder = swift_target_folder();
|
let swift_target_folder = swift_target_folder();
|
||||||
|
let swift_cache_folder = swift_cache_folder();
|
||||||
if !Command::new("swift")
|
if !Command::new("swift")
|
||||||
.arg("build")
|
.arg("build")
|
||||||
.arg("--disable-automatic-resolution")
|
.arg("--disable-automatic-resolution")
|
||||||
.args(["--configuration", &env::var("PROFILE").unwrap()])
|
.args(["--configuration", &env::var("PROFILE").unwrap()])
|
||||||
.args(["--triple", &swift_target.target.triple])
|
.args(["--triple", &swift_target.target.triple])
|
||||||
.args(["--build-path".into(), swift_target_folder])
|
.args(["--build-path".into(), swift_target_folder])
|
||||||
|
.args(["--cache-path".into(), swift_cache_folder])
|
||||||
.current_dir(&swift_package_root)
|
.current_dir(&swift_package_root)
|
||||||
.status()
|
.status()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
|
@ -133,9 +135,17 @@ fn swift_package_root() -> PathBuf {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn swift_target_folder() -> PathBuf {
|
fn swift_target_folder() -> PathBuf {
|
||||||
|
let target = env::var("TARGET").unwrap();
|
||||||
env::current_dir()
|
env::current_dir()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.join(format!("../../target/{SWIFT_PACKAGE_NAME}"))
|
.join(format!("../../target/{target}/{SWIFT_PACKAGE_NAME}_target"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn swift_cache_folder() -> PathBuf {
|
||||||
|
let target = env::var("TARGET").unwrap();
|
||||||
|
env::current_dir()
|
||||||
|
.unwrap()
|
||||||
|
.join(format!("../../target/{target}/{SWIFT_PACKAGE_NAME}_cache"))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn copy_dir(source: &Path, destination: &Path) {
|
fn copy_dir(source: &Path, destination: &Path) {
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use std::time::Duration;
|
use std::{sync::Arc, time::Duration};
|
||||||
|
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use gpui::{actions, keymap_matcher::Binding, Menu, MenuItem};
|
use gpui::{actions, KeyBinding};
|
||||||
use live_kit_client::{
|
use live_kit_client::{
|
||||||
LocalAudioTrack, LocalVideoTrack, RemoteAudioTrackUpdate, RemoteVideoTrackUpdate, Room,
|
LocalAudioTrack, LocalVideoTrack, RemoteAudioTrackUpdate, RemoteVideoTrackUpdate, Room,
|
||||||
};
|
};
|
||||||
|
@ -9,30 +9,32 @@ use live_kit_server::token::{self, VideoGrant};
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
use simplelog::SimpleLogger;
|
use simplelog::SimpleLogger;
|
||||||
|
|
||||||
actions!(capture, [Quit]);
|
actions!(live_kit_client, [Quit]);
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
SimpleLogger::init(LevelFilter::Info, Default::default()).expect("could not initialize logger");
|
SimpleLogger::init(LevelFilter::Info, Default::default()).expect("could not initialize logger");
|
||||||
|
|
||||||
gpui::App::new(()).unwrap().run(|cx| {
|
gpui::App::production(Arc::new(())).run(|cx| {
|
||||||
#[cfg(any(test, feature = "test-support"))]
|
#[cfg(any(test, feature = "test-support"))]
|
||||||
println!("USING TEST LIVEKIT");
|
println!("USING TEST LIVEKIT");
|
||||||
|
|
||||||
#[cfg(not(any(test, feature = "test-support")))]
|
#[cfg(not(any(test, feature = "test-support")))]
|
||||||
println!("USING REAL LIVEKIT");
|
println!("USING REAL LIVEKIT");
|
||||||
|
|
||||||
cx.platform().activate(true);
|
cx.activate(true);
|
||||||
cx.add_global_action(quit);
|
|
||||||
|
|
||||||
cx.add_bindings([Binding::new("cmd-q", Quit, None)]);
|
cx.on_action(quit);
|
||||||
cx.set_menus(vec![Menu {
|
cx.bind_keys([KeyBinding::new("cmd-q", Quit, None)]);
|
||||||
name: "Zed",
|
|
||||||
items: vec![MenuItem::Action {
|
// todo!()
|
||||||
name: "Quit",
|
// cx.set_menus(vec![Menu {
|
||||||
action: Box::new(Quit),
|
// name: "Zed",
|
||||||
os_action: None,
|
// items: vec![MenuItem::Action {
|
||||||
}],
|
// name: "Quit",
|
||||||
}]);
|
// action: Box::new(Quit),
|
||||||
|
// os_action: None,
|
||||||
|
// }],
|
||||||
|
// }]);
|
||||||
|
|
||||||
let live_kit_url = std::env::var("LIVE_KIT_URL").unwrap_or("http://localhost:7880".into());
|
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_key = std::env::var("LIVE_KIT_KEY").unwrap_or("devkey".into());
|
||||||
|
@ -100,7 +102,7 @@ fn main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
println!("Pausing for 5 seconds to test audio, make some noise!");
|
println!("Pausing for 5 seconds to test audio, make some noise!");
|
||||||
let timer = cx.background().timer(Duration::from_secs(5));
|
let timer = cx.background_executor().timer(Duration::from_secs(5));
|
||||||
timer.await;
|
timer.await;
|
||||||
let remote_audio_track = room_b
|
let remote_audio_track = room_b
|
||||||
.remote_audio_tracks("test-participant-1")
|
.remote_audio_tracks("test-participant-1")
|
||||||
|
@ -163,12 +165,12 @@ fn main() {
|
||||||
panic!("unexpected message");
|
panic!("unexpected message");
|
||||||
}
|
}
|
||||||
|
|
||||||
cx.platform().quit();
|
cx.update(|cx| cx.shutdown()).ok();
|
||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn quit(_: &Quit, cx: &mut gpui::AppContext) {
|
fn quit(_: &Quit, cx: &mut gpui::AppContext) {
|
||||||
cx.platform().quit();
|
cx.quit();
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,6 +17,29 @@ use std::{
|
||||||
sync::{Arc, Weak},
|
sync::{Arc, Weak},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// SAFETY: Most live kit types are threadsafe:
|
||||||
|
// https://github.com/livekit/client-sdk-swift#thread-safety
|
||||||
|
macro_rules! pointer_type {
|
||||||
|
($pointer_name:ident) => {
|
||||||
|
#[repr(transparent)]
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub struct $pointer_name(pub *const std::ffi::c_void);
|
||||||
|
unsafe impl Send for $pointer_name {}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
mod swift {
|
||||||
|
pointer_type!(Room);
|
||||||
|
pointer_type!(LocalAudioTrack);
|
||||||
|
pointer_type!(RemoteAudioTrack);
|
||||||
|
pointer_type!(LocalVideoTrack);
|
||||||
|
pointer_type!(RemoteVideoTrack);
|
||||||
|
pointer_type!(LocalTrackPublication);
|
||||||
|
pointer_type!(RemoteTrackPublication);
|
||||||
|
pointer_type!(MacOSDisplay);
|
||||||
|
pointer_type!(RoomDelegate);
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
fn LKRoomDelegateCreate(
|
fn LKRoomDelegateCreate(
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
|
@ -25,8 +48,8 @@ extern "C" {
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
publisher_id: CFStringRef,
|
publisher_id: CFStringRef,
|
||||||
track_id: CFStringRef,
|
track_id: CFStringRef,
|
||||||
remote_track: *const c_void,
|
remote_track: swift::RemoteAudioTrack,
|
||||||
remote_publication: *const c_void,
|
remote_publication: swift::RemoteTrackPublication,
|
||||||
),
|
),
|
||||||
on_did_unsubscribe_from_remote_audio_track: extern "C" fn(
|
on_did_unsubscribe_from_remote_audio_track: extern "C" fn(
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
|
@ -46,49 +69,50 @@ extern "C" {
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
publisher_id: CFStringRef,
|
publisher_id: CFStringRef,
|
||||||
track_id: CFStringRef,
|
track_id: CFStringRef,
|
||||||
remote_track: *const c_void,
|
remote_track: swift::RemoteVideoTrack,
|
||||||
),
|
),
|
||||||
on_did_unsubscribe_from_remote_video_track: extern "C" fn(
|
on_did_unsubscribe_from_remote_video_track: extern "C" fn(
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
publisher_id: CFStringRef,
|
publisher_id: CFStringRef,
|
||||||
track_id: CFStringRef,
|
track_id: CFStringRef,
|
||||||
),
|
),
|
||||||
) -> *const c_void;
|
) -> swift::RoomDelegate;
|
||||||
|
|
||||||
fn LKRoomCreate(delegate: *const c_void) -> *const c_void;
|
fn LKRoomCreate(delegate: swift::RoomDelegate) -> swift::Room;
|
||||||
fn LKRoomConnect(
|
fn LKRoomConnect(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
url: CFStringRef,
|
url: CFStringRef,
|
||||||
token: CFStringRef,
|
token: CFStringRef,
|
||||||
callback: extern "C" fn(*mut c_void, CFStringRef),
|
callback: extern "C" fn(*mut c_void, CFStringRef),
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
);
|
);
|
||||||
fn LKRoomDisconnect(room: *const c_void);
|
fn LKRoomDisconnect(room: swift::Room);
|
||||||
fn LKRoomPublishVideoTrack(
|
fn LKRoomPublishVideoTrack(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
track: *const c_void,
|
track: swift::LocalVideoTrack,
|
||||||
callback: extern "C" fn(*mut c_void, *mut c_void, CFStringRef),
|
callback: extern "C" fn(*mut c_void, swift::LocalTrackPublication, CFStringRef),
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
);
|
);
|
||||||
fn LKRoomPublishAudioTrack(
|
fn LKRoomPublishAudioTrack(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
track: *const c_void,
|
track: swift::LocalAudioTrack,
|
||||||
callback: extern "C" fn(*mut c_void, *mut c_void, CFStringRef),
|
callback: extern "C" fn(*mut c_void, swift::LocalTrackPublication, CFStringRef),
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
);
|
);
|
||||||
fn LKRoomUnpublishTrack(room: *const c_void, publication: *const c_void);
|
fn LKRoomUnpublishTrack(room: swift::Room, publication: swift::LocalTrackPublication);
|
||||||
|
|
||||||
fn LKRoomAudioTracksForRemoteParticipant(
|
fn LKRoomAudioTracksForRemoteParticipant(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
participant_id: CFStringRef,
|
participant_id: CFStringRef,
|
||||||
) -> CFArrayRef;
|
) -> CFArrayRef;
|
||||||
|
|
||||||
fn LKRoomAudioTrackPublicationsForRemoteParticipant(
|
fn LKRoomAudioTrackPublicationsForRemoteParticipant(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
participant_id: CFStringRef,
|
participant_id: CFStringRef,
|
||||||
) -> CFArrayRef;
|
) -> CFArrayRef;
|
||||||
|
|
||||||
fn LKRoomVideoTracksForRemoteParticipant(
|
fn LKRoomVideoTracksForRemoteParticipant(
|
||||||
room: *const c_void,
|
room: swift::Room,
|
||||||
participant_id: CFStringRef,
|
participant_id: CFStringRef,
|
||||||
) -> CFArrayRef;
|
) -> CFArrayRef;
|
||||||
|
|
||||||
|
@ -98,9 +122,9 @@ extern "C" {
|
||||||
on_drop: extern "C" fn(callback_data: *mut c_void),
|
on_drop: extern "C" fn(callback_data: *mut c_void),
|
||||||
) -> *const c_void;
|
) -> *const c_void;
|
||||||
|
|
||||||
fn LKRemoteAudioTrackGetSid(track: *const c_void) -> CFStringRef;
|
fn LKRemoteAudioTrackGetSid(track: swift::RemoteAudioTrack) -> CFStringRef;
|
||||||
fn LKVideoTrackAddRenderer(track: *const c_void, renderer: *const c_void);
|
fn LKVideoTrackAddRenderer(track: swift::RemoteVideoTrack, renderer: *const c_void);
|
||||||
fn LKRemoteVideoTrackGetSid(track: *const c_void) -> CFStringRef;
|
fn LKRemoteVideoTrackGetSid(track: swift::RemoteVideoTrack) -> CFStringRef;
|
||||||
|
|
||||||
fn LKDisplaySources(
|
fn LKDisplaySources(
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
|
@ -110,25 +134,25 @@ extern "C" {
|
||||||
error: CFStringRef,
|
error: CFStringRef,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
fn LKCreateScreenShareTrackForDisplay(display: *const c_void) -> *const c_void;
|
fn LKCreateScreenShareTrackForDisplay(display: swift::MacOSDisplay) -> swift::LocalVideoTrack;
|
||||||
fn LKLocalAudioTrackCreateTrack() -> *const c_void;
|
fn LKLocalAudioTrackCreateTrack() -> swift::LocalAudioTrack;
|
||||||
|
|
||||||
fn LKLocalTrackPublicationSetMute(
|
fn LKLocalTrackPublicationSetMute(
|
||||||
publication: *const c_void,
|
publication: swift::LocalTrackPublication,
|
||||||
muted: bool,
|
muted: bool,
|
||||||
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
);
|
);
|
||||||
|
|
||||||
fn LKRemoteTrackPublicationSetEnabled(
|
fn LKRemoteTrackPublicationSetEnabled(
|
||||||
publication: *const c_void,
|
publication: swift::RemoteTrackPublication,
|
||||||
enabled: bool,
|
enabled: bool,
|
||||||
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
||||||
callback_data: *mut c_void,
|
callback_data: *mut c_void,
|
||||||
);
|
);
|
||||||
|
|
||||||
fn LKRemoteTrackPublicationIsMuted(publication: *const c_void) -> bool;
|
fn LKRemoteTrackPublicationIsMuted(publication: swift::RemoteTrackPublication) -> bool;
|
||||||
fn LKRemoteTrackPublicationGetSid(publication: *const c_void) -> CFStringRef;
|
fn LKRemoteTrackPublicationGetSid(publication: swift::RemoteTrackPublication) -> CFStringRef;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type Sid = String;
|
pub type Sid = String;
|
||||||
|
@ -140,30 +164,29 @@ pub enum ConnectionState {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Room {
|
pub struct Room {
|
||||||
native_room: *const c_void,
|
native_room: Mutex<swift::Room>,
|
||||||
connection: Mutex<(
|
connection: Mutex<(
|
||||||
watch::Sender<ConnectionState>,
|
watch::Sender<ConnectionState>,
|
||||||
watch::Receiver<ConnectionState>,
|
watch::Receiver<ConnectionState>,
|
||||||
)>,
|
)>,
|
||||||
remote_audio_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteAudioTrackUpdate>>>,
|
remote_audio_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteAudioTrackUpdate>>>,
|
||||||
remote_video_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteVideoTrackUpdate>>>,
|
remote_video_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteVideoTrackUpdate>>>,
|
||||||
_delegate: RoomDelegate,
|
_delegate: Mutex<RoomDelegate>,
|
||||||
}
|
}
|
||||||
|
|
||||||
// SAFETY: LiveKit objects are thread-safe: https://github.com/livekit/client-sdk-swift#thread-safety
|
trait AssertSendSync: Send {}
|
||||||
unsafe impl Send for Room {}
|
impl AssertSendSync for Room {}
|
||||||
unsafe impl Sync for Room {}
|
|
||||||
|
|
||||||
impl Room {
|
impl Room {
|
||||||
pub fn new() -> Arc<Self> {
|
pub fn new() -> Arc<Self> {
|
||||||
Arc::new_cyclic(|weak_room| {
|
Arc::new_cyclic(|weak_room| {
|
||||||
let delegate = RoomDelegate::new(weak_room.clone());
|
let delegate = RoomDelegate::new(weak_room.clone());
|
||||||
Self {
|
Self {
|
||||||
native_room: unsafe { LKRoomCreate(delegate.native_delegate) },
|
native_room: Mutex::new(unsafe { LKRoomCreate(delegate.native_delegate) }),
|
||||||
connection: Mutex::new(watch::channel_with(ConnectionState::Disconnected)),
|
connection: Mutex::new(watch::channel_with(ConnectionState::Disconnected)),
|
||||||
remote_audio_track_subscribers: Default::default(),
|
remote_audio_track_subscribers: Default::default(),
|
||||||
remote_video_track_subscribers: Default::default(),
|
remote_video_track_subscribers: Default::default(),
|
||||||
_delegate: delegate,
|
_delegate: Mutex::new(delegate),
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -178,7 +201,7 @@ impl Room {
|
||||||
let (did_connect, tx, rx) = Self::build_done_callback();
|
let (did_connect, tx, rx) = Self::build_done_callback();
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRoomConnect(
|
LKRoomConnect(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
url.as_concrete_TypeRef(),
|
url.as_concrete_TypeRef(),
|
||||||
token.as_concrete_TypeRef(),
|
token.as_concrete_TypeRef(),
|
||||||
did_connect,
|
did_connect,
|
||||||
|
@ -210,7 +233,7 @@ impl Room {
|
||||||
} else {
|
} else {
|
||||||
let sources = CFArray::wrap_under_get_rule(sources)
|
let sources = CFArray::wrap_under_get_rule(sources)
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|source| MacOSDisplay::new(*source))
|
.map(|source| MacOSDisplay::new(swift::MacOSDisplay(*source)))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let _ = tx.send(Ok(sources));
|
let _ = tx.send(Ok(sources));
|
||||||
|
@ -232,7 +255,11 @@ impl Room {
|
||||||
track: LocalVideoTrack,
|
track: LocalVideoTrack,
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
||||||
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
||||||
extern "C" fn callback(tx: *mut c_void, publication: *mut c_void, error: CFStringRef) {
|
extern "C" fn callback(
|
||||||
|
tx: *mut c_void,
|
||||||
|
publication: swift::LocalTrackPublication,
|
||||||
|
error: CFStringRef,
|
||||||
|
) {
|
||||||
let tx =
|
let tx =
|
||||||
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
||||||
if error.is_null() {
|
if error.is_null() {
|
||||||
|
@ -244,7 +271,7 @@ impl Room {
|
||||||
}
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRoomPublishVideoTrack(
|
LKRoomPublishVideoTrack(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
track.0,
|
track.0,
|
||||||
callback,
|
callback,
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
Box::into_raw(Box::new(tx)) as *mut c_void,
|
||||||
|
@ -258,7 +285,11 @@ impl Room {
|
||||||
track: LocalAudioTrack,
|
track: LocalAudioTrack,
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
||||||
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
||||||
extern "C" fn callback(tx: *mut c_void, publication: *mut c_void, error: CFStringRef) {
|
extern "C" fn callback(
|
||||||
|
tx: *mut c_void,
|
||||||
|
publication: swift::LocalTrackPublication,
|
||||||
|
error: CFStringRef,
|
||||||
|
) {
|
||||||
let tx =
|
let tx =
|
||||||
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
||||||
if error.is_null() {
|
if error.is_null() {
|
||||||
|
@ -270,7 +301,7 @@ impl Room {
|
||||||
}
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRoomPublishAudioTrack(
|
LKRoomPublishAudioTrack(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
track.0,
|
track.0,
|
||||||
callback,
|
callback,
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
Box::into_raw(Box::new(tx)) as *mut c_void,
|
||||||
|
@ -281,14 +312,14 @@ impl Room {
|
||||||
|
|
||||||
pub fn unpublish_track(&self, publication: LocalTrackPublication) {
|
pub fn unpublish_track(&self, publication: LocalTrackPublication) {
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRoomUnpublishTrack(self.native_room, publication.0);
|
LKRoomUnpublishTrack(*self.native_room.lock(), publication.0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn remote_video_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteVideoTrack>> {
|
pub fn remote_video_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteVideoTrack>> {
|
||||||
unsafe {
|
unsafe {
|
||||||
let tracks = LKRoomVideoTracksForRemoteParticipant(
|
let tracks = LKRoomVideoTracksForRemoteParticipant(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
CFString::new(participant_id).as_concrete_TypeRef(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -299,7 +330,7 @@ impl Room {
|
||||||
tracks
|
tracks
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|native_track| {
|
.map(|native_track| {
|
||||||
let native_track = *native_track;
|
let native_track = swift::RemoteVideoTrack(*native_track);
|
||||||
let id =
|
let id =
|
||||||
CFString::wrap_under_get_rule(LKRemoteVideoTrackGetSid(native_track))
|
CFString::wrap_under_get_rule(LKRemoteVideoTrackGetSid(native_track))
|
||||||
.to_string();
|
.to_string();
|
||||||
|
@ -317,7 +348,7 @@ impl Room {
|
||||||
pub fn remote_audio_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteAudioTrack>> {
|
pub fn remote_audio_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteAudioTrack>> {
|
||||||
unsafe {
|
unsafe {
|
||||||
let tracks = LKRoomAudioTracksForRemoteParticipant(
|
let tracks = LKRoomAudioTracksForRemoteParticipant(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
CFString::new(participant_id).as_concrete_TypeRef(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -328,7 +359,7 @@ impl Room {
|
||||||
tracks
|
tracks
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|native_track| {
|
.map(|native_track| {
|
||||||
let native_track = *native_track;
|
let native_track = swift::RemoteAudioTrack(*native_track);
|
||||||
let id =
|
let id =
|
||||||
CFString::wrap_under_get_rule(LKRemoteAudioTrackGetSid(native_track))
|
CFString::wrap_under_get_rule(LKRemoteAudioTrackGetSid(native_track))
|
||||||
.to_string();
|
.to_string();
|
||||||
|
@ -349,7 +380,7 @@ impl Room {
|
||||||
) -> Vec<Arc<RemoteTrackPublication>> {
|
) -> Vec<Arc<RemoteTrackPublication>> {
|
||||||
unsafe {
|
unsafe {
|
||||||
let tracks = LKRoomAudioTrackPublicationsForRemoteParticipant(
|
let tracks = LKRoomAudioTrackPublicationsForRemoteParticipant(
|
||||||
self.native_room,
|
*self.native_room.lock(),
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
CFString::new(participant_id).as_concrete_TypeRef(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -360,7 +391,8 @@ impl Room {
|
||||||
tracks
|
tracks
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|native_track_publication| {
|
.map(|native_track_publication| {
|
||||||
let native_track_publication = *native_track_publication;
|
let native_track_publication =
|
||||||
|
swift::RemoteTrackPublication(*native_track_publication);
|
||||||
Arc::new(RemoteTrackPublication::new(native_track_publication))
|
Arc::new(RemoteTrackPublication::new(native_track_publication))
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
|
@ -467,28 +499,32 @@ impl Room {
|
||||||
rx,
|
rx,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn set_display_sources(&self, _: Vec<MacOSDisplay>) {
|
||||||
|
unreachable!("This is a test-only function")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for Room {
|
impl Drop for Room {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRoomDisconnect(self.native_room);
|
let native_room = &*self.native_room.lock();
|
||||||
CFRelease(self.native_room);
|
LKRoomDisconnect(*native_room);
|
||||||
|
CFRelease(native_room.0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct RoomDelegate {
|
struct RoomDelegate {
|
||||||
native_delegate: *const c_void,
|
native_delegate: swift::RoomDelegate,
|
||||||
weak_room: *const Room,
|
_weak_room: Weak<Room>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RoomDelegate {
|
impl RoomDelegate {
|
||||||
fn new(weak_room: Weak<Room>) -> Self {
|
fn new(weak_room: Weak<Room>) -> Self {
|
||||||
let weak_room = Weak::into_raw(weak_room);
|
|
||||||
let native_delegate = unsafe {
|
let native_delegate = unsafe {
|
||||||
LKRoomDelegateCreate(
|
LKRoomDelegateCreate(
|
||||||
weak_room as *mut c_void,
|
weak_room.as_ptr() as *mut c_void,
|
||||||
Self::on_did_disconnect,
|
Self::on_did_disconnect,
|
||||||
Self::on_did_subscribe_to_remote_audio_track,
|
Self::on_did_subscribe_to_remote_audio_track,
|
||||||
Self::on_did_unsubscribe_from_remote_audio_track,
|
Self::on_did_unsubscribe_from_remote_audio_track,
|
||||||
|
@ -500,7 +536,7 @@ impl RoomDelegate {
|
||||||
};
|
};
|
||||||
Self {
|
Self {
|
||||||
native_delegate,
|
native_delegate,
|
||||||
weak_room,
|
_weak_room: weak_room,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -516,8 +552,8 @@ impl RoomDelegate {
|
||||||
room: *mut c_void,
|
room: *mut c_void,
|
||||||
publisher_id: CFStringRef,
|
publisher_id: CFStringRef,
|
||||||
track_id: CFStringRef,
|
track_id: CFStringRef,
|
||||||
track: *const c_void,
|
track: swift::RemoteAudioTrack,
|
||||||
publication: *const c_void,
|
publication: swift::RemoteTrackPublication,
|
||||||
) {
|
) {
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
||||||
|
@ -584,7 +620,7 @@ impl RoomDelegate {
|
||||||
room: *mut c_void,
|
room: *mut c_void,
|
||||||
publisher_id: CFStringRef,
|
publisher_id: CFStringRef,
|
||||||
track_id: CFStringRef,
|
track_id: CFStringRef,
|
||||||
track: *const c_void,
|
track: swift::RemoteVideoTrack,
|
||||||
) {
|
) {
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
||||||
|
@ -614,14 +650,12 @@ impl RoomDelegate {
|
||||||
impl Drop for RoomDelegate {
|
impl Drop for RoomDelegate {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRelease(self.native_delegate);
|
CFRelease(self.native_delegate.0);
|
||||||
let _ = Weak::from_raw(self.weak_room);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct LocalAudioTrack(*const c_void);
|
pub struct LocalAudioTrack(swift::LocalAudioTrack);
|
||||||
unsafe impl Send for LocalAudioTrack {}
|
|
||||||
|
|
||||||
impl LocalAudioTrack {
|
impl LocalAudioTrack {
|
||||||
pub fn create() -> Self {
|
pub fn create() -> Self {
|
||||||
|
@ -631,12 +665,11 @@ impl LocalAudioTrack {
|
||||||
|
|
||||||
impl Drop for LocalAudioTrack {
|
impl Drop for LocalAudioTrack {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.0) }
|
unsafe { CFRelease(self.0 .0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct LocalVideoTrack(*const c_void);
|
pub struct LocalVideoTrack(swift::LocalVideoTrack);
|
||||||
unsafe impl Send for LocalVideoTrack {}
|
|
||||||
|
|
||||||
impl LocalVideoTrack {
|
impl LocalVideoTrack {
|
||||||
pub fn screen_share_for_display(display: &MacOSDisplay) -> Self {
|
pub fn screen_share_for_display(display: &MacOSDisplay) -> Self {
|
||||||
|
@ -646,17 +679,16 @@ impl LocalVideoTrack {
|
||||||
|
|
||||||
impl Drop for LocalVideoTrack {
|
impl Drop for LocalVideoTrack {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.0) }
|
unsafe { CFRelease(self.0 .0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct LocalTrackPublication(*const c_void);
|
pub struct LocalTrackPublication(swift::LocalTrackPublication);
|
||||||
unsafe impl Send for LocalTrackPublication {}
|
|
||||||
|
|
||||||
impl LocalTrackPublication {
|
impl LocalTrackPublication {
|
||||||
pub fn new(native_track_publication: *const c_void) -> Self {
|
pub fn new(native_track_publication: swift::LocalTrackPublication) -> Self {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRetain(native_track_publication);
|
CFRetain(native_track_publication.0);
|
||||||
}
|
}
|
||||||
Self(native_track_publication)
|
Self(native_track_publication)
|
||||||
}
|
}
|
||||||
|
@ -689,28 +721,35 @@ impl LocalTrackPublication {
|
||||||
|
|
||||||
impl Drop for LocalTrackPublication {
|
impl Drop for LocalTrackPublication {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.0) }
|
unsafe { CFRelease(self.0 .0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct RemoteTrackPublication(*const c_void);
|
pub struct RemoteTrackPublication {
|
||||||
|
native_publication: Mutex<swift::RemoteTrackPublication>,
|
||||||
unsafe impl Send for RemoteTrackPublication {}
|
}
|
||||||
|
|
||||||
impl RemoteTrackPublication {
|
impl RemoteTrackPublication {
|
||||||
pub fn new(native_track_publication: *const c_void) -> Self {
|
pub fn new(native_track_publication: swift::RemoteTrackPublication) -> Self {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRetain(native_track_publication);
|
CFRetain(native_track_publication.0);
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
native_publication: Mutex::new(native_track_publication),
|
||||||
}
|
}
|
||||||
Self(native_track_publication)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn sid(&self) -> String {
|
pub fn sid(&self) -> String {
|
||||||
unsafe { CFString::wrap_under_get_rule(LKRemoteTrackPublicationGetSid(self.0)).to_string() }
|
unsafe {
|
||||||
|
CFString::wrap_under_get_rule(LKRemoteTrackPublicationGetSid(
|
||||||
|
*self.native_publication.lock(),
|
||||||
|
))
|
||||||
|
.to_string()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_muted(&self) -> bool {
|
pub fn is_muted(&self) -> bool {
|
||||||
unsafe { LKRemoteTrackPublicationIsMuted(self.0) }
|
unsafe { LKRemoteTrackPublicationIsMuted(*self.native_publication.lock()) }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_enabled(&self, enabled: bool) -> impl Future<Output = Result<()>> {
|
pub fn set_enabled(&self, enabled: bool) -> impl Future<Output = Result<()>> {
|
||||||
|
@ -728,7 +767,7 @@ impl RemoteTrackPublication {
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
LKRemoteTrackPublicationSetEnabled(
|
LKRemoteTrackPublicationSetEnabled(
|
||||||
self.0,
|
*self.native_publication.lock(),
|
||||||
enabled,
|
enabled,
|
||||||
complete_callback,
|
complete_callback,
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
Box::into_raw(Box::new(tx)) as *mut c_void,
|
||||||
|
@ -741,26 +780,24 @@ impl RemoteTrackPublication {
|
||||||
|
|
||||||
impl Drop for RemoteTrackPublication {
|
impl Drop for RemoteTrackPublication {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.0) }
|
unsafe { CFRelease((*self.native_publication.lock()).0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct RemoteAudioTrack {
|
pub struct RemoteAudioTrack {
|
||||||
_native_track: *const c_void,
|
native_track: Mutex<swift::RemoteAudioTrack>,
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
publisher_id: String,
|
publisher_id: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe impl Send for RemoteAudioTrack {}
|
|
||||||
|
|
||||||
impl RemoteAudioTrack {
|
impl RemoteAudioTrack {
|
||||||
fn new(native_track: *const c_void, sid: Sid, publisher_id: String) -> Self {
|
fn new(native_track: swift::RemoteAudioTrack, sid: Sid, publisher_id: String) -> Self {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRetain(native_track);
|
CFRetain(native_track.0);
|
||||||
}
|
}
|
||||||
Self {
|
Self {
|
||||||
_native_track: native_track,
|
native_track: Mutex::new(native_track),
|
||||||
sid,
|
sid,
|
||||||
publisher_id,
|
publisher_id,
|
||||||
}
|
}
|
||||||
|
@ -783,22 +820,26 @@ impl RemoteAudioTrack {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Drop for RemoteAudioTrack {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe { CFRelease(self.native_track.lock().0) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct RemoteVideoTrack {
|
pub struct RemoteVideoTrack {
|
||||||
native_track: *const c_void,
|
native_track: Mutex<swift::RemoteVideoTrack>,
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
publisher_id: String,
|
publisher_id: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe impl Send for RemoteVideoTrack {}
|
|
||||||
|
|
||||||
impl RemoteVideoTrack {
|
impl RemoteVideoTrack {
|
||||||
fn new(native_track: *const c_void, sid: Sid, publisher_id: String) -> Self {
|
fn new(native_track: swift::RemoteVideoTrack, sid: Sid, publisher_id: String) -> Self {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRetain(native_track);
|
CFRetain(native_track.0);
|
||||||
}
|
}
|
||||||
Self {
|
Self {
|
||||||
native_track,
|
native_track: Mutex::new(native_track),
|
||||||
sid,
|
sid,
|
||||||
publisher_id,
|
publisher_id,
|
||||||
}
|
}
|
||||||
|
@ -847,7 +888,7 @@ impl RemoteVideoTrack {
|
||||||
on_frame,
|
on_frame,
|
||||||
on_drop,
|
on_drop,
|
||||||
);
|
);
|
||||||
LKVideoTrackAddRenderer(self.native_track, renderer);
|
LKVideoTrackAddRenderer(*self.native_track.lock(), renderer);
|
||||||
rx
|
rx
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -855,7 +896,7 @@ impl RemoteVideoTrack {
|
||||||
|
|
||||||
impl Drop for RemoteVideoTrack {
|
impl Drop for RemoteVideoTrack {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.native_track) }
|
unsafe { CFRelease(self.native_track.lock().0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -871,14 +912,12 @@ pub enum RemoteAudioTrackUpdate {
|
||||||
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct MacOSDisplay(*const c_void);
|
pub struct MacOSDisplay(swift::MacOSDisplay);
|
||||||
|
|
||||||
unsafe impl Send for MacOSDisplay {}
|
|
||||||
|
|
||||||
impl MacOSDisplay {
|
impl MacOSDisplay {
|
||||||
fn new(ptr: *const c_void) -> Self {
|
fn new(ptr: swift::MacOSDisplay) -> Self {
|
||||||
unsafe {
|
unsafe {
|
||||||
CFRetain(ptr);
|
CFRetain(ptr.0);
|
||||||
}
|
}
|
||||||
Self(ptr)
|
Self(ptr)
|
||||||
}
|
}
|
||||||
|
@ -886,7 +925,7 @@ impl MacOSDisplay {
|
||||||
|
|
||||||
impl Drop for MacOSDisplay {
|
impl Drop for MacOSDisplay {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { CFRelease(self.0) }
|
unsafe { CFRelease(self.0 .0) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use anyhow::{anyhow, Result};
|
use anyhow::{anyhow, Context, Result};
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use collections::{BTreeMap, HashMap};
|
use collections::{BTreeMap, HashMap};
|
||||||
use futures::Stream;
|
use futures::Stream;
|
||||||
use gpui::executor::Background;
|
use gpui::BackgroundExecutor;
|
||||||
use live_kit_server::token;
|
use live_kit_server::token;
|
||||||
use media::core_video::CVImageBuffer;
|
use media::core_video::CVImageBuffer;
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
|
@ -16,7 +16,7 @@ pub struct TestServer {
|
||||||
pub api_key: String,
|
pub api_key: String,
|
||||||
pub secret_key: String,
|
pub secret_key: String,
|
||||||
rooms: Mutex<HashMap<String, TestServerRoom>>,
|
rooms: Mutex<HashMap<String, TestServerRoom>>,
|
||||||
background: Arc<Background>,
|
executor: BackgroundExecutor,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TestServer {
|
impl TestServer {
|
||||||
|
@ -24,7 +24,7 @@ impl TestServer {
|
||||||
url: String,
|
url: String,
|
||||||
api_key: String,
|
api_key: String,
|
||||||
secret_key: String,
|
secret_key: String,
|
||||||
background: Arc<Background>,
|
executor: BackgroundExecutor,
|
||||||
) -> Result<Arc<TestServer>> {
|
) -> Result<Arc<TestServer>> {
|
||||||
let mut servers = SERVERS.lock();
|
let mut servers = SERVERS.lock();
|
||||||
if servers.contains_key(&url) {
|
if servers.contains_key(&url) {
|
||||||
|
@ -35,7 +35,7 @@ impl TestServer {
|
||||||
api_key,
|
api_key,
|
||||||
secret_key,
|
secret_key,
|
||||||
rooms: Default::default(),
|
rooms: Default::default(),
|
||||||
background,
|
executor,
|
||||||
});
|
});
|
||||||
servers.insert(url, server.clone());
|
servers.insert(url, server.clone());
|
||||||
Ok(server)
|
Ok(server)
|
||||||
|
@ -65,7 +65,7 @@ impl TestServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn create_room(&self, room: String) -> Result<()> {
|
pub async fn create_room(&self, room: String) -> Result<()> {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let mut server_rooms = self.rooms.lock();
|
let mut server_rooms = self.rooms.lock();
|
||||||
if server_rooms.contains_key(&room) {
|
if server_rooms.contains_key(&room) {
|
||||||
Err(anyhow!("room {:?} already exists", room))
|
Err(anyhow!("room {:?} already exists", room))
|
||||||
|
@ -77,7 +77,7 @@ impl TestServer {
|
||||||
|
|
||||||
async fn delete_room(&self, room: String) -> Result<()> {
|
async fn delete_room(&self, room: String) -> Result<()> {
|
||||||
// TODO: clear state associated with all `Room`s.
|
// TODO: clear state associated with all `Room`s.
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let mut server_rooms = self.rooms.lock();
|
let mut server_rooms = self.rooms.lock();
|
||||||
server_rooms
|
server_rooms
|
||||||
.remove(&room)
|
.remove(&room)
|
||||||
|
@ -86,7 +86,7 @@ impl TestServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn join_room(&self, token: String, client_room: Arc<Room>) -> Result<()> {
|
async fn join_room(&self, token: String, client_room: Arc<Room>) -> Result<()> {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
||||||
let identity = claims.sub.unwrap().to_string();
|
let identity = claims.sub.unwrap().to_string();
|
||||||
let room_name = claims.video.room.unwrap();
|
let room_name = claims.video.room.unwrap();
|
||||||
|
@ -115,7 +115,7 @@ impl TestServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn leave_room(&self, token: String) -> Result<()> {
|
async fn leave_room(&self, token: String) -> Result<()> {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
||||||
let identity = claims.sub.unwrap().to_string();
|
let identity = claims.sub.unwrap().to_string();
|
||||||
let room_name = claims.video.room.unwrap();
|
let room_name = claims.video.room.unwrap();
|
||||||
|
@ -136,7 +136,7 @@ impl TestServer {
|
||||||
async fn remove_participant(&self, room_name: String, identity: String) -> Result<()> {
|
async fn remove_participant(&self, room_name: String, identity: String) -> Result<()> {
|
||||||
// TODO: clear state associated with the `Room`.
|
// TODO: clear state associated with the `Room`.
|
||||||
|
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let mut server_rooms = self.rooms.lock();
|
let mut server_rooms = self.rooms.lock();
|
||||||
let room = server_rooms
|
let room = server_rooms
|
||||||
.get_mut(&room_name)
|
.get_mut(&room_name)
|
||||||
|
@ -152,7 +152,7 @@ impl TestServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn disconnect_client(&self, client_identity: String) {
|
pub async fn disconnect_client(&self, client_identity: String) {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let mut server_rooms = self.rooms.lock();
|
let mut server_rooms = self.rooms.lock();
|
||||||
for room in server_rooms.values_mut() {
|
for room in server_rooms.values_mut() {
|
||||||
if let Some(room) = room.client_rooms.remove(&client_identity) {
|
if let Some(room) = room.client_rooms.remove(&client_identity) {
|
||||||
|
@ -162,7 +162,7 @@ impl TestServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn publish_video_track(&self, token: String, local_track: LocalVideoTrack) -> Result<()> {
|
async fn publish_video_track(&self, token: String, local_track: LocalVideoTrack) -> Result<()> {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
||||||
let identity = claims.sub.unwrap().to_string();
|
let identity = claims.sub.unwrap().to_string();
|
||||||
let room_name = claims.video.room.unwrap();
|
let room_name = claims.video.room.unwrap();
|
||||||
|
@ -200,7 +200,7 @@ impl TestServer {
|
||||||
token: String,
|
token: String,
|
||||||
_local_track: &LocalAudioTrack,
|
_local_track: &LocalAudioTrack,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
self.background.simulate_random_delay().await;
|
self.executor.simulate_random_delay().await;
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
||||||
let identity = claims.sub.unwrap().to_string();
|
let identity = claims.sub.unwrap().to_string();
|
||||||
let room_name = claims.video.room.unwrap();
|
let room_name = claims.video.room.unwrap();
|
||||||
|
@ -364,7 +364,10 @@ impl Room {
|
||||||
let token = token.to_string();
|
let token = token.to_string();
|
||||||
async move {
|
async move {
|
||||||
let server = TestServer::get(&url)?;
|
let server = TestServer::get(&url)?;
|
||||||
server.join_room(token.clone(), this.clone()).await?;
|
server
|
||||||
|
.join_room(token.clone(), this.clone())
|
||||||
|
.await
|
||||||
|
.context("room join")?;
|
||||||
*this.0.lock().connection.0.borrow_mut() = ConnectionState::Connected { url, token };
|
*this.0.lock().connection.0.borrow_mut() = ConnectionState::Connected { url, token };
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -374,7 +377,7 @@ impl Room {
|
||||||
let this = self.clone();
|
let this = self.clone();
|
||||||
async move {
|
async move {
|
||||||
let server = this.test_server();
|
let server = this.test_server();
|
||||||
server.background.simulate_random_delay().await;
|
server.executor.simulate_random_delay().await;
|
||||||
Ok(this.0.lock().display_sources.clone())
|
Ok(this.0.lock().display_sources.clone())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -492,8 +495,8 @@ impl Drop for Room {
|
||||||
ConnectionState::Disconnected,
|
ConnectionState::Disconnected,
|
||||||
) {
|
) {
|
||||||
if let Ok(server) = TestServer::get(&token) {
|
if let Ok(server) = TestServer::get(&token) {
|
||||||
let background = server.background.clone();
|
let executor = server.executor.clone();
|
||||||
background
|
executor
|
||||||
.spawn(async move { server.leave_room(token).await.unwrap() })
|
.spawn(async move { server.leave_room(token).await.unwrap() })
|
||||||
.detach();
|
.detach();
|
||||||
}
|
}
|
||||||
|
@ -547,6 +550,7 @@ impl LocalAudioTrack {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
pub struct RemoteVideoTrack {
|
pub struct RemoteVideoTrack {
|
||||||
sid: Sid,
|
sid: Sid,
|
||||||
publisher_id: Sid,
|
publisher_id: Sid,
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
[live_kit_client_test]
|
|
||||||
rustflags = ["-C", "link-args=-ObjC"]
|
|
|
@ -1,71 +0,0 @@
|
||||||
[package]
|
|
||||||
name = "live_kit_client2"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
description = "Bindings to LiveKit Swift client SDK"
|
|
||||||
publish = false
|
|
||||||
|
|
||||||
[lib]
|
|
||||||
path = "src/live_kit_client2.rs"
|
|
||||||
doctest = false
|
|
||||||
|
|
||||||
[[example]]
|
|
||||||
name = "test_app2"
|
|
||||||
|
|
||||||
[features]
|
|
||||||
test-support = [
|
|
||||||
"async-trait",
|
|
||||||
"collections/test-support",
|
|
||||||
"gpui/test-support",
|
|
||||||
"live_kit_server",
|
|
||||||
"nanoid",
|
|
||||||
]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
collections = { path = "../collections", optional = true }
|
|
||||||
gpui = { package = "gpui2", path = "../gpui2", optional = true }
|
|
||||||
live_kit_server = { path = "../live_kit_server", optional = true }
|
|
||||||
media = { path = "../media" }
|
|
||||||
|
|
||||||
anyhow.workspace = true
|
|
||||||
async-broadcast = "0.4"
|
|
||||||
core-foundation = "0.9.3"
|
|
||||||
core-graphics = "0.22.3"
|
|
||||||
futures.workspace = true
|
|
||||||
log.workspace = true
|
|
||||||
parking_lot.workspace = true
|
|
||||||
postage.workspace = true
|
|
||||||
|
|
||||||
async-trait = { workspace = true, optional = true }
|
|
||||||
nanoid = { version ="0.4", optional = true}
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
collections = { path = "../collections", features = ["test-support"] }
|
|
||||||
gpui = { package = "gpui2", path = "../gpui2", features = ["test-support"] }
|
|
||||||
live_kit_server = { path = "../live_kit_server" }
|
|
||||||
media = { path = "../media" }
|
|
||||||
nanoid = "0.4"
|
|
||||||
|
|
||||||
anyhow.workspace = true
|
|
||||||
async-trait.workspace = true
|
|
||||||
block = "0.1"
|
|
||||||
bytes = "1.2"
|
|
||||||
byteorder = "1.4"
|
|
||||||
cocoa = "0.24"
|
|
||||||
core-foundation = "0.9.3"
|
|
||||||
core-graphics = "0.22.3"
|
|
||||||
foreign-types = "0.3"
|
|
||||||
futures.workspace = true
|
|
||||||
hmac = "0.12"
|
|
||||||
jwt = "0.16"
|
|
||||||
objc = "0.2"
|
|
||||||
parking_lot.workspace = true
|
|
||||||
serde.workspace = true
|
|
||||||
serde_derive.workspace = true
|
|
||||||
sha2 = "0.10"
|
|
||||||
simplelog = "0.9"
|
|
||||||
|
|
||||||
[build-dependencies]
|
|
||||||
serde.workspace = true
|
|
||||||
serde_derive.workspace = true
|
|
||||||
serde_json.workspace = true
|
|
|
@ -1,52 +0,0 @@
|
||||||
{
|
|
||||||
"object": {
|
|
||||||
"pins": [
|
|
||||||
{
|
|
||||||
"package": "LiveKit",
|
|
||||||
"repositoryURL": "https://github.com/livekit/client-sdk-swift.git",
|
|
||||||
"state": {
|
|
||||||
"branch": null,
|
|
||||||
"revision": "8b9cefed8d1669ec8fce41376b56dce3036a5f50",
|
|
||||||
"version": "1.1.4"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"package": "Promises",
|
|
||||||
"repositoryURL": "https://github.com/google/promises.git",
|
|
||||||
"state": {
|
|
||||||
"branch": null,
|
|
||||||
"revision": "ec957ccddbcc710ccc64c9dcbd4c7006fcf8b73a",
|
|
||||||
"version": "2.2.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"package": "WebRTC",
|
|
||||||
"repositoryURL": "https://github.com/webrtc-sdk/Specs.git",
|
|
||||||
"state": {
|
|
||||||
"branch": null,
|
|
||||||
"revision": "4fa8d6d647fc759cdd0265fd413d2f28ea2e0e08",
|
|
||||||
"version": "114.5735.8"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"package": "swift-log",
|
|
||||||
"repositoryURL": "https://github.com/apple/swift-log.git",
|
|
||||||
"state": {
|
|
||||||
"branch": null,
|
|
||||||
"revision": "32e8d724467f8fe623624570367e3d50c5638e46",
|
|
||||||
"version": "1.5.2"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"package": "SwiftProtobuf",
|
|
||||||
"repositoryURL": "https://github.com/apple/swift-protobuf.git",
|
|
||||||
"state": {
|
|
||||||
"branch": null,
|
|
||||||
"revision": "ce20dc083ee485524b802669890291c0d8090170",
|
|
||||||
"version": "1.22.1"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"version": 1
|
|
||||||
}
|
|
|
@ -1,27 +0,0 @@
|
||||||
// swift-tools-version: 5.5
|
|
||||||
|
|
||||||
import PackageDescription
|
|
||||||
|
|
||||||
let package = Package(
|
|
||||||
name: "LiveKitBridge2",
|
|
||||||
platforms: [
|
|
||||||
.macOS(.v10_15)
|
|
||||||
],
|
|
||||||
products: [
|
|
||||||
// Products define the executables and libraries a package produces, and make them visible to other packages.
|
|
||||||
.library(
|
|
||||||
name: "LiveKitBridge2",
|
|
||||||
type: .static,
|
|
||||||
targets: ["LiveKitBridge2"]),
|
|
||||||
],
|
|
||||||
dependencies: [
|
|
||||||
.package(url: "https://github.com/livekit/client-sdk-swift.git", .exact("1.1.4")),
|
|
||||||
],
|
|
||||||
targets: [
|
|
||||||
// Targets are the basic building blocks of a package. A target can define a module or a test suite.
|
|
||||||
// Targets can depend on other targets in this package, and on products in packages this package depends on.
|
|
||||||
.target(
|
|
||||||
name: "LiveKitBridge2",
|
|
||||||
dependencies: [.product(name: "LiveKit", package: "client-sdk-swift")]),
|
|
||||||
]
|
|
||||||
)
|
|
|
@ -1,3 +0,0 @@
|
||||||
# LiveKitBridge2
|
|
||||||
|
|
||||||
A description of this package.
|
|
|
@ -1,327 +0,0 @@
|
||||||
import Foundation
|
|
||||||
import LiveKit
|
|
||||||
import WebRTC
|
|
||||||
import ScreenCaptureKit
|
|
||||||
|
|
||||||
class LKRoomDelegate: RoomDelegate {
|
|
||||||
var data: UnsafeRawPointer
|
|
||||||
var onDidDisconnect: @convention(c) (UnsafeRawPointer) -> Void
|
|
||||||
var onDidSubscribeToRemoteAudioTrack: @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer, UnsafeRawPointer) -> Void
|
|
||||||
var onDidUnsubscribeFromRemoteAudioTrack: @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void
|
|
||||||
var onMuteChangedFromRemoteAudioTrack: @convention(c) (UnsafeRawPointer, CFString, Bool) -> Void
|
|
||||||
var onActiveSpeakersChanged: @convention(c) (UnsafeRawPointer, CFArray) -> Void
|
|
||||||
var onDidSubscribeToRemoteVideoTrack: @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer) -> Void
|
|
||||||
var onDidUnsubscribeFromRemoteVideoTrack: @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void
|
|
||||||
|
|
||||||
init(
|
|
||||||
data: UnsafeRawPointer,
|
|
||||||
onDidDisconnect: @escaping @convention(c) (UnsafeRawPointer) -> Void,
|
|
||||||
onDidSubscribeToRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer, UnsafeRawPointer) -> Void,
|
|
||||||
onDidUnsubscribeFromRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void,
|
|
||||||
onMuteChangedFromRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, Bool) -> Void,
|
|
||||||
onActiveSpeakersChanged: @convention(c) (UnsafeRawPointer, CFArray) -> Void,
|
|
||||||
onDidSubscribeToRemoteVideoTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer) -> Void,
|
|
||||||
onDidUnsubscribeFromRemoteVideoTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void)
|
|
||||||
{
|
|
||||||
self.data = data
|
|
||||||
self.onDidDisconnect = onDidDisconnect
|
|
||||||
self.onDidSubscribeToRemoteAudioTrack = onDidSubscribeToRemoteAudioTrack
|
|
||||||
self.onDidUnsubscribeFromRemoteAudioTrack = onDidUnsubscribeFromRemoteAudioTrack
|
|
||||||
self.onDidSubscribeToRemoteVideoTrack = onDidSubscribeToRemoteVideoTrack
|
|
||||||
self.onDidUnsubscribeFromRemoteVideoTrack = onDidUnsubscribeFromRemoteVideoTrack
|
|
||||||
self.onMuteChangedFromRemoteAudioTrack = onMuteChangedFromRemoteAudioTrack
|
|
||||||
self.onActiveSpeakersChanged = onActiveSpeakersChanged
|
|
||||||
}
|
|
||||||
|
|
||||||
func room(_ room: Room, didUpdate connectionState: ConnectionState, oldValue: ConnectionState) {
|
|
||||||
if connectionState.isDisconnected {
|
|
||||||
self.onDidDisconnect(self.data)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func room(_ room: Room, participant: RemoteParticipant, didSubscribe publication: RemoteTrackPublication, track: Track) {
|
|
||||||
if track.kind == .video {
|
|
||||||
self.onDidSubscribeToRemoteVideoTrack(self.data, participant.identity as CFString, track.sid! as CFString, Unmanaged.passUnretained(track).toOpaque())
|
|
||||||
} else if track.kind == .audio {
|
|
||||||
self.onDidSubscribeToRemoteAudioTrack(self.data, participant.identity as CFString, track.sid! as CFString, Unmanaged.passUnretained(track).toOpaque(), Unmanaged.passUnretained(publication).toOpaque())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func room(_ room: Room, participant: Participant, didUpdate publication: TrackPublication, muted: Bool) {
|
|
||||||
if publication.kind == .audio {
|
|
||||||
self.onMuteChangedFromRemoteAudioTrack(self.data, publication.sid as CFString, muted)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func room(_ room: Room, didUpdate speakers: [Participant]) {
|
|
||||||
guard let speaker_ids = speakers.compactMap({ $0.identity as CFString }) as CFArray? else { return }
|
|
||||||
self.onActiveSpeakersChanged(self.data, speaker_ids)
|
|
||||||
}
|
|
||||||
|
|
||||||
func room(_ room: Room, participant: RemoteParticipant, didUnsubscribe publication: RemoteTrackPublication, track: Track) {
|
|
||||||
if track.kind == .video {
|
|
||||||
self.onDidUnsubscribeFromRemoteVideoTrack(self.data, participant.identity as CFString, track.sid! as CFString)
|
|
||||||
} else if track.kind == .audio {
|
|
||||||
self.onDidUnsubscribeFromRemoteAudioTrack(self.data, participant.identity as CFString, track.sid! as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class LKVideoRenderer: NSObject, VideoRenderer {
|
|
||||||
var data: UnsafeRawPointer
|
|
||||||
var onFrame: @convention(c) (UnsafeRawPointer, CVPixelBuffer) -> Bool
|
|
||||||
var onDrop: @convention(c) (UnsafeRawPointer) -> Void
|
|
||||||
var adaptiveStreamIsEnabled: Bool = false
|
|
||||||
var adaptiveStreamSize: CGSize = .zero
|
|
||||||
weak var track: VideoTrack?
|
|
||||||
|
|
||||||
init(data: UnsafeRawPointer, onFrame: @escaping @convention(c) (UnsafeRawPointer, CVPixelBuffer) -> Bool, onDrop: @escaping @convention(c) (UnsafeRawPointer) -> Void) {
|
|
||||||
self.data = data
|
|
||||||
self.onFrame = onFrame
|
|
||||||
self.onDrop = onDrop
|
|
||||||
}
|
|
||||||
|
|
||||||
deinit {
|
|
||||||
self.onDrop(self.data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func setSize(_ size: CGSize) {
|
|
||||||
}
|
|
||||||
|
|
||||||
func renderFrame(_ frame: RTCVideoFrame?) {
|
|
||||||
let buffer = frame?.buffer as? RTCCVPixelBuffer
|
|
||||||
if let pixelBuffer = buffer?.pixelBuffer {
|
|
||||||
if !self.onFrame(self.data, pixelBuffer) {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
self.track?.remove(videoRenderer: self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomDelegateCreate")
|
|
||||||
public func LKRoomDelegateCreate(
|
|
||||||
data: UnsafeRawPointer,
|
|
||||||
onDidDisconnect: @escaping @convention(c) (UnsafeRawPointer) -> Void,
|
|
||||||
onDidSubscribeToRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer, UnsafeRawPointer) -> Void,
|
|
||||||
onDidUnsubscribeFromRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void,
|
|
||||||
onMuteChangedFromRemoteAudioTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, Bool) -> Void,
|
|
||||||
onActiveSpeakerChanged: @escaping @convention(c) (UnsafeRawPointer, CFArray) -> Void,
|
|
||||||
onDidSubscribeToRemoteVideoTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString, UnsafeRawPointer) -> Void,
|
|
||||||
onDidUnsubscribeFromRemoteVideoTrack: @escaping @convention(c) (UnsafeRawPointer, CFString, CFString) -> Void
|
|
||||||
) -> UnsafeMutableRawPointer {
|
|
||||||
let delegate = LKRoomDelegate(
|
|
||||||
data: data,
|
|
||||||
onDidDisconnect: onDidDisconnect,
|
|
||||||
onDidSubscribeToRemoteAudioTrack: onDidSubscribeToRemoteAudioTrack,
|
|
||||||
onDidUnsubscribeFromRemoteAudioTrack: onDidUnsubscribeFromRemoteAudioTrack,
|
|
||||||
onMuteChangedFromRemoteAudioTrack: onMuteChangedFromRemoteAudioTrack,
|
|
||||||
onActiveSpeakersChanged: onActiveSpeakerChanged,
|
|
||||||
onDidSubscribeToRemoteVideoTrack: onDidSubscribeToRemoteVideoTrack,
|
|
||||||
onDidUnsubscribeFromRemoteVideoTrack: onDidUnsubscribeFromRemoteVideoTrack
|
|
||||||
)
|
|
||||||
return Unmanaged.passRetained(delegate).toOpaque()
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomCreate")
|
|
||||||
public func LKRoomCreate(delegate: UnsafeRawPointer) -> UnsafeMutableRawPointer {
|
|
||||||
let delegate = Unmanaged<LKRoomDelegate>.fromOpaque(delegate).takeUnretainedValue()
|
|
||||||
return Unmanaged.passRetained(Room(delegate: delegate)).toOpaque()
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomConnect")
|
|
||||||
public func LKRoomConnect(room: UnsafeRawPointer, url: CFString, token: CFString, callback: @escaping @convention(c) (UnsafeRawPointer, CFString?) -> Void, callback_data: UnsafeRawPointer) {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
|
|
||||||
room.connect(url as String, token as String).then { _ in
|
|
||||||
callback(callback_data, UnsafeRawPointer(nil) as! CFString?)
|
|
||||||
}.catch { error in
|
|
||||||
callback(callback_data, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomDisconnect")
|
|
||||||
public func LKRoomDisconnect(room: UnsafeRawPointer) {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
room.disconnect()
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomPublishVideoTrack")
|
|
||||||
public func LKRoomPublishVideoTrack(room: UnsafeRawPointer, track: UnsafeRawPointer, callback: @escaping @convention(c) (UnsafeRawPointer, UnsafeMutableRawPointer?, CFString?) -> Void, callback_data: UnsafeRawPointer) {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
let track = Unmanaged<LocalVideoTrack>.fromOpaque(track).takeUnretainedValue()
|
|
||||||
room.localParticipant?.publishVideoTrack(track: track).then { publication in
|
|
||||||
callback(callback_data, Unmanaged.passRetained(publication).toOpaque(), nil)
|
|
||||||
}.catch { error in
|
|
||||||
callback(callback_data, nil, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomPublishAudioTrack")
|
|
||||||
public func LKRoomPublishAudioTrack(room: UnsafeRawPointer, track: UnsafeRawPointer, callback: @escaping @convention(c) (UnsafeRawPointer, UnsafeMutableRawPointer?, CFString?) -> Void, callback_data: UnsafeRawPointer) {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
let track = Unmanaged<LocalAudioTrack>.fromOpaque(track).takeUnretainedValue()
|
|
||||||
room.localParticipant?.publishAudioTrack(track: track).then { publication in
|
|
||||||
callback(callback_data, Unmanaged.passRetained(publication).toOpaque(), nil)
|
|
||||||
}.catch { error in
|
|
||||||
callback(callback_data, nil, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@_cdecl("LKRoomUnpublishTrack")
|
|
||||||
public func LKRoomUnpublishTrack(room: UnsafeRawPointer, publication: UnsafeRawPointer) {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
let publication = Unmanaged<LocalTrackPublication>.fromOpaque(publication).takeUnretainedValue()
|
|
||||||
let _ = room.localParticipant?.unpublish(publication: publication)
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomAudioTracksForRemoteParticipant")
|
|
||||||
public func LKRoomAudioTracksForRemoteParticipant(room: UnsafeRawPointer, participantId: CFString) -> CFArray? {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
|
|
||||||
for (_, participant) in room.remoteParticipants {
|
|
||||||
if participant.identity == participantId as String {
|
|
||||||
return participant.audioTracks.compactMap { $0.track as? RemoteAudioTrack } as CFArray?
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil;
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomAudioTrackPublicationsForRemoteParticipant")
|
|
||||||
public func LKRoomAudioTrackPublicationsForRemoteParticipant(room: UnsafeRawPointer, participantId: CFString) -> CFArray? {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
|
|
||||||
for (_, participant) in room.remoteParticipants {
|
|
||||||
if participant.identity == participantId as String {
|
|
||||||
return participant.audioTracks.compactMap { $0 as? RemoteTrackPublication } as CFArray?
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil;
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRoomVideoTracksForRemoteParticipant")
|
|
||||||
public func LKRoomVideoTracksForRemoteParticipant(room: UnsafeRawPointer, participantId: CFString) -> CFArray? {
|
|
||||||
let room = Unmanaged<Room>.fromOpaque(room).takeUnretainedValue()
|
|
||||||
|
|
||||||
for (_, participant) in room.remoteParticipants {
|
|
||||||
if participant.identity == participantId as String {
|
|
||||||
return participant.videoTracks.compactMap { $0.track as? RemoteVideoTrack } as CFArray?
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil;
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKLocalAudioTrackCreateTrack")
|
|
||||||
public func LKLocalAudioTrackCreateTrack() -> UnsafeMutableRawPointer {
|
|
||||||
let track = LocalAudioTrack.createTrack(options: AudioCaptureOptions(
|
|
||||||
echoCancellation: true,
|
|
||||||
noiseSuppression: true
|
|
||||||
))
|
|
||||||
|
|
||||||
return Unmanaged.passRetained(track).toOpaque()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@_cdecl("LKCreateScreenShareTrackForDisplay")
|
|
||||||
public func LKCreateScreenShareTrackForDisplay(display: UnsafeMutableRawPointer) -> UnsafeMutableRawPointer {
|
|
||||||
let display = Unmanaged<MacOSDisplay>.fromOpaque(display).takeUnretainedValue()
|
|
||||||
let track = LocalVideoTrack.createMacOSScreenShareTrack(source: display, preferredMethod: .legacy)
|
|
||||||
return Unmanaged.passRetained(track).toOpaque()
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKVideoRendererCreate")
|
|
||||||
public func LKVideoRendererCreate(data: UnsafeRawPointer, onFrame: @escaping @convention(c) (UnsafeRawPointer, CVPixelBuffer) -> Bool, onDrop: @escaping @convention(c) (UnsafeRawPointer) -> Void) -> UnsafeMutableRawPointer {
|
|
||||||
Unmanaged.passRetained(LKVideoRenderer(data: data, onFrame: onFrame, onDrop: onDrop)).toOpaque()
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKVideoTrackAddRenderer")
|
|
||||||
public func LKVideoTrackAddRenderer(track: UnsafeRawPointer, renderer: UnsafeRawPointer) {
|
|
||||||
let track = Unmanaged<Track>.fromOpaque(track).takeUnretainedValue() as! VideoTrack
|
|
||||||
let renderer = Unmanaged<LKVideoRenderer>.fromOpaque(renderer).takeRetainedValue()
|
|
||||||
renderer.track = track
|
|
||||||
track.add(videoRenderer: renderer)
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRemoteVideoTrackGetSid")
|
|
||||||
public func LKRemoteVideoTrackGetSid(track: UnsafeRawPointer) -> CFString {
|
|
||||||
let track = Unmanaged<RemoteVideoTrack>.fromOpaque(track).takeUnretainedValue()
|
|
||||||
return track.sid! as CFString
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRemoteAudioTrackGetSid")
|
|
||||||
public func LKRemoteAudioTrackGetSid(track: UnsafeRawPointer) -> CFString {
|
|
||||||
let track = Unmanaged<RemoteAudioTrack>.fromOpaque(track).takeUnretainedValue()
|
|
||||||
return track.sid! as CFString
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKDisplaySources")
|
|
||||||
public func LKDisplaySources(data: UnsafeRawPointer, callback: @escaping @convention(c) (UnsafeRawPointer, CFArray?, CFString?) -> Void) {
|
|
||||||
MacOSScreenCapturer.sources(for: .display, includeCurrentApplication: false, preferredMethod: .legacy).then { displaySources in
|
|
||||||
callback(data, displaySources as CFArray, nil)
|
|
||||||
}.catch { error in
|
|
||||||
callback(data, nil, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKLocalTrackPublicationSetMute")
|
|
||||||
public func LKLocalTrackPublicationSetMute(
|
|
||||||
publication: UnsafeRawPointer,
|
|
||||||
muted: Bool,
|
|
||||||
on_complete: @escaping @convention(c) (UnsafeRawPointer, CFString?) -> Void,
|
|
||||||
callback_data: UnsafeRawPointer
|
|
||||||
) {
|
|
||||||
let publication = Unmanaged<LocalTrackPublication>.fromOpaque(publication).takeUnretainedValue()
|
|
||||||
|
|
||||||
if muted {
|
|
||||||
publication.mute().then {
|
|
||||||
on_complete(callback_data, nil)
|
|
||||||
}.catch { error in
|
|
||||||
on_complete(callback_data, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
publication.unmute().then {
|
|
||||||
on_complete(callback_data, nil)
|
|
||||||
}.catch { error in
|
|
||||||
on_complete(callback_data, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRemoteTrackPublicationSetEnabled")
|
|
||||||
public func LKRemoteTrackPublicationSetEnabled(
|
|
||||||
publication: UnsafeRawPointer,
|
|
||||||
enabled: Bool,
|
|
||||||
on_complete: @escaping @convention(c) (UnsafeRawPointer, CFString?) -> Void,
|
|
||||||
callback_data: UnsafeRawPointer
|
|
||||||
) {
|
|
||||||
let publication = Unmanaged<RemoteTrackPublication>.fromOpaque(publication).takeUnretainedValue()
|
|
||||||
|
|
||||||
publication.set(enabled: enabled).then {
|
|
||||||
on_complete(callback_data, nil)
|
|
||||||
}.catch { error in
|
|
||||||
on_complete(callback_data, error.localizedDescription as CFString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRemoteTrackPublicationIsMuted")
|
|
||||||
public func LKRemoteTrackPublicationIsMuted(
|
|
||||||
publication: UnsafeRawPointer
|
|
||||||
) -> Bool {
|
|
||||||
let publication = Unmanaged<RemoteTrackPublication>.fromOpaque(publication).takeUnretainedValue()
|
|
||||||
|
|
||||||
return publication.muted
|
|
||||||
}
|
|
||||||
|
|
||||||
@_cdecl("LKRemoteTrackPublicationGetSid")
|
|
||||||
public func LKRemoteTrackPublicationGetSid(
|
|
||||||
publication: UnsafeRawPointer
|
|
||||||
) -> CFString {
|
|
||||||
let publication = Unmanaged<RemoteTrackPublication>.fromOpaque(publication).takeUnretainedValue()
|
|
||||||
|
|
||||||
return publication.sid as CFString
|
|
||||||
}
|
|
|
@ -1,182 +0,0 @@
|
||||||
use serde::Deserialize;
|
|
||||||
use std::{
|
|
||||||
env,
|
|
||||||
path::{Path, PathBuf},
|
|
||||||
process::Command,
|
|
||||||
};
|
|
||||||
|
|
||||||
const SWIFT_PACKAGE_NAME: &str = "LiveKitBridge2";
|
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
pub struct SwiftTargetInfo {
|
|
||||||
pub triple: String,
|
|
||||||
pub unversioned_triple: String,
|
|
||||||
pub module_triple: String,
|
|
||||||
pub swift_runtime_compatibility_version: String,
|
|
||||||
#[serde(rename = "librariesRequireRPath")]
|
|
||||||
pub libraries_require_rpath: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
pub struct SwiftPaths {
|
|
||||||
pub runtime_library_paths: Vec<String>,
|
|
||||||
pub runtime_library_import_paths: Vec<String>,
|
|
||||||
pub runtime_resource_path: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
|
||||||
pub struct SwiftTarget {
|
|
||||||
pub target: SwiftTargetInfo,
|
|
||||||
pub paths: SwiftPaths,
|
|
||||||
}
|
|
||||||
|
|
||||||
const MACOS_TARGET_VERSION: &str = "10.15.7";
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
if cfg!(not(any(test, feature = "test-support"))) {
|
|
||||||
let swift_target = get_swift_target();
|
|
||||||
|
|
||||||
build_bridge(&swift_target);
|
|
||||||
link_swift_stdlib(&swift_target);
|
|
||||||
link_webrtc_framework(&swift_target);
|
|
||||||
|
|
||||||
// Register exported Objective-C selectors, protocols, etc when building example binaries.
|
|
||||||
println!("cargo:rustc-link-arg=-Wl,-ObjC");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_bridge(swift_target: &SwiftTarget) {
|
|
||||||
println!("cargo:rerun-if-env-changed=MACOSX_DEPLOYMENT_TARGET");
|
|
||||||
println!("cargo:rerun-if-changed={}/Sources", SWIFT_PACKAGE_NAME);
|
|
||||||
println!(
|
|
||||||
"cargo:rerun-if-changed={}/Package.swift",
|
|
||||||
SWIFT_PACKAGE_NAME
|
|
||||||
);
|
|
||||||
println!(
|
|
||||||
"cargo:rerun-if-changed={}/Package.resolved",
|
|
||||||
SWIFT_PACKAGE_NAME
|
|
||||||
);
|
|
||||||
|
|
||||||
let swift_package_root = swift_package_root();
|
|
||||||
let swift_target_folder = swift_target_folder();
|
|
||||||
let swift_cache_folder = swift_cache_folder();
|
|
||||||
if !Command::new("swift")
|
|
||||||
.arg("build")
|
|
||||||
.arg("--disable-automatic-resolution")
|
|
||||||
.args(["--configuration", &env::var("PROFILE").unwrap()])
|
|
||||||
.args(["--triple", &swift_target.target.triple])
|
|
||||||
.args(["--build-path".into(), swift_target_folder])
|
|
||||||
.args(["--cache-path".into(), swift_cache_folder])
|
|
||||||
.current_dir(&swift_package_root)
|
|
||||||
.status()
|
|
||||||
.unwrap()
|
|
||||||
.success()
|
|
||||||
{
|
|
||||||
panic!(
|
|
||||||
"Failed to compile swift package in {}",
|
|
||||||
swift_package_root.display()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
println!(
|
|
||||||
"cargo:rustc-link-search=native={}",
|
|
||||||
swift_target.out_dir_path().display()
|
|
||||||
);
|
|
||||||
println!("cargo:rustc-link-lib=static={}", SWIFT_PACKAGE_NAME);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn link_swift_stdlib(swift_target: &SwiftTarget) {
|
|
||||||
for path in &swift_target.paths.runtime_library_paths {
|
|
||||||
println!("cargo:rustc-link-search=native={}", path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn link_webrtc_framework(swift_target: &SwiftTarget) {
|
|
||||||
let swift_out_dir_path = swift_target.out_dir_path();
|
|
||||||
println!("cargo:rustc-link-lib=framework=WebRTC");
|
|
||||||
println!(
|
|
||||||
"cargo:rustc-link-search=framework={}",
|
|
||||||
swift_out_dir_path.display()
|
|
||||||
);
|
|
||||||
// Find WebRTC.framework as a sibling of the executable when running tests.
|
|
||||||
println!("cargo:rustc-link-arg=-Wl,-rpath,@executable_path");
|
|
||||||
// Find WebRTC.framework in parent directory of the executable when running examples.
|
|
||||||
println!("cargo:rustc-link-arg=-Wl,-rpath,@executable_path/..");
|
|
||||||
|
|
||||||
let source_path = swift_out_dir_path.join("WebRTC.framework");
|
|
||||||
let deps_dir_path =
|
|
||||||
PathBuf::from(env::var("OUT_DIR").unwrap()).join("../../../deps/WebRTC.framework");
|
|
||||||
let target_dir_path =
|
|
||||||
PathBuf::from(env::var("OUT_DIR").unwrap()).join("../../../WebRTC.framework");
|
|
||||||
copy_dir(&source_path, &deps_dir_path);
|
|
||||||
copy_dir(&source_path, &target_dir_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_swift_target() -> SwiftTarget {
|
|
||||||
let mut arch = env::var("CARGO_CFG_TARGET_ARCH").unwrap();
|
|
||||||
if arch == "aarch64" {
|
|
||||||
arch = "arm64".into();
|
|
||||||
}
|
|
||||||
let target = format!("{}-apple-macosx{}", arch, MACOS_TARGET_VERSION);
|
|
||||||
|
|
||||||
let swift_target_info_str = Command::new("swift")
|
|
||||||
.args(["-target", &target, "-print-target-info"])
|
|
||||||
.output()
|
|
||||||
.unwrap()
|
|
||||||
.stdout;
|
|
||||||
|
|
||||||
serde_json::from_slice(&swift_target_info_str).unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn swift_package_root() -> PathBuf {
|
|
||||||
env::current_dir().unwrap().join(SWIFT_PACKAGE_NAME)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn swift_target_folder() -> PathBuf {
|
|
||||||
let target = env::var("TARGET").unwrap();
|
|
||||||
env::current_dir()
|
|
||||||
.unwrap()
|
|
||||||
.join(format!("../../target/{target}/{SWIFT_PACKAGE_NAME}_target"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn swift_cache_folder() -> PathBuf {
|
|
||||||
let target = env::var("TARGET").unwrap();
|
|
||||||
env::current_dir()
|
|
||||||
.unwrap()
|
|
||||||
.join(format!("../../target/{target}/{SWIFT_PACKAGE_NAME}_cache"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn copy_dir(source: &Path, destination: &Path) {
|
|
||||||
assert!(
|
|
||||||
Command::new("rm")
|
|
||||||
.arg("-rf")
|
|
||||||
.arg(destination)
|
|
||||||
.status()
|
|
||||||
.unwrap()
|
|
||||||
.success(),
|
|
||||||
"could not remove {:?} before copying",
|
|
||||||
destination
|
|
||||||
);
|
|
||||||
|
|
||||||
assert!(
|
|
||||||
Command::new("cp")
|
|
||||||
.arg("-R")
|
|
||||||
.args([source, destination])
|
|
||||||
.status()
|
|
||||||
.unwrap()
|
|
||||||
.success(),
|
|
||||||
"could not copy {:?} to {:?}",
|
|
||||||
source,
|
|
||||||
destination
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SwiftTarget {
|
|
||||||
fn out_dir_path(&self) -> PathBuf {
|
|
||||||
swift_target_folder()
|
|
||||||
.join(&self.target.unversioned_triple)
|
|
||||||
.join(env::var("PROFILE").unwrap())
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,176 +0,0 @@
|
||||||
use std::{sync::Arc, time::Duration};
|
|
||||||
|
|
||||||
use futures::StreamExt;
|
|
||||||
use gpui::{actions, KeyBinding};
|
|
||||||
use live_kit_client2::{
|
|
||||||
LocalAudioTrack, LocalVideoTrack, RemoteAudioTrackUpdate, RemoteVideoTrackUpdate, Room,
|
|
||||||
};
|
|
||||||
use live_kit_server::token::{self, VideoGrant};
|
|
||||||
use log::LevelFilter;
|
|
||||||
use simplelog::SimpleLogger;
|
|
||||||
|
|
||||||
actions!(live_kit_client, [Quit]);
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
SimpleLogger::init(LevelFilter::Info, Default::default()).expect("could not initialize logger");
|
|
||||||
|
|
||||||
gpui::App::production(Arc::new(())).run(|cx| {
|
|
||||||
#[cfg(any(test, feature = "test-support"))]
|
|
||||||
println!("USING TEST LIVEKIT");
|
|
||||||
|
|
||||||
#[cfg(not(any(test, feature = "test-support")))]
|
|
||||||
println!("USING REAL LIVEKIT");
|
|
||||||
|
|
||||||
cx.activate(true);
|
|
||||||
|
|
||||||
cx.on_action(quit);
|
|
||||||
cx.bind_keys([KeyBinding::new("cmd-q", Quit, None)]);
|
|
||||||
|
|
||||||
// todo!()
|
|
||||||
// cx.set_menus(vec![Menu {
|
|
||||||
// name: "Zed",
|
|
||||||
// items: vec![MenuItem::Action {
|
|
||||||
// name: "Quit",
|
|
||||||
// action: Box::new(Quit),
|
|
||||||
// os_action: None,
|
|
||||||
// }],
|
|
||||||
// }]);
|
|
||||||
|
|
||||||
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();
|
|
||||||
|
|
||||||
let mut audio_track_updates = room_b.remote_audio_track_updates();
|
|
||||||
let audio_track = LocalAudioTrack::create();
|
|
||||||
let audio_track_publication = room_a.publish_audio_track(audio_track).await.unwrap();
|
|
||||||
|
|
||||||
if let RemoteAudioTrackUpdate::Subscribed(track, _) =
|
|
||||||
audio_track_updates.next().await.unwrap()
|
|
||||||
{
|
|
||||||
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");
|
|
||||||
}
|
|
||||||
|
|
||||||
audio_track_publication.set_mute(true).await.unwrap();
|
|
||||||
|
|
||||||
println!("waiting for mute changed!");
|
|
||||||
if let RemoteAudioTrackUpdate::MuteChanged { track_id, muted } =
|
|
||||||
audio_track_updates.next().await.unwrap()
|
|
||||||
{
|
|
||||||
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();
|
|
||||||
|
|
||||||
if let RemoteAudioTrackUpdate::MuteChanged { track_id, muted } =
|
|
||||||
audio_track_updates.next().await.unwrap()
|
|
||||||
{
|
|
||||||
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");
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("Pausing for 5 seconds to test audio, make some noise!");
|
|
||||||
let timer = cx.background_executor().timer(Duration::from_secs(5));
|
|
||||||
timer.await;
|
|
||||||
let remote_audio_track = room_b
|
|
||||||
.remote_audio_tracks("test-participant-1")
|
|
||||||
.pop()
|
|
||||||
.unwrap();
|
|
||||||
room_a.unpublish_track(audio_track_publication);
|
|
||||||
|
|
||||||
// Clear out any active speakers changed messages
|
|
||||||
let mut next = audio_track_updates.next().await.unwrap();
|
|
||||||
while let RemoteAudioTrackUpdate::ActiveSpeakersChanged { speakers } = next {
|
|
||||||
println!("Speakers changed: {:?}", speakers);
|
|
||||||
next = audio_track_updates.next().await.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
if let RemoteAudioTrackUpdate::Unsubscribed {
|
|
||||||
publisher_id,
|
|
||||||
track_id,
|
|
||||||
} = next
|
|
||||||
{
|
|
||||||
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");
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut video_track_updates = room_b.remote_video_track_updates();
|
|
||||||
let displays = room_a.display_sources().await.unwrap();
|
|
||||||
let display = displays.into_iter().next().unwrap();
|
|
||||||
|
|
||||||
let local_video_track = LocalVideoTrack::screen_share_for_display(&display);
|
|
||||||
let local_video_track_publication =
|
|
||||||
room_a.publish_video_track(local_video_track).await.unwrap();
|
|
||||||
|
|
||||||
if let RemoteVideoTrackUpdate::Subscribed(track) =
|
|
||||||
video_track_updates.next().await.unwrap()
|
|
||||||
{
|
|
||||||
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");
|
|
||||||
assert_eq!(track.publisher_id(), "test-participant-1");
|
|
||||||
} else {
|
|
||||||
panic!("unexpected message");
|
|
||||||
}
|
|
||||||
|
|
||||||
let remote_video_track = room_b
|
|
||||||
.remote_video_tracks("test-participant-1")
|
|
||||||
.pop()
|
|
||||||
.unwrap();
|
|
||||||
room_a.unpublish_track(local_video_track_publication);
|
|
||||||
if let RemoteVideoTrackUpdate::Unsubscribed {
|
|
||||||
publisher_id,
|
|
||||||
track_id,
|
|
||||||
} = video_track_updates.next().await.unwrap()
|
|
||||||
{
|
|
||||||
assert_eq!(publisher_id, "test-participant-1");
|
|
||||||
assert_eq!(remote_video_track.sid(), track_id);
|
|
||||||
assert_eq!(room_b.remote_video_tracks("test-participant-1").len(), 0);
|
|
||||||
} else {
|
|
||||||
panic!("unexpected message");
|
|
||||||
}
|
|
||||||
|
|
||||||
cx.update(|cx| cx.shutdown()).ok();
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn quit(_: &Quit, cx: &mut gpui::AppContext) {
|
|
||||||
cx.quit();
|
|
||||||
}
|
|
|
@ -1,11 +0,0 @@
|
||||||
#[cfg(not(any(test, feature = "test-support")))]
|
|
||||||
pub mod prod;
|
|
||||||
|
|
||||||
#[cfg(not(any(test, feature = "test-support")))]
|
|
||||||
pub use prod::*;
|
|
||||||
|
|
||||||
#[cfg(any(test, feature = "test-support"))]
|
|
||||||
pub mod test;
|
|
||||||
|
|
||||||
#[cfg(any(test, feature = "test-support"))]
|
|
||||||
pub use test::*;
|
|
|
@ -1,947 +0,0 @@
|
||||||
use anyhow::{anyhow, Context, Result};
|
|
||||||
use core_foundation::{
|
|
||||||
array::{CFArray, CFArrayRef},
|
|
||||||
base::{CFRelease, CFRetain, TCFType},
|
|
||||||
string::{CFString, CFStringRef},
|
|
||||||
};
|
|
||||||
use futures::{
|
|
||||||
channel::{mpsc, oneshot},
|
|
||||||
Future,
|
|
||||||
};
|
|
||||||
pub use media::core_video::CVImageBuffer;
|
|
||||||
use media::core_video::CVImageBufferRef;
|
|
||||||
use parking_lot::Mutex;
|
|
||||||
use postage::watch;
|
|
||||||
use std::{
|
|
||||||
ffi::c_void,
|
|
||||||
sync::{Arc, Weak},
|
|
||||||
};
|
|
||||||
|
|
||||||
// SAFETY: Most live kit types are threadsafe:
|
|
||||||
// https://github.com/livekit/client-sdk-swift#thread-safety
|
|
||||||
macro_rules! pointer_type {
|
|
||||||
($pointer_name:ident) => {
|
|
||||||
#[repr(transparent)]
|
|
||||||
#[derive(Copy, Clone, Debug)]
|
|
||||||
pub struct $pointer_name(pub *const std::ffi::c_void);
|
|
||||||
unsafe impl Send for $pointer_name {}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
mod swift {
|
|
||||||
pointer_type!(Room);
|
|
||||||
pointer_type!(LocalAudioTrack);
|
|
||||||
pointer_type!(RemoteAudioTrack);
|
|
||||||
pointer_type!(LocalVideoTrack);
|
|
||||||
pointer_type!(RemoteVideoTrack);
|
|
||||||
pointer_type!(LocalTrackPublication);
|
|
||||||
pointer_type!(RemoteTrackPublication);
|
|
||||||
pointer_type!(MacOSDisplay);
|
|
||||||
pointer_type!(RoomDelegate);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" {
|
|
||||||
fn LKRoomDelegateCreate(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
on_did_disconnect: extern "C" fn(callback_data: *mut c_void),
|
|
||||||
on_did_subscribe_to_remote_audio_track: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
remote_track: swift::RemoteAudioTrack,
|
|
||||||
remote_publication: swift::RemoteTrackPublication,
|
|
||||||
),
|
|
||||||
on_did_unsubscribe_from_remote_audio_track: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
),
|
|
||||||
on_mute_changed_from_remote_audio_track: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
muted: bool,
|
|
||||||
),
|
|
||||||
on_active_speakers_changed: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
participants: CFArrayRef,
|
|
||||||
),
|
|
||||||
on_did_subscribe_to_remote_video_track: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
remote_track: swift::RemoteVideoTrack,
|
|
||||||
),
|
|
||||||
on_did_unsubscribe_from_remote_video_track: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
),
|
|
||||||
) -> swift::RoomDelegate;
|
|
||||||
|
|
||||||
fn LKRoomCreate(delegate: swift::RoomDelegate) -> swift::Room;
|
|
||||||
fn LKRoomConnect(
|
|
||||||
room: swift::Room,
|
|
||||||
url: CFStringRef,
|
|
||||||
token: CFStringRef,
|
|
||||||
callback: extern "C" fn(*mut c_void, CFStringRef),
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
);
|
|
||||||
fn LKRoomDisconnect(room: swift::Room);
|
|
||||||
fn LKRoomPublishVideoTrack(
|
|
||||||
room: swift::Room,
|
|
||||||
track: swift::LocalVideoTrack,
|
|
||||||
callback: extern "C" fn(*mut c_void, swift::LocalTrackPublication, CFStringRef),
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
);
|
|
||||||
fn LKRoomPublishAudioTrack(
|
|
||||||
room: swift::Room,
|
|
||||||
track: swift::LocalAudioTrack,
|
|
||||||
callback: extern "C" fn(*mut c_void, swift::LocalTrackPublication, CFStringRef),
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
);
|
|
||||||
fn LKRoomUnpublishTrack(room: swift::Room, publication: swift::LocalTrackPublication);
|
|
||||||
|
|
||||||
fn LKRoomAudioTracksForRemoteParticipant(
|
|
||||||
room: swift::Room,
|
|
||||||
participant_id: CFStringRef,
|
|
||||||
) -> CFArrayRef;
|
|
||||||
|
|
||||||
fn LKRoomAudioTrackPublicationsForRemoteParticipant(
|
|
||||||
room: swift::Room,
|
|
||||||
participant_id: CFStringRef,
|
|
||||||
) -> CFArrayRef;
|
|
||||||
|
|
||||||
fn LKRoomVideoTracksForRemoteParticipant(
|
|
||||||
room: swift::Room,
|
|
||||||
participant_id: CFStringRef,
|
|
||||||
) -> CFArrayRef;
|
|
||||||
|
|
||||||
fn LKVideoRendererCreate(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
on_frame: extern "C" fn(callback_data: *mut c_void, frame: CVImageBufferRef) -> bool,
|
|
||||||
on_drop: extern "C" fn(callback_data: *mut c_void),
|
|
||||||
) -> *const c_void;
|
|
||||||
|
|
||||||
fn LKRemoteAudioTrackGetSid(track: swift::RemoteAudioTrack) -> CFStringRef;
|
|
||||||
fn LKVideoTrackAddRenderer(track: swift::RemoteVideoTrack, renderer: *const c_void);
|
|
||||||
fn LKRemoteVideoTrackGetSid(track: swift::RemoteVideoTrack) -> CFStringRef;
|
|
||||||
|
|
||||||
fn LKDisplaySources(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
callback: extern "C" fn(
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
sources: CFArrayRef,
|
|
||||||
error: CFStringRef,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
fn LKCreateScreenShareTrackForDisplay(display: swift::MacOSDisplay) -> swift::LocalVideoTrack;
|
|
||||||
fn LKLocalAudioTrackCreateTrack() -> swift::LocalAudioTrack;
|
|
||||||
|
|
||||||
fn LKLocalTrackPublicationSetMute(
|
|
||||||
publication: swift::LocalTrackPublication,
|
|
||||||
muted: bool,
|
|
||||||
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
);
|
|
||||||
|
|
||||||
fn LKRemoteTrackPublicationSetEnabled(
|
|
||||||
publication: swift::RemoteTrackPublication,
|
|
||||||
enabled: bool,
|
|
||||||
on_complete: extern "C" fn(callback_data: *mut c_void, error: CFStringRef),
|
|
||||||
callback_data: *mut c_void,
|
|
||||||
);
|
|
||||||
|
|
||||||
fn LKRemoteTrackPublicationIsMuted(publication: swift::RemoteTrackPublication) -> bool;
|
|
||||||
fn LKRemoteTrackPublicationGetSid(publication: swift::RemoteTrackPublication) -> CFStringRef;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub type Sid = String;
|
|
||||||
|
|
||||||
#[derive(Clone, Eq, PartialEq)]
|
|
||||||
pub enum ConnectionState {
|
|
||||||
Disconnected,
|
|
||||||
Connected { url: String, token: String },
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Room {
|
|
||||||
native_room: Mutex<swift::Room>,
|
|
||||||
connection: Mutex<(
|
|
||||||
watch::Sender<ConnectionState>,
|
|
||||||
watch::Receiver<ConnectionState>,
|
|
||||||
)>,
|
|
||||||
remote_audio_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteAudioTrackUpdate>>>,
|
|
||||||
remote_video_track_subscribers: Mutex<Vec<mpsc::UnboundedSender<RemoteVideoTrackUpdate>>>,
|
|
||||||
_delegate: Mutex<RoomDelegate>,
|
|
||||||
}
|
|
||||||
|
|
||||||
trait AssertSendSync: Send {}
|
|
||||||
impl AssertSendSync for Room {}
|
|
||||||
|
|
||||||
impl Room {
|
|
||||||
pub fn new() -> Arc<Self> {
|
|
||||||
Arc::new_cyclic(|weak_room| {
|
|
||||||
let delegate = RoomDelegate::new(weak_room.clone());
|
|
||||||
Self {
|
|
||||||
native_room: Mutex::new(unsafe { LKRoomCreate(delegate.native_delegate) }),
|
|
||||||
connection: Mutex::new(watch::channel_with(ConnectionState::Disconnected)),
|
|
||||||
remote_audio_track_subscribers: Default::default(),
|
|
||||||
remote_video_track_subscribers: Default::default(),
|
|
||||||
_delegate: Mutex::new(delegate),
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn status(&self) -> watch::Receiver<ConnectionState> {
|
|
||||||
self.connection.lock().1.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn connect(self: &Arc<Self>, url: &str, token: &str) -> impl Future<Output = Result<()>> {
|
|
||||||
let url = CFString::new(url);
|
|
||||||
let token = CFString::new(token);
|
|
||||||
let (did_connect, tx, rx) = Self::build_done_callback();
|
|
||||||
unsafe {
|
|
||||||
LKRoomConnect(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
url.as_concrete_TypeRef(),
|
|
||||||
token.as_concrete_TypeRef(),
|
|
||||||
did_connect,
|
|
||||||
tx,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
let this = self.clone();
|
|
||||||
let url = url.to_string();
|
|
||||||
let token = token.to_string();
|
|
||||||
async move {
|
|
||||||
rx.await.unwrap().context("error connecting to room")?;
|
|
||||||
*this.connection.lock().0.borrow_mut() = ConnectionState::Connected { url, token };
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn did_disconnect(&self) {
|
|
||||||
*self.connection.lock().0.borrow_mut() = ConnectionState::Disconnected;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn display_sources(self: &Arc<Self>) -> impl Future<Output = Result<Vec<MacOSDisplay>>> {
|
|
||||||
extern "C" fn callback(tx: *mut c_void, sources: CFArrayRef, error: CFStringRef) {
|
|
||||||
unsafe {
|
|
||||||
let tx = Box::from_raw(tx as *mut oneshot::Sender<Result<Vec<MacOSDisplay>>>);
|
|
||||||
|
|
||||||
if sources.is_null() {
|
|
||||||
let _ = tx.send(Err(anyhow!("{}", CFString::wrap_under_get_rule(error))));
|
|
||||||
} else {
|
|
||||||
let sources = CFArray::wrap_under_get_rule(sources)
|
|
||||||
.into_iter()
|
|
||||||
.map(|source| MacOSDisplay::new(swift::MacOSDisplay(*source)))
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let _ = tx.send(Ok(sources));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let (tx, rx) = oneshot::channel();
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
LKDisplaySources(Box::into_raw(Box::new(tx)) as *mut _, callback);
|
|
||||||
}
|
|
||||||
|
|
||||||
async move { rx.await.unwrap() }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publish_video_track(
|
|
||||||
self: &Arc<Self>,
|
|
||||||
track: LocalVideoTrack,
|
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
|
||||||
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
|
||||||
extern "C" fn callback(
|
|
||||||
tx: *mut c_void,
|
|
||||||
publication: swift::LocalTrackPublication,
|
|
||||||
error: CFStringRef,
|
|
||||||
) {
|
|
||||||
let tx =
|
|
||||||
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
|
||||||
if error.is_null() {
|
|
||||||
let _ = tx.send(Ok(LocalTrackPublication::new(publication)));
|
|
||||||
} else {
|
|
||||||
let error = unsafe { CFString::wrap_under_get_rule(error).to_string() };
|
|
||||||
let _ = tx.send(Err(anyhow!(error)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
unsafe {
|
|
||||||
LKRoomPublishVideoTrack(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
track.0,
|
|
||||||
callback,
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
async { rx.await.unwrap().context("error publishing video track") }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publish_audio_track(
|
|
||||||
self: &Arc<Self>,
|
|
||||||
track: LocalAudioTrack,
|
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
|
||||||
let (tx, rx) = oneshot::channel::<Result<LocalTrackPublication>>();
|
|
||||||
extern "C" fn callback(
|
|
||||||
tx: *mut c_void,
|
|
||||||
publication: swift::LocalTrackPublication,
|
|
||||||
error: CFStringRef,
|
|
||||||
) {
|
|
||||||
let tx =
|
|
||||||
unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<LocalTrackPublication>>) };
|
|
||||||
if error.is_null() {
|
|
||||||
let _ = tx.send(Ok(LocalTrackPublication::new(publication)));
|
|
||||||
} else {
|
|
||||||
let error = unsafe { CFString::wrap_under_get_rule(error).to_string() };
|
|
||||||
let _ = tx.send(Err(anyhow!(error)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
unsafe {
|
|
||||||
LKRoomPublishAudioTrack(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
track.0,
|
|
||||||
callback,
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
async { rx.await.unwrap().context("error publishing audio track") }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn unpublish_track(&self, publication: LocalTrackPublication) {
|
|
||||||
unsafe {
|
|
||||||
LKRoomUnpublishTrack(*self.native_room.lock(), publication.0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_video_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteVideoTrack>> {
|
|
||||||
unsafe {
|
|
||||||
let tracks = LKRoomVideoTracksForRemoteParticipant(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if tracks.is_null() {
|
|
||||||
Vec::new()
|
|
||||||
} else {
|
|
||||||
let tracks = CFArray::wrap_under_get_rule(tracks);
|
|
||||||
tracks
|
|
||||||
.into_iter()
|
|
||||||
.map(|native_track| {
|
|
||||||
let native_track = swift::RemoteVideoTrack(*native_track);
|
|
||||||
let id =
|
|
||||||
CFString::wrap_under_get_rule(LKRemoteVideoTrackGetSid(native_track))
|
|
||||||
.to_string();
|
|
||||||
Arc::new(RemoteVideoTrack::new(
|
|
||||||
native_track,
|
|
||||||
id,
|
|
||||||
participant_id.into(),
|
|
||||||
))
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_audio_tracks(&self, participant_id: &str) -> Vec<Arc<RemoteAudioTrack>> {
|
|
||||||
unsafe {
|
|
||||||
let tracks = LKRoomAudioTracksForRemoteParticipant(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if tracks.is_null() {
|
|
||||||
Vec::new()
|
|
||||||
} else {
|
|
||||||
let tracks = CFArray::wrap_under_get_rule(tracks);
|
|
||||||
tracks
|
|
||||||
.into_iter()
|
|
||||||
.map(|native_track| {
|
|
||||||
let native_track = swift::RemoteAudioTrack(*native_track);
|
|
||||||
let id =
|
|
||||||
CFString::wrap_under_get_rule(LKRemoteAudioTrackGetSid(native_track))
|
|
||||||
.to_string();
|
|
||||||
Arc::new(RemoteAudioTrack::new(
|
|
||||||
native_track,
|
|
||||||
id,
|
|
||||||
participant_id.into(),
|
|
||||||
))
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_audio_track_publications(
|
|
||||||
&self,
|
|
||||||
participant_id: &str,
|
|
||||||
) -> Vec<Arc<RemoteTrackPublication>> {
|
|
||||||
unsafe {
|
|
||||||
let tracks = LKRoomAudioTrackPublicationsForRemoteParticipant(
|
|
||||||
*self.native_room.lock(),
|
|
||||||
CFString::new(participant_id).as_concrete_TypeRef(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if tracks.is_null() {
|
|
||||||
Vec::new()
|
|
||||||
} else {
|
|
||||||
let tracks = CFArray::wrap_under_get_rule(tracks);
|
|
||||||
tracks
|
|
||||||
.into_iter()
|
|
||||||
.map(|native_track_publication| {
|
|
||||||
let native_track_publication =
|
|
||||||
swift::RemoteTrackPublication(*native_track_publication);
|
|
||||||
Arc::new(RemoteTrackPublication::new(native_track_publication))
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_audio_track_updates(&self) -> mpsc::UnboundedReceiver<RemoteAudioTrackUpdate> {
|
|
||||||
let (tx, rx) = mpsc::unbounded();
|
|
||||||
self.remote_audio_track_subscribers.lock().push(tx);
|
|
||||||
rx
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_video_track_updates(&self) -> mpsc::UnboundedReceiver<RemoteVideoTrackUpdate> {
|
|
||||||
let (tx, rx) = mpsc::unbounded();
|
|
||||||
self.remote_video_track_subscribers.lock().push(tx);
|
|
||||||
rx
|
|
||||||
}
|
|
||||||
|
|
||||||
fn did_subscribe_to_remote_audio_track(
|
|
||||||
&self,
|
|
||||||
track: RemoteAudioTrack,
|
|
||||||
publication: RemoteTrackPublication,
|
|
||||||
) {
|
|
||||||
let track = Arc::new(track);
|
|
||||||
let publication = Arc::new(publication);
|
|
||||||
self.remote_audio_track_subscribers.lock().retain(|tx| {
|
|
||||||
tx.unbounded_send(RemoteAudioTrackUpdate::Subscribed(
|
|
||||||
track.clone(),
|
|
||||||
publication.clone(),
|
|
||||||
))
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn did_unsubscribe_from_remote_audio_track(&self, publisher_id: String, track_id: String) {
|
|
||||||
self.remote_audio_track_subscribers.lock().retain(|tx| {
|
|
||||||
tx.unbounded_send(RemoteAudioTrackUpdate::Unsubscribed {
|
|
||||||
publisher_id: publisher_id.clone(),
|
|
||||||
track_id: track_id.clone(),
|
|
||||||
})
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn mute_changed_from_remote_audio_track(&self, track_id: String, muted: bool) {
|
|
||||||
self.remote_audio_track_subscribers.lock().retain(|tx| {
|
|
||||||
tx.unbounded_send(RemoteAudioTrackUpdate::MuteChanged {
|
|
||||||
track_id: track_id.clone(),
|
|
||||||
muted,
|
|
||||||
})
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// A vec of publisher IDs
|
|
||||||
fn active_speakers_changed(&self, speakers: Vec<String>) {
|
|
||||||
self.remote_audio_track_subscribers
|
|
||||||
.lock()
|
|
||||||
.retain(move |tx| {
|
|
||||||
tx.unbounded_send(RemoteAudioTrackUpdate::ActiveSpeakersChanged {
|
|
||||||
speakers: speakers.clone(),
|
|
||||||
})
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn did_subscribe_to_remote_video_track(&self, track: RemoteVideoTrack) {
|
|
||||||
let track = Arc::new(track);
|
|
||||||
self.remote_video_track_subscribers.lock().retain(|tx| {
|
|
||||||
tx.unbounded_send(RemoteVideoTrackUpdate::Subscribed(track.clone()))
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn did_unsubscribe_from_remote_video_track(&self, publisher_id: String, track_id: String) {
|
|
||||||
self.remote_video_track_subscribers.lock().retain(|tx| {
|
|
||||||
tx.unbounded_send(RemoteVideoTrackUpdate::Unsubscribed {
|
|
||||||
publisher_id: publisher_id.clone(),
|
|
||||||
track_id: track_id.clone(),
|
|
||||||
})
|
|
||||||
.is_ok()
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_done_callback() -> (
|
|
||||||
extern "C" fn(*mut c_void, CFStringRef),
|
|
||||||
*mut c_void,
|
|
||||||
oneshot::Receiver<Result<()>>,
|
|
||||||
) {
|
|
||||||
let (tx, rx) = oneshot::channel();
|
|
||||||
extern "C" fn done_callback(tx: *mut c_void, error: CFStringRef) {
|
|
||||||
let tx = unsafe { Box::from_raw(tx as *mut oneshot::Sender<Result<()>>) };
|
|
||||||
if error.is_null() {
|
|
||||||
let _ = tx.send(Ok(()));
|
|
||||||
} else {
|
|
||||||
let error = unsafe { CFString::wrap_under_get_rule(error).to_string() };
|
|
||||||
let _ = tx.send(Err(anyhow!(error)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
(
|
|
||||||
done_callback,
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
rx,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_display_sources(&self, _: Vec<MacOSDisplay>) {
|
|
||||||
unreachable!("This is a test-only function")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for Room {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe {
|
|
||||||
let native_room = &*self.native_room.lock();
|
|
||||||
LKRoomDisconnect(*native_room);
|
|
||||||
CFRelease(native_room.0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct RoomDelegate {
|
|
||||||
native_delegate: swift::RoomDelegate,
|
|
||||||
_weak_room: Weak<Room>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RoomDelegate {
|
|
||||||
fn new(weak_room: Weak<Room>) -> Self {
|
|
||||||
let native_delegate = unsafe {
|
|
||||||
LKRoomDelegateCreate(
|
|
||||||
weak_room.as_ptr() as *mut c_void,
|
|
||||||
Self::on_did_disconnect,
|
|
||||||
Self::on_did_subscribe_to_remote_audio_track,
|
|
||||||
Self::on_did_unsubscribe_from_remote_audio_track,
|
|
||||||
Self::on_mute_change_from_remote_audio_track,
|
|
||||||
Self::on_active_speakers_changed,
|
|
||||||
Self::on_did_subscribe_to_remote_video_track,
|
|
||||||
Self::on_did_unsubscribe_from_remote_video_track,
|
|
||||||
)
|
|
||||||
};
|
|
||||||
Self {
|
|
||||||
native_delegate,
|
|
||||||
_weak_room: weak_room,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_did_disconnect(room: *mut c_void) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.did_disconnect();
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_did_subscribe_to_remote_audio_track(
|
|
||||||
room: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
track: swift::RemoteAudioTrack,
|
|
||||||
publication: swift::RemoteTrackPublication,
|
|
||||||
) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
|
||||||
let track_id = unsafe { CFString::wrap_under_get_rule(track_id).to_string() };
|
|
||||||
let track = RemoteAudioTrack::new(track, track_id, publisher_id);
|
|
||||||
let publication = RemoteTrackPublication::new(publication);
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.did_subscribe_to_remote_audio_track(track, publication);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_did_unsubscribe_from_remote_audio_track(
|
|
||||||
room: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
|
||||||
let track_id = unsafe { CFString::wrap_under_get_rule(track_id).to_string() };
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.did_unsubscribe_from_remote_audio_track(publisher_id, track_id);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_mute_change_from_remote_audio_track(
|
|
||||||
room: *mut c_void,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
muted: bool,
|
|
||||||
) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let track_id = unsafe { CFString::wrap_under_get_rule(track_id).to_string() };
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.mute_changed_from_remote_audio_track(track_id, muted);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_active_speakers_changed(room: *mut c_void, participants: CFArrayRef) {
|
|
||||||
if participants.is_null() {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let speakers = unsafe {
|
|
||||||
CFArray::wrap_under_get_rule(participants)
|
|
||||||
.into_iter()
|
|
||||||
.map(
|
|
||||||
|speaker: core_foundation::base::ItemRef<'_, *const c_void>| {
|
|
||||||
CFString::wrap_under_get_rule(*speaker as CFStringRef).to_string()
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.collect()
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.active_speakers_changed(speakers);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_did_subscribe_to_remote_video_track(
|
|
||||||
room: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
track: swift::RemoteVideoTrack,
|
|
||||||
) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
|
||||||
let track_id = unsafe { CFString::wrap_under_get_rule(track_id).to_string() };
|
|
||||||
let track = RemoteVideoTrack::new(track, track_id, publisher_id);
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.did_subscribe_to_remote_video_track(track);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_did_unsubscribe_from_remote_video_track(
|
|
||||||
room: *mut c_void,
|
|
||||||
publisher_id: CFStringRef,
|
|
||||||
track_id: CFStringRef,
|
|
||||||
) {
|
|
||||||
let room = unsafe { Weak::from_raw(room as *mut Room) };
|
|
||||||
let publisher_id = unsafe { CFString::wrap_under_get_rule(publisher_id).to_string() };
|
|
||||||
let track_id = unsafe { CFString::wrap_under_get_rule(track_id).to_string() };
|
|
||||||
if let Some(room) = room.upgrade() {
|
|
||||||
room.did_unsubscribe_from_remote_video_track(publisher_id, track_id);
|
|
||||||
}
|
|
||||||
let _ = Weak::into_raw(room);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for RoomDelegate {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe {
|
|
||||||
CFRelease(self.native_delegate.0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct LocalAudioTrack(swift::LocalAudioTrack);
|
|
||||||
|
|
||||||
impl LocalAudioTrack {
|
|
||||||
pub fn create() -> Self {
|
|
||||||
Self(unsafe { LKLocalAudioTrackCreateTrack() })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for LocalAudioTrack {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.0 .0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct LocalVideoTrack(swift::LocalVideoTrack);
|
|
||||||
|
|
||||||
impl LocalVideoTrack {
|
|
||||||
pub fn screen_share_for_display(display: &MacOSDisplay) -> Self {
|
|
||||||
Self(unsafe { LKCreateScreenShareTrackForDisplay(display.0) })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for LocalVideoTrack {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.0 .0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct LocalTrackPublication(swift::LocalTrackPublication);
|
|
||||||
|
|
||||||
impl LocalTrackPublication {
|
|
||||||
pub fn new(native_track_publication: swift::LocalTrackPublication) -> Self {
|
|
||||||
unsafe {
|
|
||||||
CFRetain(native_track_publication.0);
|
|
||||||
}
|
|
||||||
Self(native_track_publication)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_mute(&self, muted: bool) -> impl Future<Output = Result<()>> {
|
|
||||||
let (tx, rx) = futures::channel::oneshot::channel();
|
|
||||||
|
|
||||||
extern "C" fn complete_callback(callback_data: *mut c_void, error: CFStringRef) {
|
|
||||||
let tx = unsafe { Box::from_raw(callback_data as *mut oneshot::Sender<Result<()>>) };
|
|
||||||
if error.is_null() {
|
|
||||||
tx.send(Ok(())).ok();
|
|
||||||
} else {
|
|
||||||
let error = unsafe { CFString::wrap_under_get_rule(error).to_string() };
|
|
||||||
tx.send(Err(anyhow!(error))).ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
LKLocalTrackPublicationSetMute(
|
|
||||||
self.0,
|
|
||||||
muted,
|
|
||||||
complete_callback,
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
async move { rx.await.unwrap() }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for LocalTrackPublication {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.0 .0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct RemoteTrackPublication {
|
|
||||||
native_publication: Mutex<swift::RemoteTrackPublication>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RemoteTrackPublication {
|
|
||||||
pub fn new(native_track_publication: swift::RemoteTrackPublication) -> Self {
|
|
||||||
unsafe {
|
|
||||||
CFRetain(native_track_publication.0);
|
|
||||||
}
|
|
||||||
Self {
|
|
||||||
native_publication: Mutex::new(native_track_publication),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sid(&self) -> String {
|
|
||||||
unsafe {
|
|
||||||
CFString::wrap_under_get_rule(LKRemoteTrackPublicationGetSid(
|
|
||||||
*self.native_publication.lock(),
|
|
||||||
))
|
|
||||||
.to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_muted(&self) -> bool {
|
|
||||||
unsafe { LKRemoteTrackPublicationIsMuted(*self.native_publication.lock()) }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_enabled(&self, enabled: bool) -> impl Future<Output = Result<()>> {
|
|
||||||
let (tx, rx) = futures::channel::oneshot::channel();
|
|
||||||
|
|
||||||
extern "C" fn complete_callback(callback_data: *mut c_void, error: CFStringRef) {
|
|
||||||
let tx = unsafe { Box::from_raw(callback_data as *mut oneshot::Sender<Result<()>>) };
|
|
||||||
if error.is_null() {
|
|
||||||
tx.send(Ok(())).ok();
|
|
||||||
} else {
|
|
||||||
let error = unsafe { CFString::wrap_under_get_rule(error).to_string() };
|
|
||||||
tx.send(Err(anyhow!(error))).ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
LKRemoteTrackPublicationSetEnabled(
|
|
||||||
*self.native_publication.lock(),
|
|
||||||
enabled,
|
|
||||||
complete_callback,
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
async move { rx.await.unwrap() }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for RemoteTrackPublication {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease((*self.native_publication.lock()).0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct RemoteAudioTrack {
|
|
||||||
native_track: Mutex<swift::RemoteAudioTrack>,
|
|
||||||
sid: Sid,
|
|
||||||
publisher_id: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RemoteAudioTrack {
|
|
||||||
fn new(native_track: swift::RemoteAudioTrack, sid: Sid, publisher_id: String) -> Self {
|
|
||||||
unsafe {
|
|
||||||
CFRetain(native_track.0);
|
|
||||||
}
|
|
||||||
Self {
|
|
||||||
native_track: Mutex::new(native_track),
|
|
||||||
sid,
|
|
||||||
publisher_id,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sid(&self) -> &str {
|
|
||||||
&self.sid
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publisher_id(&self) -> &str {
|
|
||||||
&self.publisher_id
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn enable(&self) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn disable(&self) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for RemoteAudioTrack {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.native_track.lock().0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct RemoteVideoTrack {
|
|
||||||
native_track: Mutex<swift::RemoteVideoTrack>,
|
|
||||||
sid: Sid,
|
|
||||||
publisher_id: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RemoteVideoTrack {
|
|
||||||
fn new(native_track: swift::RemoteVideoTrack, sid: Sid, publisher_id: String) -> Self {
|
|
||||||
unsafe {
|
|
||||||
CFRetain(native_track.0);
|
|
||||||
}
|
|
||||||
Self {
|
|
||||||
native_track: Mutex::new(native_track),
|
|
||||||
sid,
|
|
||||||
publisher_id,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sid(&self) -> &str {
|
|
||||||
&self.sid
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publisher_id(&self) -> &str {
|
|
||||||
&self.publisher_id
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn frames(&self) -> async_broadcast::Receiver<Frame> {
|
|
||||||
extern "C" fn on_frame(callback_data: *mut c_void, frame: CVImageBufferRef) -> bool {
|
|
||||||
unsafe {
|
|
||||||
let tx = Box::from_raw(callback_data as *mut async_broadcast::Sender<Frame>);
|
|
||||||
let buffer = CVImageBuffer::wrap_under_get_rule(frame);
|
|
||||||
let result = tx.try_broadcast(Frame(buffer));
|
|
||||||
let _ = Box::into_raw(tx);
|
|
||||||
match result {
|
|
||||||
Ok(_) => true,
|
|
||||||
Err(async_broadcast::TrySendError::Closed(_))
|
|
||||||
| Err(async_broadcast::TrySendError::Inactive(_)) => {
|
|
||||||
log::warn!("no active receiver for frame");
|
|
||||||
false
|
|
||||||
}
|
|
||||||
Err(async_broadcast::TrySendError::Full(_)) => {
|
|
||||||
log::warn!("skipping frame as receiver is not keeping up");
|
|
||||||
true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extern "C" fn on_drop(callback_data: *mut c_void) {
|
|
||||||
unsafe {
|
|
||||||
let _ = Box::from_raw(callback_data as *mut async_broadcast::Sender<Frame>);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let (tx, rx) = async_broadcast::broadcast(64);
|
|
||||||
unsafe {
|
|
||||||
let renderer = LKVideoRendererCreate(
|
|
||||||
Box::into_raw(Box::new(tx)) as *mut c_void,
|
|
||||||
on_frame,
|
|
||||||
on_drop,
|
|
||||||
);
|
|
||||||
LKVideoTrackAddRenderer(*self.native_track.lock(), renderer);
|
|
||||||
rx
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for RemoteVideoTrack {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.native_track.lock().0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum RemoteVideoTrackUpdate {
|
|
||||||
Subscribed(Arc<RemoteVideoTrack>),
|
|
||||||
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum RemoteAudioTrackUpdate {
|
|
||||||
ActiveSpeakersChanged { speakers: Vec<Sid> },
|
|
||||||
MuteChanged { track_id: Sid, muted: bool },
|
|
||||||
Subscribed(Arc<RemoteAudioTrack>, Arc<RemoteTrackPublication>),
|
|
||||||
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct MacOSDisplay(swift::MacOSDisplay);
|
|
||||||
|
|
||||||
impl MacOSDisplay {
|
|
||||||
fn new(ptr: swift::MacOSDisplay) -> Self {
|
|
||||||
unsafe {
|
|
||||||
CFRetain(ptr.0);
|
|
||||||
}
|
|
||||||
Self(ptr)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for MacOSDisplay {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
unsafe { CFRelease(self.0 .0) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct Frame(CVImageBuffer);
|
|
||||||
|
|
||||||
impl Frame {
|
|
||||||
pub fn width(&self) -> usize {
|
|
||||||
self.0.width()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn height(&self) -> usize {
|
|
||||||
self.0.height()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn image(&self) -> CVImageBuffer {
|
|
||||||
self.0.clone()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,651 +0,0 @@
|
||||||
use anyhow::{anyhow, Context, Result};
|
|
||||||
use async_trait::async_trait;
|
|
||||||
use collections::{BTreeMap, HashMap};
|
|
||||||
use futures::Stream;
|
|
||||||
use gpui::BackgroundExecutor;
|
|
||||||
use live_kit_server::token;
|
|
||||||
use media::core_video::CVImageBuffer;
|
|
||||||
use parking_lot::Mutex;
|
|
||||||
use postage::watch;
|
|
||||||
use std::{future::Future, mem, sync::Arc};
|
|
||||||
|
|
||||||
static SERVERS: Mutex<BTreeMap<String, Arc<TestServer>>> = Mutex::new(BTreeMap::new());
|
|
||||||
|
|
||||||
pub struct TestServer {
|
|
||||||
pub url: String,
|
|
||||||
pub api_key: String,
|
|
||||||
pub secret_key: String,
|
|
||||||
rooms: Mutex<HashMap<String, TestServerRoom>>,
|
|
||||||
executor: BackgroundExecutor,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TestServer {
|
|
||||||
pub fn create(
|
|
||||||
url: String,
|
|
||||||
api_key: String,
|
|
||||||
secret_key: String,
|
|
||||||
executor: BackgroundExecutor,
|
|
||||||
) -> Result<Arc<TestServer>> {
|
|
||||||
let mut servers = SERVERS.lock();
|
|
||||||
if servers.contains_key(&url) {
|
|
||||||
Err(anyhow!("a server with url {:?} already exists", url))
|
|
||||||
} else {
|
|
||||||
let server = Arc::new(TestServer {
|
|
||||||
url: url.clone(),
|
|
||||||
api_key,
|
|
||||||
secret_key,
|
|
||||||
rooms: Default::default(),
|
|
||||||
executor,
|
|
||||||
});
|
|
||||||
servers.insert(url, server.clone());
|
|
||||||
Ok(server)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get(url: &str) -> Result<Arc<TestServer>> {
|
|
||||||
Ok(SERVERS
|
|
||||||
.lock()
|
|
||||||
.get(url)
|
|
||||||
.ok_or_else(|| anyhow!("no server found for url"))?
|
|
||||||
.clone())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn teardown(&self) -> Result<()> {
|
|
||||||
SERVERS
|
|
||||||
.lock()
|
|
||||||
.remove(&self.url)
|
|
||||||
.ok_or_else(|| anyhow!("server with url {:?} does not exist", self.url))?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn create_api_client(&self) -> TestApiClient {
|
|
||||||
TestApiClient {
|
|
||||||
url: self.url.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn create_room(&self, room: String) -> Result<()> {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
if server_rooms.contains_key(&room) {
|
|
||||||
Err(anyhow!("room {:?} already exists", room))
|
|
||||||
} else {
|
|
||||||
server_rooms.insert(room, Default::default());
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn delete_room(&self, room: String) -> Result<()> {
|
|
||||||
// TODO: clear state associated with all `Room`s.
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
server_rooms
|
|
||||||
.remove(&room)
|
|
||||||
.ok_or_else(|| anyhow!("room {:?} does not exist", room))?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn join_room(&self, token: String, client_room: Arc<Room>) -> Result<()> {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let identity = claims.sub.unwrap().to_string();
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = (*server_rooms).entry(room_name.to_string()).or_default();
|
|
||||||
|
|
||||||
if room.client_rooms.contains_key(&identity) {
|
|
||||||
Err(anyhow!(
|
|
||||||
"{:?} attempted to join room {:?} twice",
|
|
||||||
identity,
|
|
||||||
room_name
|
|
||||||
))
|
|
||||||
} else {
|
|
||||||
for track in &room.video_tracks {
|
|
||||||
client_room
|
|
||||||
.0
|
|
||||||
.lock()
|
|
||||||
.video_track_updates
|
|
||||||
.0
|
|
||||||
.try_broadcast(RemoteVideoTrackUpdate::Subscribed(track.clone()))
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
room.client_rooms.insert(identity, client_room);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn leave_room(&self, token: String) -> Result<()> {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let identity = claims.sub.unwrap().to_string();
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&*room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
room.client_rooms.remove(&identity).ok_or_else(|| {
|
|
||||||
anyhow!(
|
|
||||||
"{:?} attempted to leave room {:?} before joining it",
|
|
||||||
identity,
|
|
||||||
room_name
|
|
||||||
)
|
|
||||||
})?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn remove_participant(&self, room_name: String, identity: String) -> Result<()> {
|
|
||||||
// TODO: clear state associated with the `Room`.
|
|
||||||
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
room.client_rooms.remove(&identity).ok_or_else(|| {
|
|
||||||
anyhow!(
|
|
||||||
"participant {:?} did not join room {:?}",
|
|
||||||
identity,
|
|
||||||
room_name
|
|
||||||
)
|
|
||||||
})?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn disconnect_client(&self, client_identity: String) {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
for room in server_rooms.values_mut() {
|
|
||||||
if let Some(room) = room.client_rooms.remove(&client_identity) {
|
|
||||||
*room.0.lock().connection.0.borrow_mut() = ConnectionState::Disconnected;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn publish_video_track(&self, token: String, local_track: LocalVideoTrack) -> Result<()> {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let identity = claims.sub.unwrap().to_string();
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&*room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
|
|
||||||
let track = Arc::new(RemoteVideoTrack {
|
|
||||||
sid: nanoid::nanoid!(17),
|
|
||||||
publisher_id: identity.clone(),
|
|
||||||
frames_rx: local_track.frames_rx.clone(),
|
|
||||||
});
|
|
||||||
|
|
||||||
room.video_tracks.push(track.clone());
|
|
||||||
|
|
||||||
for (id, client_room) in &room.client_rooms {
|
|
||||||
if *id != identity {
|
|
||||||
let _ = client_room
|
|
||||||
.0
|
|
||||||
.lock()
|
|
||||||
.video_track_updates
|
|
||||||
.0
|
|
||||||
.try_broadcast(RemoteVideoTrackUpdate::Subscribed(track.clone()))
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn publish_audio_track(
|
|
||||||
&self,
|
|
||||||
token: String,
|
|
||||||
_local_track: &LocalAudioTrack,
|
|
||||||
) -> Result<()> {
|
|
||||||
self.executor.simulate_random_delay().await;
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let identity = claims.sub.unwrap().to_string();
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&*room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
|
|
||||||
let track = Arc::new(RemoteAudioTrack {
|
|
||||||
sid: nanoid::nanoid!(17),
|
|
||||||
publisher_id: identity.clone(),
|
|
||||||
});
|
|
||||||
|
|
||||||
let publication = Arc::new(RemoteTrackPublication);
|
|
||||||
|
|
||||||
room.audio_tracks.push(track.clone());
|
|
||||||
|
|
||||||
for (id, client_room) in &room.client_rooms {
|
|
||||||
if *id != identity {
|
|
||||||
let _ = client_room
|
|
||||||
.0
|
|
||||||
.lock()
|
|
||||||
.audio_track_updates
|
|
||||||
.0
|
|
||||||
.try_broadcast(RemoteAudioTrackUpdate::Subscribed(
|
|
||||||
track.clone(),
|
|
||||||
publication.clone(),
|
|
||||||
))
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn video_tracks(&self, token: String) -> Result<Vec<Arc<RemoteVideoTrack>>> {
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&*room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
Ok(room.video_tracks.clone())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn audio_tracks(&self, token: String) -> Result<Vec<Arc<RemoteAudioTrack>>> {
|
|
||||||
let claims = live_kit_server::token::validate(&token, &self.secret_key)?;
|
|
||||||
let room_name = claims.video.room.unwrap();
|
|
||||||
|
|
||||||
let mut server_rooms = self.rooms.lock();
|
|
||||||
let room = server_rooms
|
|
||||||
.get_mut(&*room_name)
|
|
||||||
.ok_or_else(|| anyhow!("room {} does not exist", room_name))?;
|
|
||||||
Ok(room.audio_tracks.clone())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Default)]
|
|
||||||
struct TestServerRoom {
|
|
||||||
client_rooms: HashMap<Sid, Arc<Room>>,
|
|
||||||
video_tracks: Vec<Arc<RemoteVideoTrack>>,
|
|
||||||
audio_tracks: Vec<Arc<RemoteAudioTrack>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TestServerRoom {}
|
|
||||||
|
|
||||||
pub struct TestApiClient {
|
|
||||||
url: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[async_trait]
|
|
||||||
impl live_kit_server::api::Client for TestApiClient {
|
|
||||||
fn url(&self) -> &str {
|
|
||||||
&self.url
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn create_room(&self, name: String) -> Result<()> {
|
|
||||||
let server = TestServer::get(&self.url)?;
|
|
||||||
server.create_room(name).await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn delete_room(&self, name: String) -> Result<()> {
|
|
||||||
let server = TestServer::get(&self.url)?;
|
|
||||||
server.delete_room(name).await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn remove_participant(&self, room: String, identity: String) -> Result<()> {
|
|
||||||
let server = TestServer::get(&self.url)?;
|
|
||||||
server.remove_participant(room, identity).await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn room_token(&self, room: &str, identity: &str) -> Result<String> {
|
|
||||||
let server = TestServer::get(&self.url)?;
|
|
||||||
token::create(
|
|
||||||
&server.api_key,
|
|
||||||
&server.secret_key,
|
|
||||||
Some(identity),
|
|
||||||
token::VideoGrant::to_join(room),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn guest_token(&self, room: &str, identity: &str) -> Result<String> {
|
|
||||||
let server = TestServer::get(&self.url)?;
|
|
||||||
token::create(
|
|
||||||
&server.api_key,
|
|
||||||
&server.secret_key,
|
|
||||||
Some(identity),
|
|
||||||
token::VideoGrant::for_guest(room),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub type Sid = String;
|
|
||||||
|
|
||||||
struct RoomState {
|
|
||||||
connection: (
|
|
||||||
watch::Sender<ConnectionState>,
|
|
||||||
watch::Receiver<ConnectionState>,
|
|
||||||
),
|
|
||||||
display_sources: Vec<MacOSDisplay>,
|
|
||||||
audio_track_updates: (
|
|
||||||
async_broadcast::Sender<RemoteAudioTrackUpdate>,
|
|
||||||
async_broadcast::Receiver<RemoteAudioTrackUpdate>,
|
|
||||||
),
|
|
||||||
video_track_updates: (
|
|
||||||
async_broadcast::Sender<RemoteVideoTrackUpdate>,
|
|
||||||
async_broadcast::Receiver<RemoteVideoTrackUpdate>,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Eq, PartialEq)]
|
|
||||||
pub enum ConnectionState {
|
|
||||||
Disconnected,
|
|
||||||
Connected { url: String, token: String },
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Room(Mutex<RoomState>);
|
|
||||||
|
|
||||||
impl Room {
|
|
||||||
pub fn new() -> Arc<Self> {
|
|
||||||
Arc::new(Self(Mutex::new(RoomState {
|
|
||||||
connection: watch::channel_with(ConnectionState::Disconnected),
|
|
||||||
display_sources: Default::default(),
|
|
||||||
video_track_updates: async_broadcast::broadcast(128),
|
|
||||||
audio_track_updates: async_broadcast::broadcast(128),
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn status(&self) -> watch::Receiver<ConnectionState> {
|
|
||||||
self.0.lock().connection.1.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn connect(self: &Arc<Self>, url: &str, token: &str) -> impl Future<Output = Result<()>> {
|
|
||||||
let this = self.clone();
|
|
||||||
let url = url.to_string();
|
|
||||||
let token = token.to_string();
|
|
||||||
async move {
|
|
||||||
let server = TestServer::get(&url)?;
|
|
||||||
server
|
|
||||||
.join_room(token.clone(), this.clone())
|
|
||||||
.await
|
|
||||||
.context("room join")?;
|
|
||||||
*this.0.lock().connection.0.borrow_mut() = ConnectionState::Connected { url, token };
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn display_sources(self: &Arc<Self>) -> impl Future<Output = Result<Vec<MacOSDisplay>>> {
|
|
||||||
let this = self.clone();
|
|
||||||
async move {
|
|
||||||
let server = this.test_server();
|
|
||||||
server.executor.simulate_random_delay().await;
|
|
||||||
Ok(this.0.lock().display_sources.clone())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publish_video_track(
|
|
||||||
self: &Arc<Self>,
|
|
||||||
track: LocalVideoTrack,
|
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
|
||||||
let this = self.clone();
|
|
||||||
let track = track.clone();
|
|
||||||
async move {
|
|
||||||
this.test_server()
|
|
||||||
.publish_video_track(this.token(), track)
|
|
||||||
.await?;
|
|
||||||
Ok(LocalTrackPublication)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn publish_audio_track(
|
|
||||||
self: &Arc<Self>,
|
|
||||||
track: LocalAudioTrack,
|
|
||||||
) -> impl Future<Output = Result<LocalTrackPublication>> {
|
|
||||||
let this = self.clone();
|
|
||||||
let track = track.clone();
|
|
||||||
async move {
|
|
||||||
this.test_server()
|
|
||||||
.publish_audio_track(this.token(), &track)
|
|
||||||
.await?;
|
|
||||||
Ok(LocalTrackPublication)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn unpublish_track(&self, _publication: LocalTrackPublication) {}
|
|
||||||
|
|
||||||
pub fn remote_audio_tracks(&self, publisher_id: &str) -> Vec<Arc<RemoteAudioTrack>> {
|
|
||||||
if !self.is_connected() {
|
|
||||||
return Vec::new();
|
|
||||||
}
|
|
||||||
|
|
||||||
self.test_server()
|
|
||||||
.audio_tracks(self.token())
|
|
||||||
.unwrap()
|
|
||||||
.into_iter()
|
|
||||||
.filter(|track| track.publisher_id() == publisher_id)
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_audio_track_publications(
|
|
||||||
&self,
|
|
||||||
publisher_id: &str,
|
|
||||||
) -> Vec<Arc<RemoteTrackPublication>> {
|
|
||||||
if !self.is_connected() {
|
|
||||||
return Vec::new();
|
|
||||||
}
|
|
||||||
|
|
||||||
self.test_server()
|
|
||||||
.audio_tracks(self.token())
|
|
||||||
.unwrap()
|
|
||||||
.into_iter()
|
|
||||||
.filter(|track| track.publisher_id() == publisher_id)
|
|
||||||
.map(|_track| Arc::new(RemoteTrackPublication {}))
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_video_tracks(&self, publisher_id: &str) -> Vec<Arc<RemoteVideoTrack>> {
|
|
||||||
if !self.is_connected() {
|
|
||||||
return Vec::new();
|
|
||||||
}
|
|
||||||
|
|
||||||
self.test_server()
|
|
||||||
.video_tracks(self.token())
|
|
||||||
.unwrap()
|
|
||||||
.into_iter()
|
|
||||||
.filter(|track| track.publisher_id() == publisher_id)
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_audio_track_updates(&self) -> impl Stream<Item = RemoteAudioTrackUpdate> {
|
|
||||||
self.0.lock().audio_track_updates.1.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remote_video_track_updates(&self) -> impl Stream<Item = RemoteVideoTrackUpdate> {
|
|
||||||
self.0.lock().video_track_updates.1.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_display_sources(&self, sources: Vec<MacOSDisplay>) {
|
|
||||||
self.0.lock().display_sources = sources;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn test_server(&self) -> Arc<TestServer> {
|
|
||||||
match self.0.lock().connection.1.borrow().clone() {
|
|
||||||
ConnectionState::Disconnected => panic!("must be connected to call this method"),
|
|
||||||
ConnectionState::Connected { url, .. } => TestServer::get(&url).unwrap(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn token(&self) -> String {
|
|
||||||
match self.0.lock().connection.1.borrow().clone() {
|
|
||||||
ConnectionState::Disconnected => panic!("must be connected to call this method"),
|
|
||||||
ConnectionState::Connected { token, .. } => token,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_connected(&self) -> bool {
|
|
||||||
match *self.0.lock().connection.1.borrow() {
|
|
||||||
ConnectionState::Disconnected => false,
|
|
||||||
ConnectionState::Connected { .. } => true,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for Room {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
if let ConnectionState::Connected { token, .. } = mem::replace(
|
|
||||||
&mut *self.0.lock().connection.0.borrow_mut(),
|
|
||||||
ConnectionState::Disconnected,
|
|
||||||
) {
|
|
||||||
if let Ok(server) = TestServer::get(&token) {
|
|
||||||
let executor = server.executor.clone();
|
|
||||||
executor
|
|
||||||
.spawn(async move { server.leave_room(token).await.unwrap() })
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct LocalTrackPublication;
|
|
||||||
|
|
||||||
impl LocalTrackPublication {
|
|
||||||
pub fn set_mute(&self, _mute: bool) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct RemoteTrackPublication;
|
|
||||||
|
|
||||||
impl RemoteTrackPublication {
|
|
||||||
pub fn set_enabled(&self, _enabled: bool) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_muted(&self) -> bool {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sid(&self) -> String {
|
|
||||||
"".to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct LocalVideoTrack {
|
|
||||||
frames_rx: async_broadcast::Receiver<Frame>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl LocalVideoTrack {
|
|
||||||
pub fn screen_share_for_display(display: &MacOSDisplay) -> Self {
|
|
||||||
Self {
|
|
||||||
frames_rx: display.frames.1.clone(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct LocalAudioTrack;
|
|
||||||
|
|
||||||
impl LocalAudioTrack {
|
|
||||||
pub fn create() -> Self {
|
|
||||||
Self
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct RemoteVideoTrack {
|
|
||||||
sid: Sid,
|
|
||||||
publisher_id: Sid,
|
|
||||||
frames_rx: async_broadcast::Receiver<Frame>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RemoteVideoTrack {
|
|
||||||
pub fn sid(&self) -> &str {
|
|
||||||
&self.sid
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publisher_id(&self) -> &str {
|
|
||||||
&self.publisher_id
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn frames(&self) -> async_broadcast::Receiver<Frame> {
|
|
||||||
self.frames_rx.clone()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct RemoteAudioTrack {
|
|
||||||
sid: Sid,
|
|
||||||
publisher_id: Sid,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RemoteAudioTrack {
|
|
||||||
pub fn sid(&self) -> &str {
|
|
||||||
&self.sid
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn publisher_id(&self) -> &str {
|
|
||||||
&self.publisher_id
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn enable(&self) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn disable(&self) -> impl Future<Output = Result<()>> {
|
|
||||||
async { Ok(()) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub enum RemoteVideoTrackUpdate {
|
|
||||||
Subscribed(Arc<RemoteVideoTrack>),
|
|
||||||
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub enum RemoteAudioTrackUpdate {
|
|
||||||
ActiveSpeakersChanged { speakers: Vec<Sid> },
|
|
||||||
MuteChanged { track_id: Sid, muted: bool },
|
|
||||||
Subscribed(Arc<RemoteAudioTrack>, Arc<RemoteTrackPublication>),
|
|
||||||
Unsubscribed { publisher_id: Sid, track_id: Sid },
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct MacOSDisplay {
|
|
||||||
frames: (
|
|
||||||
async_broadcast::Sender<Frame>,
|
|
||||||
async_broadcast::Receiver<Frame>,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl MacOSDisplay {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
frames: async_broadcast::broadcast(128),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn send_frame(&self, frame: Frame) {
|
|
||||||
self.frames.0.try_broadcast(frame).unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
||||||
pub struct Frame {
|
|
||||||
pub label: String,
|
|
||||||
pub width: usize,
|
|
||||||
pub height: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Frame {
|
|
||||||
pub fn width(&self) -> usize {
|
|
||||||
self.width
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn height(&self) -> usize {
|
|
||||||
self.height
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn image(&self) -> CVImageBuffer {
|
|
||||||
unimplemented!("you can't call this in test mode")
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Add table
Add a link
Reference in a new issue