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#[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 play_id: Option<String>,
47 auto_hangup: Option<bool>,
49 streaming: Option<bool>,
53 end_of_stream: Option<bool>,
55 option: Option<SynthesisOption>,
56 wait_input_timeout: Option<u32>,
57 base64: Option<bool>,
59 cache_key: Option<String>,
61 },
62 Play {
63 url: String,
64 play_id: Option<String>,
65 auto_hangup: Option<bool>,
66 wait_input_timeout: Option<u32>,
67 },
68 Interrupt {
69 graceful: Option<bool>,
70 fade_out_ms: Option<u32>,
71 },
72 Pause {},
73 Resume {},
74 Hangup {
75 reason: Option<String>,
76 initiator: Option<String>,
77 },
78 Refer {
79 caller: String,
80 callee: String,
82 options: Option<ReferOption>,
83 },
84 Mute {
85 track_id: Option<String>,
86 },
87 Unmute {
88 track_id: Option<String>,
89 },
90 History {
91 speaker: String,
92 text: String,
93 },
94}
95
96#[derive(Debug)]
98pub struct RoutingState {
99 round_robin_counters: Arc<Mutex<HashMap<String, usize>>>,
101}
102
103impl Default for RoutingState {
104 fn default() -> Self {
105 Self::new()
106 }
107}
108
109impl RoutingState {
110 pub fn new() -> Self {
111 Self {
112 round_robin_counters: Arc::new(Mutex::new(HashMap::new())),
113 }
114 }
115
116 pub fn next_round_robin_index(&self, destination_key: &str, trunk_count: usize) -> usize {
118 if trunk_count == 0 {
119 return 0;
120 }
121
122 let mut counters = self.round_robin_counters.lock().unwrap();
123 let counter = counters
124 .entry(destination_key.to_string())
125 .or_insert_with(|| 0);
126 let r = *counter % trunk_count;
127 *counter += 1;
128 return r;
129 }
130}