active_call/call/
mod.rs

1use crate::{CallOption, ReferOption, media::recorder::RecorderOption, synthesis::SynthesisOption};
2use serde::{Deserialize, Serialize};
3use serde_with::skip_serializing_none;
4use std::{
5    collections::HashMap,
6    sync::{Arc, Mutex},
7};
8
9pub mod active_call;
10pub mod sip;
11pub use active_call::ActiveCall;
12pub use active_call::ActiveCallRef;
13pub use active_call::ActiveCallType;
14
15pub type CommandSender = tokio::sync::broadcast::Sender<Command>;
16pub type CommandReceiver = tokio::sync::broadcast::Receiver<Command>;
17
18// WebSocket Commands
19#[skip_serializing_none]
20#[derive(Debug, Deserialize, Serialize, Clone)]
21#[serde(
22    tag = "command",
23    rename_all = "camelCase",
24    rename_all_fields = "camelCase"
25)]
26pub enum Command {
27    Invite {
28        option: CallOption,
29    },
30    Accept {
31        option: CallOption,
32    },
33    Reject {
34        reason: String,
35        code: Option<u32>,
36    },
37    Ringing {
38        recorder: Option<RecorderOption>,
39        early_media: Option<bool>,
40        ringtone: Option<String>,
41    },
42    Tts {
43        text: String,
44        speaker: Option<String>,
45        /// If the play_id is the same, it will not interrupt the previous playback
46        play_id: Option<String>,
47        /// If auto_hangup is true, it means the call will be hung up automatically after the TTS playback is finished
48        auto_hangup: Option<bool>,
49        /// If streaming is true, it means the input text is streaming text,
50        /// and end_of_stream needs to be used to determine if it's finished,
51        /// equivalent to LLM's streaming output to TTS synthesis
52        streaming: Option<bool>,
53        /// If end_of_stream is true, it means the input text is finished
54        end_of_stream: Option<bool>,
55        option: Option<SynthesisOption>,
56        wait_input_timeout: Option<u32>,
57        /// if true, the text is base64 encoded pcm samples
58        base64: Option<bool>,
59    },
60    Play {
61        url: String,
62        play_id: Option<String>,
63        auto_hangup: Option<bool>,
64        wait_input_timeout: Option<u32>,
65    },
66    Interrupt {
67        graceful: Option<bool>,
68        fade_out_ms: Option<u32>,
69    },
70    Pause {},
71    Resume {},
72    Hangup {
73        reason: Option<String>,
74        initiator: Option<String>,
75    },
76    Refer {
77        caller: String,
78        /// aor of the calee, e.g., sip:bob@restsend.com
79        callee: String,
80        options: Option<ReferOption>,
81    },
82    Mute {
83        track_id: Option<String>,
84    },
85    Unmute {
86        track_id: Option<String>,
87    },
88    History {
89        speaker: String,
90        text: String,
91    },
92}
93
94/// Routing state for managing stateful load balancing
95#[derive(Debug)]
96pub struct RoutingState {
97    /// Round-robin counters for each destination group
98    round_robin_counters: Arc<Mutex<HashMap<String, usize>>>,
99}
100
101impl Default for RoutingState {
102    fn default() -> Self {
103        Self::new()
104    }
105}
106
107impl RoutingState {
108    pub fn new() -> Self {
109        Self {
110            round_robin_counters: Arc::new(Mutex::new(HashMap::new())),
111        }
112    }
113
114    /// Get the next trunk index for round-robin selection
115    pub fn next_round_robin_index(&self, destination_key: &str, trunk_count: usize) -> usize {
116        if trunk_count == 0 {
117            return 0;
118        }
119
120        let mut counters = self.round_robin_counters.lock().unwrap();
121        let counter = counters
122            .entry(destination_key.to_string())
123            .or_insert_with(|| 0);
124        let r = *counter % trunk_count;
125        *counter += 1;
126        return r;
127    }
128}