Files
cgrates/sessionmanager/sessionmanager.go
Radu Ioan Fericean 02e73e3680 first real data tests
2012-05-16 15:03:10 +03:00

127 lines
3.3 KiB
Go

/*
Rating system designed to be used in VoIP Carriers World
Copyright (C) 2012 Radu Ioan Fericean
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>
*/
package sessionmanager
import (
"bufio"
"fmt"
"log"
"net"
)
// The freeswitch session manager type holding a buffer for the network connection,
// the active sessions, and a session delegate doing specific actions on every session.
type SessionManager struct {
buf *bufio.Reader
sessions []*Session
sessionDelegate SessionDelegate
}
// Connects to the freeswitch mod_event_socket server and starts
// listening for events in json format.
func (sm *SessionManager) Connect(ed SessionDelegate, address, pass string) {
if ed == nil {
log.Fatal("Please provide a non nil SessionDelegate")
}
sm.sessionDelegate = ed
conn, err := net.Dial("tcp", address)
if err != nil {
log.Fatal("Could not connect to freeswitch server!")
}
sm.buf = bufio.NewReaderSize(conn, 8192)
fmt.Fprint(conn, fmt.Sprintf("auth %s\n\n", pass))
fmt.Fprint(conn, "event json all\n\n")
go func() {
for {
sm.readNextEvent()
}
}()
}
// Reads from freeswitch server buffer until it encounters a '}',
// than it creates an event object and calls the appropriate method
// on the session delegate.
func (sm *SessionManager) readNextEvent() (ev *Event) {
body, err := sm.buf.ReadString('}')
if err != nil {
log.Print("Could not read from freeswitch connection!")
}
ev = NewEvent(body)
switch ev.Fields[NAME] {
case HEARTBEAT:
sm.OnHeartBeat(ev)
case ANSWER:
sm.OnChannelAnswer(ev)
case HANGUP:
sm.OnChannelHangupComplete(ev)
default:
sm.OnOther(ev)
}
return
}
// Searches and return the session with the specifed uuid
func (sm *SessionManager) GetSession(uuid string) *Session {
for _, s := range sm.sessions {
if s.uuid == uuid {
return s
}
}
return nil
}
// Called on freeswitch's hearbeat event
func (sm *SessionManager) OnHeartBeat(ev *Event) {
if sm.sessionDelegate != nil {
sm.sessionDelegate.OnHeartBeat(ev)
} else {
log.Print("heartbeat")
}
}
// Called on freeswitch's answer event
func (sm *SessionManager) OnChannelAnswer(ev *Event) {
if sm.sessionDelegate != nil {
s := NewSession(ev, sm.sessionDelegate)
sm.sessions = append(sm.sessions, s)
sm.sessionDelegate.OnChannelAnswer(ev, s)
} else {
log.Print("answer")
}
}
// Called on freeswitch's hangup event
func (sm *SessionManager) OnChannelHangupComplete(ev *Event) {
s := sm.GetSession(ev.Fields[UUID])
if sm.sessionDelegate != nil {
sm.sessionDelegate.OnChannelHangupComplete(ev, s)
} else {
log.Print("HangupComplete")
}
if s != nil {
s.Close()
}
}
// Called on freeswitch's events not processed by the session manger,
// for logging purposes (maybe).
func (sm *SessionManager) OnOther(ev *Event) {
//log.Printf("Other event: %s", ev.Fields["Event-Name"])
}