Files
cgrates/agents/janusagent_it_test.go
2024-07-14 14:09:44 +02:00

371 lines
8.7 KiB
Go

//go:build integration
// +build integration
/*
Real-time Online/Offline Charging System (OCS) for Telecom & ISP environments
Copyright (C) ITsysCOM GmbH
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 agents
import (
"bytes"
"encoding/json"
"fmt"
"io"
"net/http"
"os/exec"
"path"
"reflect"
"strconv"
"strings"
"testing"
"time"
"github.com/cgrates/birpc"
"github.com/cgrates/birpc/context"
"github.com/cgrates/cgrates/config"
"github.com/cgrates/cgrates/engine"
"github.com/cgrates/cgrates/sessions"
"github.com/cgrates/cgrates/utils"
janus "github.com/cgrates/janusgo"
"github.com/google/go-cmp/cmp"
"github.com/google/go-cmp/cmp/cmpopts"
"nhooyr.io/websocket"
)
var (
janCfgPath string
janCfgDIR string
janCfg *config.CGRConfig
janBin string
janRPC *birpc.Client
sTestsJanus = []func(t *testing.T){
testJanitInitCfg,
testJanitResetDB,
testJanCmd,
testJanitStartEngine,
testJanitApierRpcConn,
testJanitTPFromFolder,
testJanitAgent,
testJanitStopEngine,
}
)
func TestJanusit(t *testing.T) {
switch *utils.DBType {
case utils.MetaInternal:
janCfgDIR = "janus_agent"
case utils.MetaMySQL, utils.MetaMongo, utils.MetaPostgres:
t.SkipNow()
default:
t.Fatal("Unknown Database type")
}
janBin, err = exec.LookPath("/opt/janus/bin/janus")
if err != nil || janBin == "" {
t.SkipNow()
}
for _, stests := range sTestsJanus {
t.Run(janCfgDIR, stests)
}
}
func testJanitInitCfg(t *testing.T) {
var err error
janCfgPath = path.Join(*utils.DataDir, "conf", "samples", janCfgDIR)
janCfg, err = config.NewCGRConfigFromPath(janCfgPath)
if err != nil {
t.Fatal(err)
}
}
func testJanitResetDB(t *testing.T) {
if err := engine.InitDataDb(janCfg); err != nil {
t.Fatal(err)
}
if err := engine.InitStorDb(janCfg); err != nil {
t.Fatal(err)
}
}
func testJanitStartEngine(t *testing.T) {
if _, err := engine.StopStartEngine(janCfgPath, *utils.WaitRater); err != nil {
t.Fatal(err)
}
}
func testJanitApierRpcConn(t *testing.T) {
var err error
janRPC, err = newRPCClient(janCfg.ListenCfg()) // We connect over JSON so we can also troubleshoot if needed
if err != nil {
t.Fatal(err)
}
}
func testJanCmd(t *testing.T) {
cmd := exec.Command(janBin)
if err := cmd.Start(); err != nil {
t.Fatal(err)
}
}
func testJanitTPFromFolder(t *testing.T) {
attrs := &utils.AttrLoadTpFromFolder{FolderPath: path.Join(*utils.DataDir, "tariffplans", "tutorial")}
var loadInst utils.LoadInstance
if err := janRPC.Call(context.Background(), utils.APIerSv2LoadTariffPlanFromFolder, attrs, &loadInst); err != nil {
t.Error(err)
}
time.Sleep(100 * time.Millisecond)
}
func testJanitAgent(t *testing.T) {
var sessionMsg janus.SuccessMsg
id := utils.GenUUID()
msg := janus.BaseMsg{
Type: "create",
ID: id,
}
sessionurl := fmt.Sprintf("http://%s%s", janCfg.ListenCfg().HTTPListen, janCfg.JanusAgentCfg().URL)
body, err := sendReq(true, sessionurl, msg)
if err != nil {
t.Error(err)
}
exp := janus.SuccessMsg{
Type: "success",
ID: id,
}
if err := json.Unmarshal(body, &sessionMsg); err != nil {
t.Error(err)
}
if exp.ID != id || exp.Type != "success" {
t.Error(err)
}
var pluginMsg janus.SuccessMsg
id = utils.GenUUID()
msg = janus.BaseMsg{
Type: "attach",
Plugin: "janus.plugin.echotest",
ID: id,
}
url := fmt.Sprintf("http://%s%s/%d", janCfg.ListenCfg().HTTPListen, janCfg.JanusAgentCfg().URL, sessionMsg.Data.ID)
body, err = sendReq(true, url, msg)
if err != nil {
t.Error(err)
}
exp = janus.SuccessMsg{
Type: "success",
ID: id,
}
if err := json.Unmarshal(body, &pluginMsg); err != nil {
t.Error(err)
}
if exp.ID != id || exp.Type != "success" {
t.Error(err)
}
pullResp := make(chan []byte)
done := make(chan struct{})
go func() {
url := fmt.Sprintf("http://localhost%s%s/%d?rid=1714324194137&maxev=10", janCfg.ListenCfg().HTTPListen, janCfg.JanusAgentCfg().URL, sessionMsg.Data.ID)
ticker := time.NewTicker(50 * time.Millisecond)
for {
select {
case <-ticker.C:
body, err := sendReq(false, url, nil)
if err != nil {
continue
}
pullResp <- body
case <-done:
return
}
}
}()
id = utils.GenUUID()
messagePlugin := janus.HandlerMessage{
BaseMsg: janus.BaseMsg{
Type: "message",
ID: id,
},
Body: map[string]any{
"audio": true,
"video": true,
},
}
url = fmt.Sprintf("http://%s%s/%d/%d", janCfg.ListenCfg().HTTPListen, janCfg.JanusAgentCfg().URL, sessionMsg.Data.ID, pluginMsg.Data.ID)
body, err = sendReq(true, url, messagePlugin)
if err != nil {
t.Error(err)
}
expAck := janus.AckMsg{
Type: "ack",
ID: id,
Hint: "I'm taking my time!",
}
var ackMsg janus.AckMsg
if err := json.Unmarshal(body, &ackMsg); err != nil {
t.Error(err)
}
if !reflect.DeepEqual(ackMsg, expAck) {
t.Errorf("Expected <%s> ,got <%s> instead", utils.ToJSON(expAck), utils.ToJSON(ackMsg))
}
var eventMsg []janus.EventMsg
eventmsg := <-pullResp
if err = json.Unmarshal(eventmsg, &eventMsg); err != nil {
t.Error(err)
}
expEvent := []janus.EventMsg{
{
ID: id,
Session: sessionMsg.Data.ID,
Handle: pluginMsg.Data.ID,
Type: "event",
Plugindata: janus.PluginData{
Plugin: "janus.plugin.echotest",
Data: map[string]interface{}{
"echotest": "event",
"result": "ok",
},
},
},
}
if !reflect.DeepEqual(eventMsg, expEvent) {
t.Errorf("expected <%+v>,got <%+v instead", expEvent, eventMsg)
}
close(done)
initArgs := &sessions.V1InitSessionArgs{
GetAttributes: true,
InitSession: true,
CGREvent: &utils.CGREvent{
Tenant: janCfg.GeneralCfg().DefaultTenant,
ID: utils.Sha1(),
Time: utils.TimePointer(time.Now()),
Event: map[string]any{
utils.AccountField: strings.Split("192.168.56.1", ":")[0],
utils.OriginHost: strings.Split("localhost", ":")[0],
utils.OriginID: strconv.Itoa(int(sessionMsg.Data.ID)),
utils.Destination: "testecho",
utils.AnswerTime: time.Now(),
},
},
ForceDuration: true,
}
expSess := &sessions.V1InitSessionReply{
MaxUsage: utils.DurationPointer(3 * time.Hour),
}
rply := new(sessions.V1InitSessionReply)
if err = janRPC.Call(context.Background(), utils.SessionSv1InitiateSession, initArgs, rply); err != nil {
t.Error(err)
} else if diff := cmp.Diff(expSess, rply, cmpopts.IgnoreUnexported(sessions.V1InitSessionReply{})); diff != "" {
t.Errorf("%s returned unexpected reply (-expected +want)\n%s ", utils.SessionSv1InitiateSession, diff)
}
var replyActSess []*sessions.ExternalSession
if err := janRPC.Call(context.Background(), utils.SessionSv1GetActiveSessions, utils.SessionFilter{}, &replyActSess); err != nil {
t.Error(err)
} else if len(replyActSess) != 2 {
t.Errorf("expected 2 active sessions ,got %d ", len(replyActSess))
}
conn, _, err := websocket.Dial(context.Background(), fmt.Sprintf("ws://%s", janCfg.JanusAgentCfg().JanusConns[0].Address), &websocket.DialOptions{
Subprotocols: []string{"janus-protocol"},
})
if err != nil {
t.Error(err)
}
defer conn.CloseNow()
msg = janus.BaseMsg{Type: "destroy", ID: id, Session: sessionMsg.Data.ID}
data, err := json.Marshal(msg)
if err != nil {
return
}
if err := conn.Write(context.Background(), websocket.MessageText, data); err != nil {
t.Error(err)
}
var reply string
if err = janRPC.Call(context.Background(), utils.SessionSv1SyncSessions, &utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != utils.OK {
t.Errorf("expected reply %q,received %q", utils.OK, reply)
}
}
func sendReq(isPost bool, url string, msg any) (body []byte, err error) {
var (
req *http.Request
data []byte
)
if isPost {
data, err = json.Marshal(msg)
if err != nil {
return
}
req, err = http.NewRequest(http.MethodPost, url, bytes.NewReader(data))
} else {
req, err = http.NewRequest(http.MethodGet, url, nil)
}
if err != nil {
return
}
res, err := http.DefaultClient.Do(req)
if err != nil {
return
}
defer res.Body.Close()
body, err = io.ReadAll(res.Body)
if err != nil {
return
}
return
}
func testJanitStopEngine(t *testing.T) {
if err := engine.KillEngine(*utils.WaitRater); err != nil {
t.Error(err)
}
if err := exec.Command("pkill", "janus").Run(); err != nil {
t.Error(err)
}
}