use std::fmt;
wire_enum! {
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[repr(u8)]
pub enum ChannelState {
CsNew = 0 => "CS_NEW",
CsInit = 1 => "CS_INIT",
CsRouting = 2 => "CS_ROUTING",
CsSoftExecute = 3 => "CS_SOFT_EXECUTE",
CsExecute = 4 => "CS_EXECUTE",
CsExchangeMedia = 5 => "CS_EXCHANGE_MEDIA",
CsPark = 6 => "CS_PARK",
CsConsumeMedia = 7 => "CS_CONSUME_MEDIA",
CsHibernate = 8 => "CS_HIBERNATE",
CsReset = 9 => "CS_RESET",
CsHangup = 10 => "CS_HANGUP",
CsReporting = 11 => "CS_REPORTING",
CsDestroy = 12 => "CS_DESTROY",
CsNone = 13 => "CS_NONE",
}
error ParseChannelStateError("channel state");
tests: channel_state_wire_tests;
}
impl ChannelState {
pub fn from_number(n: u8) -> Option<Self> {
match n {
0 => Some(Self::CsNew),
1 => Some(Self::CsInit),
2 => Some(Self::CsRouting),
3 => Some(Self::CsSoftExecute),
4 => Some(Self::CsExecute),
5 => Some(Self::CsExchangeMedia),
6 => Some(Self::CsPark),
7 => Some(Self::CsConsumeMedia),
8 => Some(Self::CsHibernate),
9 => Some(Self::CsReset),
10 => Some(Self::CsHangup),
11 => Some(Self::CsReporting),
12 => Some(Self::CsDestroy),
13 => Some(Self::CsNone),
_ => None,
}
}
pub fn as_number(&self) -> u8 {
*self as u8
}
}
wire_enum! {
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum CallState {
Down => "DOWN",
Dialing => "DIALING",
Ringing => "RINGING",
Early => "EARLY",
Active => "ACTIVE",
Held => "HELD",
RingWait => "RING_WAIT",
Hangup => "HANGUP",
Unheld => "UNHELD",
}
error ParseCallStateError("call state");
tests: call_state_wire_tests;
}
wire_enum! {
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum AnswerState {
Hangup => "hangup",
Answered => "answered",
Early => "early",
Ringing => "ringing",
}
error ParseAnswerStateError("answer state");
tests: answer_state_wire_tests;
}
wire_enum! {
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum CallDirection {
Inbound => "inbound",
Outbound => "outbound",
}
error ParseCallDirectionError("call direction");
tests: call_direction_wire_tests;
}
wire_enum! {
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
#[repr(u16)]
pub enum HangupCause {
None = 0 => "NONE",
UnallocatedNumber = 1 => "UNALLOCATED_NUMBER",
NoRouteTransitNet = 2 => "NO_ROUTE_TRANSIT_NET",
NoRouteDestination = 3 => "NO_ROUTE_DESTINATION",
ChannelUnacceptable = 6 => "CHANNEL_UNACCEPTABLE",
CallAwardedDelivered = 7 => "CALL_AWARDED_DELIVERED",
NormalClearing = 16 => "NORMAL_CLEARING",
UserBusy = 17 => "USER_BUSY",
NoUserResponse = 18 => "NO_USER_RESPONSE",
NoAnswer = 19 => "NO_ANSWER",
SubscriberAbsent = 20 => "SUBSCRIBER_ABSENT",
CallRejected = 21 => "CALL_REJECTED",
NumberChanged = 22 => "NUMBER_CHANGED",
RedirectionToNewDestination = 23 => "REDIRECTION_TO_NEW_DESTINATION",
ExchangeRoutingError = 25 => "EXCHANGE_ROUTING_ERROR",
DestinationOutOfOrder = 27 => "DESTINATION_OUT_OF_ORDER",
InvalidNumberFormat = 28 => "INVALID_NUMBER_FORMAT",
FacilityRejected = 29 => "FACILITY_REJECTED",
ResponseToStatusEnquiry = 30 => "RESPONSE_TO_STATUS_ENQUIRY",
NormalUnspecified = 31 => "NORMAL_UNSPECIFIED",
NormalCircuitCongestion = 34 => "NORMAL_CIRCUIT_CONGESTION",
NetworkOutOfOrder = 38 => "NETWORK_OUT_OF_ORDER",
NormalTemporaryFailure = 41 => "NORMAL_TEMPORARY_FAILURE",
SwitchCongestion = 42 => "SWITCH_CONGESTION",
AccessInfoDiscarded = 43 => "ACCESS_INFO_DISCARDED",
RequestedChanUnavail = 44 => "REQUESTED_CHAN_UNAVAIL",
PreEmpted = 45 => "PRE_EMPTED",
FacilityNotSubscribed = 50 => "FACILITY_NOT_SUBSCRIBED",
OutgoingCallBarred = 52 => "OUTGOING_CALL_BARRED",
IncomingCallBarred = 54 => "INCOMING_CALL_BARRED",
BearercapabilityNotauth = 57 => "BEARERCAPABILITY_NOTAUTH",
BearercapabilityNotavail = 58 => "BEARERCAPABILITY_NOTAVAIL",
ServiceUnavailable = 63 => "SERVICE_UNAVAILABLE",
BearercapabilityNotimpl = 65 => "BEARERCAPABILITY_NOTIMPL",
ChanNotImplemented = 66 => "CHAN_NOT_IMPLEMENTED",
FacilityNotImplemented = 69 => "FACILITY_NOT_IMPLEMENTED",
ServiceNotImplemented = 79 => "SERVICE_NOT_IMPLEMENTED",
InvalidCallReference = 81 => "INVALID_CALL_REFERENCE",
IncompatibleDestination = 88 => "INCOMPATIBLE_DESTINATION",
InvalidMsgUnspecified = 95 => "INVALID_MSG_UNSPECIFIED",
MandatoryIeMissing = 96 => "MANDATORY_IE_MISSING",
MessageTypeNonexist = 97 => "MESSAGE_TYPE_NONEXIST",
WrongMessage = 98 => "WRONG_MESSAGE",
IeNonexist = 99 => "IE_NONEXIST",
InvalidIeContents = 100 => "INVALID_IE_CONTENTS",
WrongCallState = 101 => "WRONG_CALL_STATE",
RecoveryOnTimerExpire = 102 => "RECOVERY_ON_TIMER_EXPIRE",
MandatoryIeLengthError = 103 => "MANDATORY_IE_LENGTH_ERROR",
ProtocolError = 111 => "PROTOCOL_ERROR",
Interworking = 127 => "INTERWORKING",
Success = 142 => "SUCCESS",
OriginatorCancel = 487 => "ORIGINATOR_CANCEL",
Crash = 700 => "CRASH",
SystemShutdown = 701 => "SYSTEM_SHUTDOWN",
LoseRace = 702 => "LOSE_RACE",
ManagerRequest = 703 => "MANAGER_REQUEST",
BlindTransfer = 800 => "BLIND_TRANSFER",
AttendedTransfer = 801 => "ATTENDED_TRANSFER",
AllottedTimeout = 802 => "ALLOTTED_TIMEOUT",
UserChallenge = 803 => "USER_CHALLENGE",
MediaTimeout = 804 => "MEDIA_TIMEOUT",
PickedOff = 805 => "PICKED_OFF",
UserNotRegistered = 806 => "USER_NOT_REGISTERED",
ProgressTimeout = 807 => "PROGRESS_TIMEOUT",
InvalidGateway = 808 => "INVALID_GATEWAY",
GatewayDown = 809 => "GATEWAY_DOWN",
InvalidUrl = 810 => "INVALID_URL",
InvalidProfile = 811 => "INVALID_PROFILE",
NoPickup = 812 => "NO_PICKUP",
SrtpReadError = 813 => "SRTP_READ_ERROR",
Bowout = 814 => "BOWOUT",
BusyEverywhere = 815 => "BUSY_EVERYWHERE",
Decline = 816 => "DECLINE",
DoesNotExistAnywhere = 817 => "DOES_NOT_EXIST_ANYWHERE",
NotAcceptable = 818 => "NOT_ACCEPTABLE",
Unwanted = 819 => "UNWANTED",
NoIdentity = 820 => "NO_IDENTITY",
BadIdentityInfo = 821 => "BAD_IDENTITY_INFO",
UnsupportedCertificate = 822 => "UNSUPPORTED_CERTIFICATE",
InvalidIdentity = 823 => "INVALID_IDENTITY",
StaleDate = 824 => "STALE_DATE",
RejectAll = 825 => "REJECT_ALL",
}
error ParseHangupCauseError("hangup cause");
numeric: from_number(u16);
tests: hangup_cause_wire_tests;
}
impl HangupCause {
pub fn from_sip_response(code: u16) -> Option<Self> {
match code {
200 => Some(Self::NormalClearing),
401 | 402 | 403 | 407 | 603 | 608 => Some(Self::CallRejected),
607 => Some(Self::Unwanted),
404 => Some(Self::UnallocatedNumber),
485 | 604 => Some(Self::NoRouteDestination),
408 | 504 => Some(Self::RecoveryOnTimerExpire),
410 => Some(Self::NumberChanged),
413 | 414 | 416 | 420 | 421 | 423 | 505 | 513 => Some(Self::Interworking),
480 => Some(Self::NoUserResponse),
400 | 481 | 500 | 503 => Some(Self::NormalTemporaryFailure),
486 | 600 => Some(Self::UserBusy),
484 => Some(Self::InvalidNumberFormat),
488 | 606 => Some(Self::IncompatibleDestination),
502 => Some(Self::NetworkOutOfOrder),
405 => Some(Self::ServiceUnavailable),
406 | 415 | 501 => Some(Self::ServiceNotImplemented),
482 | 483 => Some(Self::ExchangeRoutingError),
487 => Some(Self::OriginatorCancel),
428 => Some(Self::NoIdentity),
429 => Some(Self::BadIdentityInfo),
437 => Some(Self::UnsupportedCertificate),
438 => Some(Self::InvalidIdentity),
_ => None,
}
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[non_exhaustive]
pub struct ChannelTimetable {
pub profile_created: Option<i64>,
pub created: Option<i64>,
pub answered: Option<i64>,
pub progress: Option<i64>,
pub progress_media: Option<i64>,
pub hungup: Option<i64>,
pub transferred: Option<i64>,
pub resurrected: Option<i64>,
pub bridged: Option<i64>,
pub last_hold: Option<i64>,
pub hold_accum: Option<i64>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[non_exhaustive]
pub enum TimetablePrefix {
Caller,
OtherLeg,
Channel,
Hunt,
Originator,
Originatee,
PostOriginator,
PostOriginatee,
}
impl TimetablePrefix {
pub fn as_str(&self) -> &'static str {
match self {
Self::Caller => "Caller",
Self::OtherLeg => "Other-Leg",
Self::Channel => "Channel",
Self::Hunt => "Hunt",
Self::Originator => "ORIGINATOR",
Self::Originatee => "ORIGINATEE",
Self::PostOriginator => "POST-ORIGINATOR",
Self::PostOriginatee => "POST-ORIGINATEE",
}
}
}
impl fmt::Display for TimetablePrefix {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(self.as_str())
}
}
impl AsRef<str> for TimetablePrefix {
fn as_ref(&self) -> &str {
self.as_str()
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
#[non_exhaustive]
pub struct ParseTimetableError {
pub header: String,
pub value: String,
}
impl fmt::Display for ParseTimetableError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"invalid timetable value for {}: {:?}",
self.header, self.value
)
}
}
impl std::error::Error for ParseTimetableError {}
impl ParseTimetableError {
pub fn new(header: impl Into<String>, value: impl Into<String>) -> Self {
Self {
header: header.into(),
value: value.into(),
}
}
}
impl ChannelTimetable {
pub const SUFFIXES: &'static [&'static str] = &[
"Profile-Created-Time",
"Channel-Created-Time",
"Channel-Answered-Time",
"Channel-Progress-Time",
"Channel-Progress-Media-Time",
"Channel-Hangup-Time",
"Channel-Transfer-Time",
"Channel-Resurrect-Time",
"Channel-Bridged-Time",
"Channel-Last-Hold",
"Channel-Hold-Accum",
];
pub fn from_lookup<'a>(
prefix: impl AsRef<str>,
lookup: impl Fn(&str) -> Option<&'a str>,
) -> Result<Option<Self>, ParseTimetableError> {
let prefix = prefix.as_ref();
let mut tt = Self::default();
let mut found = false;
macro_rules! field {
($field:ident, $suffix:literal) => {
let header = format!("{}-{}", prefix, $suffix);
if let Some(raw) = lookup(&header) {
let v: i64 = raw
.parse()
.map_err(|_| ParseTimetableError {
header: header.clone(),
value: raw.to_string(),
})?;
tt.$field = Some(v);
found = true;
}
};
}
field!(profile_created, "Profile-Created-Time");
field!(created, "Channel-Created-Time");
field!(answered, "Channel-Answered-Time");
field!(progress, "Channel-Progress-Time");
field!(progress_media, "Channel-Progress-Media-Time");
field!(hungup, "Channel-Hangup-Time");
field!(transferred, "Channel-Transfer-Time");
field!(resurrected, "Channel-Resurrect-Time");
field!(bridged, "Channel-Bridged-Time");
field!(last_hold, "Channel-Last-Hold");
field!(hold_accum, "Channel-Hold-Accum");
if found {
Ok(Some(tt))
} else {
Ok(None)
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::event::EslEvent;
use crate::lookup::HeaderLookup;
#[test]
fn test_channel_state_from_number() {
assert_eq!(ChannelState::from_number(0), Some(ChannelState::CsNew));
assert_eq!(ChannelState::from_number(4), Some(ChannelState::CsExecute));
assert_eq!(ChannelState::from_number(10), Some(ChannelState::CsHangup));
assert_eq!(ChannelState::from_number(13), Some(ChannelState::CsNone));
assert_eq!(ChannelState::from_number(14), None);
assert_eq!(ChannelState::from_number(255), None);
}
#[test]
fn test_channel_state_as_number() {
assert_eq!(ChannelState::CsNew.as_number(), 0);
assert_eq!(ChannelState::CsExecute.as_number(), 4);
assert_eq!(ChannelState::CsHangup.as_number(), 10);
assert_eq!(ChannelState::CsNone.as_number(), 13);
}
#[test]
fn channel_state_ordering_follows_lifecycle() {
assert!(ChannelState::CsNew < ChannelState::CsInit);
assert!(ChannelState::CsInit < ChannelState::CsRouting);
assert!(ChannelState::CsRouting < ChannelState::CsExecute);
assert!(ChannelState::CsExecute < ChannelState::CsHangup);
assert!(ChannelState::CsHangup < ChannelState::CsReporting);
assert!(ChannelState::CsReporting < ChannelState::CsDestroy);
}
#[allow(clippy::nonminimal_bool)]
#[test]
fn channel_state_teardown_check() {
assert!(ChannelState::CsHangup >= ChannelState::CsHangup);
assert!(ChannelState::CsReporting >= ChannelState::CsHangup);
assert!(ChannelState::CsDestroy >= ChannelState::CsHangup);
assert!(!(ChannelState::CsExecute >= ChannelState::CsHangup));
assert!(!(ChannelState::CsPark >= ChannelState::CsHangup));
}
#[test]
fn call_state_ordering_matches_c_enum() {
assert!(CallState::Down < CallState::Dialing);
assert!(CallState::Dialing < CallState::Ringing);
assert!(CallState::Early < CallState::Active);
assert!(CallState::Active < CallState::Hangup);
}
#[test]
fn hangup_cause_display() {
assert_eq!(HangupCause::NormalClearing.to_string(), "NORMAL_CLEARING");
assert_eq!(HangupCause::UserBusy.to_string(), "USER_BUSY");
assert_eq!(
HangupCause::OriginatorCancel.to_string(),
"ORIGINATOR_CANCEL"
);
assert_eq!(HangupCause::None.to_string(), "NONE");
}
#[test]
fn hangup_cause_from_str() {
assert_eq!(
"NORMAL_CLEARING"
.parse::<HangupCause>()
.unwrap(),
HangupCause::NormalClearing
);
assert_eq!(
"USER_BUSY"
.parse::<HangupCause>()
.unwrap(),
HangupCause::UserBusy
);
}
#[test]
fn hangup_cause_from_str_rejects_wrong_case() {
assert!("normal_clearing"
.parse::<HangupCause>()
.is_err());
assert!("User_Busy"
.parse::<HangupCause>()
.is_err());
}
#[test]
fn hangup_cause_from_str_unknown() {
assert!("BOGUS_CAUSE"
.parse::<HangupCause>()
.is_err());
}
#[test]
fn hangup_cause_display_round_trip() {
let causes = [
HangupCause::None,
HangupCause::NormalClearing,
HangupCause::UserBusy,
HangupCause::NoAnswer,
HangupCause::OriginatorCancel,
HangupCause::BlindTransfer,
HangupCause::InvalidIdentity,
];
for cause in causes {
let s = cause.to_string();
let parsed: HangupCause = s
.parse()
.unwrap();
assert_eq!(parsed, cause);
}
}
#[test]
fn hangup_cause_as_number_q850() {
assert_eq!(HangupCause::None.as_number(), 0);
assert_eq!(HangupCause::UnallocatedNumber.as_number(), 1);
assert_eq!(HangupCause::NormalClearing.as_number(), 16);
assert_eq!(HangupCause::UserBusy.as_number(), 17);
assert_eq!(HangupCause::NoAnswer.as_number(), 19);
assert_eq!(HangupCause::CallRejected.as_number(), 21);
assert_eq!(HangupCause::NormalUnspecified.as_number(), 31);
assert_eq!(HangupCause::Interworking.as_number(), 127);
}
#[test]
fn hangup_cause_as_number_freeswitch_extensions() {
assert_eq!(HangupCause::Success.as_number(), 142);
assert_eq!(HangupCause::OriginatorCancel.as_number(), 487);
assert_eq!(HangupCause::Crash.as_number(), 700);
assert_eq!(HangupCause::BlindTransfer.as_number(), 800);
assert_eq!(HangupCause::InvalidIdentity.as_number(), 823);
}
#[test]
fn hangup_cause_from_number_round_trip() {
let codes: &[u16] = &[0, 1, 16, 17, 19, 21, 31, 127, 142, 487, 700, 800, 823];
for &code in codes {
let cause = HangupCause::from_number(code).unwrap();
assert_eq!(cause.as_number(), code);
}
}
#[test]
fn hangup_cause_from_number_unknown() {
assert!(HangupCause::from_number(999).is_none());
assert!(HangupCause::from_number(4).is_none());
}
#[test]
fn from_sip_response_success() {
assert_eq!(
HangupCause::from_sip_response(200),
Some(HangupCause::NormalClearing)
);
}
#[test]
fn from_sip_response_4xx_auth_rejection() {
for code in [401, 402, 403, 407] {
assert_eq!(
HangupCause::from_sip_response(code),
Some(HangupCause::CallRejected),
"SIP {code}"
);
}
}
#[test]
fn from_sip_response_4xx_routing() {
assert_eq!(
HangupCause::from_sip_response(404),
Some(HangupCause::UnallocatedNumber)
);
assert_eq!(
HangupCause::from_sip_response(485),
Some(HangupCause::NoRouteDestination)
);
assert_eq!(
HangupCause::from_sip_response(484),
Some(HangupCause::InvalidNumberFormat)
);
assert_eq!(
HangupCause::from_sip_response(410),
Some(HangupCause::NumberChanged)
);
}
#[test]
fn from_sip_response_4xx_service() {
assert_eq!(
HangupCause::from_sip_response(405),
Some(HangupCause::ServiceUnavailable)
);
for code in [406, 415, 501] {
assert_eq!(
HangupCause::from_sip_response(code),
Some(HangupCause::ServiceNotImplemented),
"SIP {code}"
);
}
}
#[test]
fn from_sip_response_4xx_interworking() {
for code in [413, 414, 416, 420, 421, 423, 505, 513] {
assert_eq!(
HangupCause::from_sip_response(code),
Some(HangupCause::Interworking),
"SIP {code}"
);
}
}
#[test]
fn from_sip_response_4xx_timeout_and_busy() {
assert_eq!(
HangupCause::from_sip_response(408),
Some(HangupCause::RecoveryOnTimerExpire)
);
assert_eq!(
HangupCause::from_sip_response(504),
Some(HangupCause::RecoveryOnTimerExpire)
);
assert_eq!(
HangupCause::from_sip_response(480),
Some(HangupCause::NoUserResponse)
);
assert_eq!(
HangupCause::from_sip_response(486),
Some(HangupCause::UserBusy)
);
assert_eq!(
HangupCause::from_sip_response(487),
Some(HangupCause::OriginatorCancel)
);
}
#[test]
fn from_sip_response_4xx_temporary_failure() {
for code in [400, 481, 500, 503] {
assert_eq!(
HangupCause::from_sip_response(code),
Some(HangupCause::NormalTemporaryFailure),
"SIP {code}"
);
}
}
#[test]
fn from_sip_response_4xx_exchange_routing() {
for code in [482, 483] {
assert_eq!(
HangupCause::from_sip_response(code),
Some(HangupCause::ExchangeRoutingError),
"SIP {code}"
);
}
}
#[test]
fn from_sip_response_4xx_media() {
assert_eq!(
HangupCause::from_sip_response(488),
Some(HangupCause::IncompatibleDestination)
);
assert_eq!(
HangupCause::from_sip_response(606),
Some(HangupCause::IncompatibleDestination)
);
}
#[test]
fn from_sip_response_5xx() {
assert_eq!(
HangupCause::from_sip_response(502),
Some(HangupCause::NetworkOutOfOrder)
);
}
#[test]
fn from_sip_response_6xx() {
assert_eq!(
HangupCause::from_sip_response(600),
Some(HangupCause::UserBusy)
);
assert_eq!(
HangupCause::from_sip_response(603),
Some(HangupCause::CallRejected)
);
assert_eq!(
HangupCause::from_sip_response(604),
Some(HangupCause::NoRouteDestination)
);
assert_eq!(
HangupCause::from_sip_response(607),
Some(HangupCause::Unwanted)
);
assert_eq!(
HangupCause::from_sip_response(608),
Some(HangupCause::CallRejected)
);
}
#[test]
fn from_sip_response_stir_shaken() {
assert_eq!(
HangupCause::from_sip_response(428),
Some(HangupCause::NoIdentity)
);
assert_eq!(
HangupCause::from_sip_response(429),
Some(HangupCause::BadIdentityInfo)
);
assert_eq!(
HangupCause::from_sip_response(437),
Some(HangupCause::UnsupportedCertificate)
);
assert_eq!(
HangupCause::from_sip_response(438),
Some(HangupCause::InvalidIdentity)
);
}
#[test]
fn from_sip_response_unmapped_returns_none() {
for code in [100, 180, 183, 301, 302] {
assert_eq!(
HangupCause::from_sip_response(code),
None,
"SIP {code} should be None"
);
}
for code in [409, 411, 412, 422, 489, 491, 493, 506, 580] {
assert_eq!(
HangupCause::from_sip_response(code),
None,
"SIP {code} should be None"
);
}
}
#[test]
fn caller_timetable_all_fields() {
let mut event = EslEvent::new();
event.set_header("Caller-Profile-Created-Time", "1700000000000000");
event.set_header("Caller-Channel-Created-Time", "1700000001000000");
event.set_header("Caller-Channel-Answered-Time", "1700000005000000");
event.set_header("Caller-Channel-Progress-Time", "1700000002000000");
event.set_header("Caller-Channel-Progress-Media-Time", "1700000003000000");
event.set_header("Caller-Channel-Hangup-Time", "0");
event.set_header("Caller-Channel-Transfer-Time", "0");
event.set_header("Caller-Channel-Resurrect-Time", "0");
event.set_header("Caller-Channel-Bridged-Time", "1700000006000000");
event.set_header("Caller-Channel-Last-Hold", "0");
event.set_header("Caller-Channel-Hold-Accum", "0");
let tt = event
.caller_timetable()
.unwrap()
.expect("should have timetable");
assert_eq!(tt.profile_created, Some(1700000000000000));
assert_eq!(tt.created, Some(1700000001000000));
assert_eq!(tt.answered, Some(1700000005000000));
assert_eq!(tt.progress, Some(1700000002000000));
assert_eq!(tt.progress_media, Some(1700000003000000));
assert_eq!(tt.hungup, Some(0));
assert_eq!(tt.transferred, Some(0));
assert_eq!(tt.resurrected, Some(0));
assert_eq!(tt.bridged, Some(1700000006000000));
assert_eq!(tt.last_hold, Some(0));
assert_eq!(tt.hold_accum, Some(0));
}
#[test]
fn other_leg_timetable() {
let mut event = EslEvent::new();
event.set_header("Other-Leg-Profile-Created-Time", "1700000000000000");
event.set_header("Other-Leg-Channel-Created-Time", "1700000001000000");
event.set_header("Other-Leg-Channel-Answered-Time", "1700000005000000");
event.set_header("Other-Leg-Channel-Progress-Time", "0");
event.set_header("Other-Leg-Channel-Progress-Media-Time", "0");
event.set_header("Other-Leg-Channel-Hangup-Time", "0");
event.set_header("Other-Leg-Channel-Transfer-Time", "0");
event.set_header("Other-Leg-Channel-Resurrect-Time", "0");
event.set_header("Other-Leg-Channel-Bridged-Time", "1700000006000000");
event.set_header("Other-Leg-Channel-Last-Hold", "0");
event.set_header("Other-Leg-Channel-Hold-Accum", "0");
let tt = event
.other_leg_timetable()
.unwrap()
.expect("should have timetable");
assert_eq!(tt.created, Some(1700000001000000));
assert_eq!(tt.bridged, Some(1700000006000000));
}
#[test]
fn timetable_no_headers() {
let event = EslEvent::new();
assert_eq!(
event
.caller_timetable()
.unwrap(),
None
);
assert_eq!(
event
.other_leg_timetable()
.unwrap(),
None
);
}
#[test]
fn timetable_partial_headers() {
let mut event = EslEvent::new();
event.set_header("Caller-Channel-Created-Time", "1700000001000000");
let tt = event
.caller_timetable()
.unwrap()
.expect("at least one field parsed");
assert_eq!(tt.created, Some(1700000001000000));
assert_eq!(tt.answered, None);
assert_eq!(tt.profile_created, None);
}
#[test]
fn timetable_invalid_value_is_error() {
let mut event = EslEvent::new();
event.set_header("Caller-Channel-Created-Time", "not_a_number");
let err = event
.caller_timetable()
.unwrap_err();
assert_eq!(err.header, "Caller-Channel-Created-Time");
assert_eq!(err.value, "not_a_number");
}
#[test]
fn timetable_valid_then_invalid_is_error() {
let mut event = EslEvent::new();
event.set_header("Caller-Profile-Created-Time", "1700000000000000");
event.set_header("Caller-Channel-Created-Time", "garbage");
let err = event
.caller_timetable()
.unwrap_err();
assert_eq!(err.header, "Caller-Channel-Created-Time");
assert_eq!(err.value, "garbage");
}
#[test]
fn timetable_zero_preserved() {
let mut event = EslEvent::new();
event.set_header("Caller-Channel-Hangup-Time", "0");
let tt = event
.caller_timetable()
.unwrap()
.expect("should have timetable");
assert_eq!(tt.hungup, Some(0));
}
#[test]
fn timetable_custom_prefix() {
let mut event = EslEvent::new();
event.set_header("Channel-Channel-Created-Time", "1700000001000000");
let tt = event
.timetable("Channel")
.unwrap()
.expect("custom prefix should work");
assert_eq!(tt.created, Some(1700000001000000));
}
}