Start adding some strongly-typed accessors to common channel vars
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@14774 d0543943-73ff-0310-b7d9-9358b9ac24b2
This commit is contained in:
parent
2a3be1dc98
commit
f22a237b44
|
@ -0,0 +1,231 @@
|
||||||
|
/*
|
||||||
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application - mod_managed
|
||||||
|
* Copyright (C) 2008, Michael Giagnocavo <mgg@giagnocavo.net>
|
||||||
|
*
|
||||||
|
* Version: MPL 1.1
|
||||||
|
*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License Version
|
||||||
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
||||||
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||||||
|
* for the specific language governing rights and limitations under the
|
||||||
|
* License.
|
||||||
|
*
|
||||||
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application - mod_managed
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is
|
||||||
|
* Michael Giagnocavo <mgg@giagnocavo.net>
|
||||||
|
* Portions created by the Initial Developer are Copyright (C)
|
||||||
|
* the Initial Developer. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Contributor(s):
|
||||||
|
*
|
||||||
|
* Michael Giagnocavo <mgg@giagnocavo.net>
|
||||||
|
*
|
||||||
|
* ChannelVariables.cs -- Strongly typed channel variables for ManagedSession
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
|
namespace FreeSWITCH.Native {
|
||||||
|
public partial class ManagedSession {
|
||||||
|
|
||||||
|
// Need to find a better place to put these - then make them public
|
||||||
|
static readonly DateTime epoch = new DateTime(1970, 1, 1);
|
||||||
|
static DateTime epochUsToDateTime(long us){
|
||||||
|
return us == 0 ?
|
||||||
|
DateTime.MinValue :
|
||||||
|
epoch.AddMilliseconds((double)((decimal)us / 1000m));
|
||||||
|
}
|
||||||
|
static bool strToBool(string s) {
|
||||||
|
if (string.IsNullOrEmpty(s)) return false;
|
||||||
|
switch (s.ToLowerInvariant()) {
|
||||||
|
case "true":
|
||||||
|
case "yes":
|
||||||
|
case "on":
|
||||||
|
case "enable":
|
||||||
|
case "enabled":
|
||||||
|
case "active":
|
||||||
|
case "allow":
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
// Numbers are true
|
||||||
|
long tmp;
|
||||||
|
return long.TryParse(s, out tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static string boolToStr(bool b) {
|
||||||
|
return b ? "true" : "false";
|
||||||
|
}
|
||||||
|
|
||||||
|
ChannelVariables _variables; // Set on ManagedSession init
|
||||||
|
public ChannelVariables Variables {
|
||||||
|
get {
|
||||||
|
if (_variables == null) {
|
||||||
|
_variables = new ChannelVariables(this);
|
||||||
|
}
|
||||||
|
return _variables;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Strongly typed access to common variables</summary>
|
||||||
|
public class ChannelVariables {
|
||||||
|
readonly ManagedSession sess;
|
||||||
|
internal ChannelVariables(ManagedSession session) {
|
||||||
|
this.sess = session;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IDictionary<string, string> GetAllVariables() {
|
||||||
|
var dic = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
||||||
|
var evt = Native.freeswitch.switch_channel_variable_first(sess.channel);
|
||||||
|
while(evt != null) {
|
||||||
|
dic.Add(evt.name, evt.value);
|
||||||
|
evt = evt.next;
|
||||||
|
}
|
||||||
|
Native.freeswitch.switch_channel_variable_last(sess.channel);
|
||||||
|
return dic;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*** Settings ***/
|
||||||
|
const string bypass_media = "bypass_media";
|
||||||
|
public bool BypassMedia {
|
||||||
|
get { return strToBool(sess.GetVariable(bypass_media)); }
|
||||||
|
set { sess.SetVariable(bypass_media, boolToStr(value)); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/*** Times ***/
|
||||||
|
const string created_time = "created_time";
|
||||||
|
const string answered_time = "answered_time";
|
||||||
|
const string hangup_time = "hangup_time";
|
||||||
|
const string progress_time = "progress_time";
|
||||||
|
const string transfer_time = "transfer_time";
|
||||||
|
|
||||||
|
public DateTime CreatedTime {
|
||||||
|
get { return epochUsToDateTime(long.Parse(sess.GetVariable(created_time))); }
|
||||||
|
}
|
||||||
|
|
||||||
|
DateTime? readUsecsDateTime(string varName) {
|
||||||
|
var v = sess.GetVariable(varName);
|
||||||
|
if (string.IsNullOrEmpty(v)) return null;
|
||||||
|
else {
|
||||||
|
var d = epochUsToDateTime(long.Parse(v));
|
||||||
|
if (d == DateTime.MinValue) return null;
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public DateTime? AnsweredTime {
|
||||||
|
get { return readUsecsDateTime(answered_time); }
|
||||||
|
}
|
||||||
|
public DateTime? HangupTime {
|
||||||
|
get { return readUsecsDateTime(hangup_time); }
|
||||||
|
}
|
||||||
|
public DateTime? ProgressTime {
|
||||||
|
get { return readUsecsDateTime(progress_time); }
|
||||||
|
}
|
||||||
|
public DateTime? TransferTime {
|
||||||
|
get { return readUsecsDateTime(transfer_time); }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*** SIP Variables ***/
|
||||||
|
const string sofia_profile_name = "sofia_profile_name";
|
||||||
|
const string sip_received_ip = "sip_received_ip";
|
||||||
|
const string sip_received_port = "sip_received_port";
|
||||||
|
const string sip_via_protocol = "sip_via_protocol";
|
||||||
|
const string sip_from_user = "sip_from_user";
|
||||||
|
const string sip_from_uri = "sip_from_uri";
|
||||||
|
const string sip_from_host = "sip_from_host";
|
||||||
|
const string sip_req_user = "sip_req_user";
|
||||||
|
const string sip_req_uri = "sip_req_uri";
|
||||||
|
const string sip_req_host = "sip_req_host";
|
||||||
|
const string sip_to_user = "sip_to_user";
|
||||||
|
const string sip_to_uri = "sip_to_uri";
|
||||||
|
const string sip_to_host = "sip_to_host";
|
||||||
|
const string sip_contact_user = "sip_contact_user";
|
||||||
|
const string sip_contact_port = "sip_contact_port";
|
||||||
|
const string sip_contact_uri = "sip_contact_uri";
|
||||||
|
const string sip_contact_host = "sip_contact_host";
|
||||||
|
const string sip_call_id = "sip_call_id";
|
||||||
|
const string sip_destination_url = "sip_destination_url";
|
||||||
|
const string sip_term_status = "sip_term_status";
|
||||||
|
const string sip_term_cause = "sip_term_status";
|
||||||
|
const string switch_r_sdp = "switch_r_sdp";
|
||||||
|
const string switch_m_sdp = "switch_m_sdp";
|
||||||
|
const string sip_hangup_phrase = "sip_hangup_phrase";
|
||||||
|
|
||||||
|
|
||||||
|
short? readShort(string varName) {
|
||||||
|
var s = sess.GetVariable(varName);
|
||||||
|
if (string.IsNullOrEmpty(s)) return null;
|
||||||
|
short res;
|
||||||
|
if (short.TryParse(s, out res)) return res;
|
||||||
|
else return null;
|
||||||
|
}
|
||||||
|
int? readInt(string varName) {
|
||||||
|
var s = sess.GetVariable(varName);
|
||||||
|
if (string.IsNullOrEmpty(s)) return null;
|
||||||
|
int res;
|
||||||
|
if (int.TryParse(s, out res)) return res;
|
||||||
|
else return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// String suffix is added when the var is better represented as
|
||||||
|
// a different data type, but for now we return string
|
||||||
|
|
||||||
|
public string SofiaProfileName { get { return sess.GetVariable(sofia_profile_name); } }
|
||||||
|
public string SipCallID { get { return sess.GetVariable(sip_call_id); } }
|
||||||
|
public string SipReceivedIP { get { return sess.GetVariable(sip_received_ip); } }
|
||||||
|
public short? SipReceivedPort { get { return readShort(sip_received_port); } }
|
||||||
|
public string SipViaProtocolString { get { return sess.GetVariable(sip_via_protocol); } }
|
||||||
|
public string SipFromUser { get { return sess.GetVariable(sip_from_user); } }
|
||||||
|
public string SipFromUriString { get { return sess.GetVariable(sip_from_uri); } }
|
||||||
|
public string SipFromHost { get { return sess.GetVariable(sip_from_host); } }
|
||||||
|
public string SipReqUser { get { return sess.GetVariable(sip_req_user); } }
|
||||||
|
public string SipReqUriString { get { return sess.GetVariable(sip_req_uri); } }
|
||||||
|
public string SipReqHost { get { return sess.GetVariable(sip_req_host); } }
|
||||||
|
public string SipToUser { get { return sess.GetVariable(sip_to_user); } }
|
||||||
|
public string SipToUriString { get { return sess.GetVariable(sip_to_uri); } }
|
||||||
|
public string SipToHost { get { return sess.GetVariable(sip_to_host); } }
|
||||||
|
public string SipContactUser { get { return sess.GetVariable(sip_contact_user); } }
|
||||||
|
public string SipContactUriString { get { return sess.GetVariable(sip_contact_uri); } }
|
||||||
|
public string SipContactHost { get { return sess.GetVariable(sip_contact_host); } }
|
||||||
|
public short? SipContactPort { get { return readShort(sip_contact_port); } }
|
||||||
|
public string SipDestinationUrlString { get { return sess.GetVariable(sip_destination_url); } }
|
||||||
|
public int? SipTermStatus { get { return readInt(sip_term_status); } }
|
||||||
|
public int? SipTermCause { get { return readInt(sip_term_cause); } }
|
||||||
|
public string SwitchRSdp { get { return sess.GetVariable(switch_r_sdp); } }
|
||||||
|
public string SwitchMSdp { get { return sess.GetVariable(switch_m_sdp); } }
|
||||||
|
public string SipHangupPhrase { get { return sess.GetVariable(sip_hangup_phrase); } }
|
||||||
|
|
||||||
|
/*** Other ***/
|
||||||
|
|
||||||
|
const string proto_specific_hangup_cause = "proto_specific_hangup_cause";
|
||||||
|
const string hangup_cause = "hangup_cause";
|
||||||
|
const string hangup_cause_q850 = "hangup_cause_q850";
|
||||||
|
const string originate_disposition = "originate_disposition";
|
||||||
|
const string direction = "direction";
|
||||||
|
|
||||||
|
public string ProtoSpecificHangupCause { get { return sess.GetVariable(proto_specific_hangup_cause); } }
|
||||||
|
public string HangupCauseString { get { return sess.GetVariable(hangup_cause); } }
|
||||||
|
public int? HangupCauseQ850 { get { return readInt(hangup_cause_q850); } }
|
||||||
|
public string OriginateDispositionString { get { return sess.GetVariable(originate_disposition); } }
|
||||||
|
|
||||||
|
public Native.switch_call_direction_t CallDirection {
|
||||||
|
get {
|
||||||
|
var s = sess.GetVariable(direction);
|
||||||
|
if (string.IsNullOrEmpty(s)) return switch_call_direction_t.SWITCH_CALL_DIRECTION_INBOUND; // I guess
|
||||||
|
return s.ToLowerInvariant() == "inbound" ? switch_call_direction_t.SWITCH_CALL_DIRECTION_INBOUND : switch_call_direction_t.SWITCH_CALL_DIRECTION_OUTBOUND;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -48,6 +48,7 @@
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="AssemblyInfo.cs" />
|
<Compile Include="AssemblyInfo.cs" />
|
||||||
|
<Compile Include="ChannelVariables.cs" />
|
||||||
<Compile Include="ManagedSession.cs" />
|
<Compile Include="ManagedSession.cs" />
|
||||||
<Compile Include="Loader.cs" />
|
<Compile Include="Loader.cs" />
|
||||||
<Compile Include="Extensions.cs" />
|
<Compile Include="Extensions.cs" />
|
||||||
|
|
|
@ -52,6 +52,9 @@ namespace FreeSWITCH.Native
|
||||||
/// <summary>Initializes the native ManagedSession. Must be called after Originate.</summary>
|
/// <summary>Initializes the native ManagedSession. Must be called after Originate.</summary>
|
||||||
public void Initialize()
|
public void Initialize()
|
||||||
{
|
{
|
||||||
|
if (allocated == 0) {
|
||||||
|
Log.WriteLine(LogLevel.Critical, "Cannot initialize a ManagedSession until it is allocated (originated successfully).");
|
||||||
|
}
|
||||||
// P/Invoke generated function pointers stick around until the delegate is collected
|
// P/Invoke generated function pointers stick around until the delegate is collected
|
||||||
// By sticking the delegates in fields, their lifetime won't be less than the session
|
// By sticking the delegates in fields, their lifetime won't be less than the session
|
||||||
// So we don't need to worry about GCHandles and all that....
|
// So we don't need to worry about GCHandles and all that....
|
||||||
|
@ -59,6 +62,7 @@ namespace FreeSWITCH.Native
|
||||||
this._inputCallbackRef = inputCallback;
|
this._inputCallbackRef = inputCallback;
|
||||||
this._hangupCallbackRef = hangupCallback;
|
this._hangupCallbackRef = hangupCallback;
|
||||||
InitManagedSession(ManagedSession.getCPtr(this).Handle, this._inputCallbackRef, this._hangupCallbackRef);
|
InitManagedSession(ManagedSession.getCPtr(this).Handle, this._inputCallbackRef, this._hangupCallbackRef);
|
||||||
|
this._variables = new ChannelVariables(this);
|
||||||
}
|
}
|
||||||
DtmfCallback _inputCallbackRef;
|
DtmfCallback _inputCallbackRef;
|
||||||
CdeclAction _hangupCallbackRef;
|
CdeclAction _hangupCallbackRef;
|
||||||
|
@ -122,15 +126,9 @@ namespace FreeSWITCH.Native
|
||||||
get { return this.Ready(); }
|
get { return this.Ready(); }
|
||||||
}
|
}
|
||||||
|
|
||||||
Guid _uuid;
|
|
||||||
bool _uuidSet;
|
|
||||||
public Guid Uuid {
|
public Guid Uuid {
|
||||||
get {
|
get {
|
||||||
if (!_uuidSet) {
|
return new Guid(this.GetUuid());
|
||||||
_uuid = new Guid(this.GetUuid());
|
|
||||||
_uuidSet = true;
|
|
||||||
}
|
|
||||||
return _uuid;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue