zed/crates/live_kit_server/src/api.rs

135 lines
3.6 KiB
Rust
Raw Normal View History

2022-10-14 16:31:03 +00:00
use crate::{proto, token};
use anyhow::{anyhow, Result};
use prost::Message;
use reqwest::header::CONTENT_TYPE;
use std::{future::Future, sync::Arc};
2022-10-14 15:00:38 +00:00
#[derive(Clone)]
2022-10-14 15:00:38 +00:00
pub struct Client {
http: reqwest::Client,
2022-10-17 12:50:05 +00:00
url: Arc<str>,
key: Arc<str>,
secret: Arc<str>,
2022-10-14 15:00:38 +00:00
}
impl Client {
2022-10-17 12:50:05 +00:00
pub fn new(mut url: String, key: String, secret: String) -> Self {
if url.ends_with('/') {
url.pop();
}
2022-10-14 15:00:38 +00:00
Self {
http: reqwest::Client::new(),
2022-10-17 12:50:05 +00:00
url: url.into(),
key: key.into(),
secret: secret.into(),
2022-10-14 15:00:38 +00:00
}
}
2022-10-17 12:50:05 +00:00
pub fn url(&self) -> &str {
&self.url
}
2022-10-14 16:31:03 +00:00
pub fn create_room(&self, name: String) -> impl Future<Output = Result<proto::Room>> {
self.request(
"twirp/livekit.RoomService/CreateRoom",
2022-10-14 16:31:03 +00:00
token::VideoGrant {
room_create: Some(true),
..Default::default()
},
proto::CreateRoomRequest {
name,
..Default::default()
},
)
}
pub fn delete_room(&self, name: String) -> impl Future<Output = Result<()>> {
let response = self.request(
"twirp/livekit.RoomService/DeleteRoom",
token::VideoGrant {
room_create: Some(true),
..Default::default()
},
proto::DeleteRoomRequest { room: name },
2022-10-14 16:31:03 +00:00
);
async move {
let _: proto::DeleteRoomResponse = response.await?;
Ok(())
}
}
pub fn remove_participant(
&self,
room: String,
identity: String,
) -> impl Future<Output = Result<()>> {
let response = self.request(
"twirp/livekit.RoomService/RemoveParticipant",
token::VideoGrant {
room_admin: Some(true),
room: Some(&room),
..Default::default()
},
proto::RoomParticipantIdentity {
room: room.clone(),
identity,
},
);
async move {
let _: proto::RemoveParticipantResponse = response.await?;
Ok(())
}
}
2022-10-14 15:00:38 +00:00
2022-10-17 16:00:54 +00:00
pub fn room_token(&self, room: &str, identity: &str) -> Result<String> {
token::create(
&self.key,
&self.secret,
Some(identity),
token::VideoGrant {
room: Some(room),
room_join: Some(true),
can_publish: Some(true),
can_subscribe: Some(true),
..Default::default()
},
)
}
fn request<Req, Res>(
&self,
path: &str,
grant: token::VideoGrant,
body: Req,
) -> impl Future<Output = Result<Res>>
where
Req: Message,
Res: Default + Message,
{
2022-10-14 16:31:03 +00:00
let client = self.http.clone();
let token = token::create(&self.key, &self.secret, None, grant);
2022-10-17 12:50:05 +00:00
let url = format!("{}/{}", self.url, path);
2022-10-14 16:31:03 +00:00
async move {
let token = token?;
let response = client
2022-10-17 12:50:05 +00:00
.post(&url)
.header(CONTENT_TYPE, "application/protobuf")
.bearer_auth(token)
.body(body.encode_to_vec())
.send()
.await?;
if response.status().is_success() {
Ok(Res::decode(response.bytes().await?)?)
} else {
Err(anyhow!(
"POST {} failed with status code {:?}, {:?}",
2022-10-17 12:50:05 +00:00
url,
response.status(),
response.text().await
))
}
2022-10-14 16:31:03 +00:00
}
2022-10-14 15:00:38 +00:00
}
}