/* 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 */ package agents import ( "bufio" "bytes" "net/http" "reflect" "strings" "testing" "time" "github.com/cgrates/cgrates/config" "github.com/cgrates/cgrates/engine" "github.com/cgrates/cgrates/utils" "github.com/cgrates/radigo" "github.com/fiorix/go-diameter/diam" "github.com/fiorix/go-diameter/diam/avp" "github.com/fiorix/go-diameter/diam/datatype" ) func TestAgReqAsNavigableMap(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.CGRID}, utils.Sha1("dsafdsaf", time.Date(2013, 11, 7, 8, 42, 26, 0, time.UTC).String()), false, false) agReq.CGRRequest.Set([]string{utils.ToR}, utils.VOICE, false, false) agReq.CGRRequest.Set([]string{utils.Account}, "1001", false, false) agReq.CGRRequest.Set([]string{utils.Destination}, "1002", false, false) agReq.CGRRequest.Set([]string{utils.AnswerTime}, time.Date(2013, 12, 30, 15, 0, 1, 0, time.UTC), false, false) agReq.CGRRequest.Set([]string{utils.RequestType}, utils.META_PREPAID, false, false) agReq.CGRRequest.Set([]string{utils.Usage}, time.Duration(3*time.Minute), false, false) cgrRply := map[string]interface{}{ utils.CapAttributes: map[string]interface{}{ "PaypalAccount": "cgrates@paypal.com", }, utils.CapMaxUsage: time.Duration(120 * time.Second), utils.Error: "", } agReq.CGRReply = config.NewNavigableMap(cgrRply) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Tenant", FieldId: utils.Tenant, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("cgrates.org", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Account", FieldId: utils.Account, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Account", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Destination", FieldId: utils.Destination, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Destination", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "RequestedUsageVoice", FieldId: "RequestedUsage", Type: utils.META_COMPOSED, Filters: []string{"*string:~*cgreq.ToR:*voice"}, Value: config.NewRSRParsersMustCompile( "~*cgreq.Usage{*duration_seconds}", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "RequestedUsageData", FieldId: "RequestedUsage", Type: utils.META_COMPOSED, Filters: []string{"*string:~*cgreq.ToR:*data"}, Value: config.NewRSRParsersMustCompile( "~*cgreq.Usage{*duration_nanoseconds}", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "RequestedUsageSMS", FieldId: "RequestedUsage", Type: utils.META_COMPOSED, Filters: []string{"*string:~*cgreq.ToR:*sms"}, Value: config.NewRSRParsersMustCompile( "~*cgreq.Usage{*duration_nanoseconds}", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "AttrPaypalAccount", FieldId: "PaypalAccount", Type: utils.META_COMPOSED, Filters: []string{"*string:~*cgrep.Error:"}, Value: config.NewRSRParsersMustCompile( "~*cgrep.Attributes.PaypalAccount", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "MaxUsage", FieldId: "MaxUsage", Type: utils.META_COMPOSED, Filters: []string{"*string:~*cgrep.Error:"}, Value: config.NewRSRParsersMustCompile( "~*cgrep.MaxUsage{*duration_seconds}", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Error", FieldId: "Error", Type: utils.META_COMPOSED, Filters: []string{"*rsr::~*cgrep.Error(!^$)"}, Value: config.NewRSRParsersMustCompile( "~*cgrep.Error", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{utils.Tenant}, []*config.NMItem{ &config.NMItem{Data: "cgrates.org", Path: []string{utils.Tenant}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{utils.Account}, []*config.NMItem{ &config.NMItem{Data: "1001", Path: []string{utils.Account}, Config: tplFlds[1]}}, false, true) eMp.Set([]string{utils.Destination}, []*config.NMItem{ &config.NMItem{Data: "1002", Path: []string{utils.Destination}, Config: tplFlds[2]}}, false, true) eMp.Set([]string{"RequestedUsage"}, []*config.NMItem{ &config.NMItem{Data: "180", Path: []string{"RequestedUsage"}, Config: tplFlds[3]}}, false, true) eMp.Set([]string{"PaypalAccount"}, []*config.NMItem{ &config.NMItem{Data: "cgrates@paypal.com", Path: []string{"PaypalAccount"}, Config: tplFlds[6]}}, false, true) eMp.Set([]string{"MaxUsage"}, []*config.NMItem{ &config.NMItem{Data: "120", Path: []string{"MaxUsage"}, Config: tplFlds[7]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqMaxCost(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.CapMaxUsage}, "120s", false, false) cgrRply := map[string]interface{}{ utils.CapMaxUsage: time.Duration(120 * time.Second), } agReq.CGRReply = config.NewNavigableMap(cgrRply) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "MaxUsage", FieldId: "MaxUsage", Type: utils.META_COMPOSED, Filters: []string{"*rsr::~*cgrep.MaxUsage(>0s)"}, Value: config.NewRSRParsersMustCompile( "~*cgrep.MaxUsage{*duration_seconds}", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{"MaxUsage"}, []*config.NMItem{ &config.NMItem{Data: "120", Path: []string{"MaxUsage"}, Config: tplFlds[0]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqParseFieldDiameter(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "MandatoryFalse", FieldId: "MandatoryFalse", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryFalse", true, utils.INFIELD_SEP), Mandatory: false}, &config.FCTemplate{Tag: "MandatoryTrue", FieldId: "MandatoryTrue", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryTrue", true, utils.INFIELD_SEP), Mandatory: true}, &config.FCTemplate{Tag: "Session-Id", Filters: []string{}, FieldId: "Session-Id", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } if _, err := agReq.ParseField(tplFlds[1]); err == nil || err.Error() != "Empty source value for fieldID: " { t.Error(err) } expected = "simuhuawei;1449573472;00002" if out, err := agReq.ParseField(tplFlds[2]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } } func TestAgReqParseFieldRadius(t *testing.T) { //creater radius message pkt := radigo.NewPacket(radigo.AccountingRequest, 1, dictRad, coder, "CGRateS.org") if err := pkt.AddAVPWithName("User-Name", "flopsy", ""); err != nil { t.Error(err) } if err := pkt.AddAVPWithName("Cisco-NAS-Port", "CGR1", "Cisco"); err != nil { t.Error(err) } //create radiusDataProvider dP := newRADataProvider(pkt) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "MandatoryFalse", FieldId: "MandatoryFalse", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryFalse", true, utils.INFIELD_SEP), Mandatory: false}, &config.FCTemplate{Tag: "MandatoryTrue", FieldId: "MandatoryTrue", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryTrue", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } if _, err := agReq.ParseField(tplFlds[1]); err == nil || err.Error() != "Empty source value for fieldID: " { t.Error(err) } } func TestAgReqParseFieldHttpUrl(t *testing.T) { //creater radius message br := bufio.NewReader(strings.NewReader(`GET /cdr?request_type=MOSMS_CDR×tamp=2008-08-15%2017:49:21&message_date=2008-08-15%2017:49:21&transactionid=100744&CDR_ID=123456&carrierid=1&mcc=222&mnc=10&imsi=235180000000000&msisdn=%2B4977000000000&destination=%2B497700000001&message_status=0&IOT=0&service_id=1 HTTP/1.1 Host: api.cgrates.org `)) req, err := http.ReadRequest(br) if err != nil { t.Error(err) } //create radiusDataProvider dP, _ := newHTTPUrlDP(req) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "MandatoryFalse", FieldId: "MandatoryFalse", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryFalse", true, utils.INFIELD_SEP), Mandatory: false}, &config.FCTemplate{Tag: "MandatoryTrue", FieldId: "MandatoryTrue", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryTrue", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } if _, err := agReq.ParseField(tplFlds[1]); err == nil || err.Error() != "Empty source value for fieldID: " { t.Error(err) } } func TestAgReqParseFieldHttpXml(t *testing.T) { //creater radius message body := ` 2005-08-26T14:16:42 2005-08-26T14:16:56 2005-08-26T14:17:34 My Call Reference 386 sampleusername 1 Conecto LLC US$0.21 yes no US$0.13 44 +441624828505 Isle of Man 38 US$0.0200 US$0.0140 US$0.0130 US$0.0082 +44 7624 494075 Isle of Man 37 US$0.2700 US$0.1890 US$0.1880 US$0.1159 ` req, err := http.NewRequest("POST", "http://localhost:8080/", bytes.NewBuffer([]byte(body))) if err != nil { t.Error(err) } //create radiusDataProvider dP, _ := newHTTPXmlDP(req) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "MandatoryFalse", FieldId: "MandatoryFalse", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryFalse", true, utils.INFIELD_SEP), Mandatory: false}, &config.FCTemplate{Tag: "MandatoryTrue", FieldId: "MandatoryTrue", Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*req.MandatoryTrue", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } if _, err := agReq.ParseField(tplFlds[1]); err == nil || err.Error() != "Empty source value for fieldID: " { t.Error(err) } } func TestAgReqEmptyFilter(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.CGRID}, utils.Sha1("dsafdsaf", time.Date(2013, 11, 7, 8, 42, 26, 0, time.UTC).String()), false, false) agReq.CGRRequest.Set([]string{utils.Account}, "1001", false, false) agReq.CGRRequest.Set([]string{utils.Destination}, "1002", false, false) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Tenant", Filters: []string{}, FieldId: utils.Tenant, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("cgrates.org", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Account", Filters: []string{}, FieldId: utils.Account, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Account", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Destination", Filters: []string{}, FieldId: utils.Destination, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Destination", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{utils.Tenant}, []*config.NMItem{ &config.NMItem{Data: "cgrates.org", Path: []string{utils.Tenant}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{utils.Account}, []*config.NMItem{ &config.NMItem{Data: "1001", Path: []string{utils.Account}, Config: tplFlds[1]}}, false, true) eMp.Set([]string{utils.Destination}, []*config.NMItem{ &config.NMItem{Data: "1002", Path: []string{utils.Destination}, Config: tplFlds[2]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqMetaExponent(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) agReq.CGRRequest.Set([]string{"Value"}, "2", false, false) agReq.CGRRequest.Set([]string{"Exponent"}, "2", false, false) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "TestExpo", Filters: []string{}, FieldId: "TestExpo", Type: utils.MetaValueExponent, Value: config.NewRSRParsersMustCompile("~*cgreq.Value;~*cgreq.Exponent", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{"TestExpo"}, []*config.NMItem{ &config.NMItem{Data: "200", Path: []string{"TestExpo"}, Config: tplFlds[0]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, \n received: %+v", eMp, mpOut) } } func TestAgReqCGRActiveRequest(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Value1", Filters: []string{}, FieldId: "Value1", Type: utils.META_CONSTANT, Value: config.NewRSRParsersMustCompile("12", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Value2", Filters: []string{}, FieldId: "Value2", Type: utils.META_CONSTANT, Value: config.NewRSRParsersMustCompile("1", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Value3", Filters: []string{}, FieldId: "Value3", Type: utils.META_CONSTANT, Value: config.NewRSRParsersMustCompile("2", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Diff", Filters: []string{}, FieldId: "Diff", Type: utils.MetaDifference, Value: config.NewRSRParsersMustCompile("~*cgrareq.Value1;~*cgrareq.Value2;~*cgrareq.Value3", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{"Value1"}, []*config.NMItem{ &config.NMItem{Data: "12", Path: []string{"Value1"}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{"Value2"}, []*config.NMItem{ &config.NMItem{Data: "1", Path: []string{"Value2"}, Config: tplFlds[1]}}, false, true) eMp.Set([]string{"Value3"}, []*config.NMItem{ &config.NMItem{Data: "2", Path: []string{"Value3"}, Config: tplFlds[2]}}, false, true) eMp.Set([]string{"Diff"}, []*config.NMItem{ &config.NMItem{Data: int64(9), Path: []string{"Diff"}, Config: tplFlds[3]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v,\n received: %+v", eMp, mpOut) } } func TestAgReqFieldAsNone(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.ToR}, utils.VOICE, false, false) agReq.CGRRequest.Set([]string{utils.Account}, "1001", false, false) agReq.CGRRequest.Set([]string{utils.Destination}, "1002", false, false) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Tenant", FieldId: utils.Tenant, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("cgrates.org", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Account", FieldId: utils.Account, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Account", true, utils.INFIELD_SEP)}, &config.FCTemplate{Type: utils.META_NONE, Blocker: true}, &config.FCTemplate{Tag: "Destination", FieldId: utils.Destination, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Destination", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{utils.Tenant}, []*config.NMItem{ &config.NMItem{Data: "cgrates.org", Path: []string{utils.Tenant}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{utils.Account}, []*config.NMItem{ &config.NMItem{Data: "1001", Path: []string{utils.Account}, Config: tplFlds[1]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqFieldAsNone2(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.ToR}, utils.VOICE, false, false) agReq.CGRRequest.Set([]string{utils.Account}, "1001", false, false) agReq.CGRRequest.Set([]string{utils.Destination}, "1002", false, false) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Tenant", FieldId: utils.Tenant, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("cgrates.org", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Account", FieldId: utils.Account, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Account", true, utils.INFIELD_SEP)}, &config.FCTemplate{Type: utils.META_NONE}, &config.FCTemplate{Tag: "Destination", FieldId: utils.Destination, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Destination", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{utils.Tenant}, []*config.NMItem{ &config.NMItem{Data: "cgrates.org", Path: []string{utils.Tenant}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{utils.Account}, []*config.NMItem{ &config.NMItem{Data: "1001", Path: []string{utils.Account}, Config: tplFlds[1]}}, false, true) eMp.Set([]string{utils.Destination}, []*config.NMItem{ &config.NMItem{Data: "1002", Path: []string{utils.Destination}, Config: tplFlds[3]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqAsNavigableMap2(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRRequest.Set([]string{utils.ToR}, utils.VOICE, false, false) agReq.CGRRequest.Set([]string{utils.Account}, "1001", false, false) agReq.CGRRequest.Set([]string{utils.Destination}, "1002", false, false) agReq.CGRRequest.Set([]string{utils.AnswerTime}, time.Date(2013, 12, 30, 15, 0, 1, 0, time.UTC), false, false) agReq.CGRRequest.Set([]string{utils.RequestType}, utils.META_PREPAID, false, false) agReq.CGRReply = config.NewNavigableMap(nil) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Tenant", FieldId: utils.Tenant, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("cgrates.org", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Account", FieldId: utils.Account, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Account", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Destination", FieldId: utils.Destination, Type: utils.META_COMPOSED, Value: config.NewRSRParsersMustCompile("~*cgreq.Destination", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Usage", FieldId: utils.Usage, Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("30s", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "CalculatedUsage", FieldId: "CalculatedUsage", Filters: []string{"*gt:~*cgrareq.Usage:0"}, Type: "*difference", Value: config.NewRSRParsersMustCompile("~*cgreq.AnswerTime;~*cgrareq.Usage", true, utils.INFIELD_SEP), }, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{utils.Tenant}, []*config.NMItem{ &config.NMItem{Data: "cgrates.org", Path: []string{utils.Tenant}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{utils.Account}, []*config.NMItem{ &config.NMItem{Data: "1001", Path: []string{utils.Account}, Config: tplFlds[1]}}, false, true) eMp.Set([]string{utils.Destination}, []*config.NMItem{ &config.NMItem{Data: "1002", Path: []string{utils.Destination}, Config: tplFlds[2]}}, false, true) eMp.Set([]string{"Usage"}, []*config.NMItem{ &config.NMItem{Data: "30s", Path: []string{"Usage"}, Config: tplFlds[3]}}, false, true) eMp.Set([]string{"CalculatedUsage"}, []*config.NMItem{ &config.NMItem{Data: time.Date(2013, 12, 30, 14, 59, 31, 0, time.UTC), Path: []string{"CalculatedUsage"}, Config: tplFlds[4]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqFieldAsInterface(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) agReq := newAgentRequest(nil, nil, nil, nil, nil, "cgrates.org", "", filterS) // populate request, emulating the way will be done in HTTPAgent agReq.CGRAReq = config.NewNavigableMap(nil) agReq.CGRAReq.Set([]string{utils.Usage}, []*config.NMItem{{Data: 3 * time.Minute}}, false, false) agReq.CGRAReq.Set([]string{utils.ToR}, []*config.NMItem{{Data: utils.VOICE}}, false, false) agReq.CGRAReq.Set([]string{utils.Account}, "1001", false, false) agReq.CGRAReq.Set([]string{utils.Destination}, "1002", false, false) path := []string{"*cgrareq", utils.Usage} var expVal interface{} expVal = 3 * time.Minute if rply, err := agReq.FieldAsInterface(path); err != nil { t.Error(err) } else if !reflect.DeepEqual(rply, expVal) { t.Errorf("Expected %v , received: %v", utils.ToJSON(expVal), utils.ToJSON(rply)) } path = []string{"*cgrareq", utils.ToR} expVal = utils.VOICE if rply, err := agReq.FieldAsInterface(path); err != nil { t.Error(err) } else if !reflect.DeepEqual(rply, expVal) { t.Errorf("Expected %v , received: %v", utils.ToJSON(expVal), utils.ToJSON(rply)) } path = []string{"*cgrareq", utils.Account} expVal = "1001" if rply, err := agReq.FieldAsInterface(path); err != nil { t.Error(err) } else if !reflect.DeepEqual(rply, expVal) { t.Errorf("Expected %v , received: %v", utils.ToJSON(expVal), utils.ToJSON(rply)) } path = []string{"*cgrareq", utils.Destination} expVal = "1002" if rply, err := agReq.FieldAsInterface(path); err != nil { t.Error(err) } else if !reflect.DeepEqual(rply, expVal) { t.Errorf("Expected %v , received: %v", utils.ToJSON(expVal), utils.ToJSON(rply)) } } func TestAgReqNewARWithCGRRplyAndRply(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) ev := map[string]interface{}{ "FirstLevel": map[string]interface{}{ "SecondLevel": map[string]interface{}{ "Fld1": "Val1", }, }, } rply := config.NewNavigableMap(ev) ev2 := map[string]interface{}{ utils.CapAttributes: map[string]interface{}{ "PaypalAccount": "cgrates@paypal.com", }, utils.CapMaxUsage: time.Duration(120 * time.Second), utils.Error: "", } cgrRply := config.NewNavigableMap(ev2) agReq := newAgentRequest(nil, nil, cgrRply, rply, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Fld1", FieldId: "Fld1", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*rep.FirstLevel.SecondLevel.Fld1", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Fld2", FieldId: "Fld2", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*cgrep.Attributes.PaypalAccount", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{"Fld1"}, []*config.NMItem{ &config.NMItem{Data: "Val1", Path: []string{"Fld1"}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{"Fld2"}, []*config.NMItem{ &config.NMItem{Data: "cgrates@paypal.com", Path: []string{"Fld2"}, Config: tplFlds[1]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, received: %+v", eMp, mpOut) } } func TestAgReqSetCGRReplyWithError(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) ev := map[string]interface{}{ "FirstLevel": map[string]interface{}{ "SecondLevel": map[string]interface{}{ "Fld1": "Val1", }, }, } rply := config.NewNavigableMap(ev) agReq := newAgentRequest(nil, nil, nil, rply, nil, "cgrates.org", "", filterS) agReq.setCGRReply(nil, utils.ErrNotFound) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Fld1", FieldId: "Fld1", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*rep.FirstLevel.SecondLevel.Fld1", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Fld2", FieldId: "Fld2", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*cgrep.Attributes.PaypalAccount", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.AsNavigableMap(tplFlds); err == nil || err.Error() != "NOT_FOUND:Fld2" { t.Error(err) } } type myEv map[string]interface{} func (ev myEv) AsNavigableMap(tpl []*config.FCTemplate) (*config.NavigableMap, error) { return config.NewNavigableMap(ev), nil } func TestAgReqSetCGRReplyWithoutError(t *testing.T) { data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) ev := map[string]interface{}{ "FirstLevel": map[string]interface{}{ "SecondLevel": map[string]interface{}{ "Fld1": "Val1", }, }, } rply := config.NewNavigableMap(ev) myEv := myEv{ utils.CapAttributes: map[string]interface{}{ "PaypalAccount": "cgrates@paypal.com", }, utils.CapMaxUsage: time.Duration(120 * time.Second), utils.Error: "", } agReq := newAgentRequest(nil, nil, nil, rply, nil, "cgrates.org", "", filterS) agReq.setCGRReply(myEv, nil) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Fld1", FieldId: "Fld1", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*rep.FirstLevel.SecondLevel.Fld1", true, utils.INFIELD_SEP)}, &config.FCTemplate{Tag: "Fld2", FieldId: "Fld2", Type: utils.MetaVariable, Value: config.NewRSRParsersMustCompile("~*cgrep.Attributes.PaypalAccount", true, utils.INFIELD_SEP)}, } eMp := config.NewNavigableMap(nil) eMp.Set([]string{"Fld1"}, []*config.NMItem{ &config.NMItem{Data: "Val1", Path: []string{"Fld1"}, Config: tplFlds[0]}}, false, true) eMp.Set([]string{"Fld2"}, []*config.NMItem{ &config.NMItem{Data: "cgrates@paypal.com", Path: []string{"Fld2"}, Config: tplFlds[1]}}, false, true) if mpOut, err := agReq.AsNavigableMap(tplFlds); err != nil { t.Error(err) } else if !reflect.DeepEqual(eMp, mpOut) { t.Errorf("expecting: %+v, \n received: %+v", eMp, mpOut) } } func TestAgReqParseFieldMetaCCUsage(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "CCUsage", Filters: []string{}, FieldId: "CCUsage", Type: utils.MetaCCUsage, Value: config.NewRSRParsersMustCompile("~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid arguments <[{"Rules":"~*req.Session-Id","AllFiltersMatch":true}]> to *cc_usage` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "CCUsage", Filters: []string{}, FieldId: "CCUsage", Type: utils.MetaCCUsage, Value: config.NewRSRParsersMustCompile("~*req.Session-Id;12s;12s", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid requestNumber to *cc_usage` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "CCUsage", Filters: []string{}, FieldId: "CCUsage", Type: utils.MetaCCUsage, Value: config.NewRSRParsersMustCompile("10;~*req.Session-Id;12s", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid usedCCTime to *cc_usage` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "CCUsage", Filters: []string{}, FieldId: "CCUsage", Type: utils.MetaCCUsage, Value: config.NewRSRParsersMustCompile("10;12s;~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid debitInterval to *cc_usage` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "CCUsage", Filters: []string{}, FieldId: "CCUsage", Type: utils.MetaCCUsage, Value: config.NewRSRParsersMustCompile("3;10s;5s", true, utils.INFIELD_SEP), Mandatory: true}, } //5s*2 + 10s expected := time.Duration(20 * time.Second) if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } } func TestAgReqParseFieldMetaUsageDifference(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Usage", Filters: []string{}, FieldId: "Usage", Type: utils.META_USAGE_DIFFERENCE, Value: config.NewRSRParsersMustCompile("~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid arguments <[{"Rules":"~*req.Session-Id","AllFiltersMatch":true}]> to *usage_difference` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "Usage", Filters: []string{}, FieldId: "Usage", Type: utils.META_USAGE_DIFFERENCE, Value: config.NewRSRParsersMustCompile("1560325161;~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `Unsupported time format` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "Usage", Filters: []string{}, FieldId: "Usage", Type: utils.META_USAGE_DIFFERENCE, Value: config.NewRSRParsersMustCompile("~*req.Session-Id;1560325161", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `Unsupported time format` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "Usage", Filters: []string{}, FieldId: "Usage", Type: utils.META_USAGE_DIFFERENCE, Value: config.NewRSRParsersMustCompile("1560325161;1560325151", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "10s" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, received: <%+v>", expected, out) } } func TestAgReqParseFieldMetaSum(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Sum", Filters: []string{}, FieldId: "Sum", Type: utils.MetaSum, Value: config.NewRSRParsersMustCompile("15;~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `strconv.ParseInt: parsing "simuhuawei;1449573472;00002": invalid syntax` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "Sum", Filters: []string{}, FieldId: "Sum", Type: utils.MetaSum, Value: config.NewRSRParsersMustCompile("15;15", true, utils.INFIELD_SEP), Mandatory: true}, } expected := int64(30) if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, %T received: <%+v> %T", expected, expected, out, out) } } func TestAgReqParseFieldMetaDifference(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "Diff", Filters: []string{}, FieldId: "Diff", Type: utils.MetaDifference, Value: config.NewRSRParsersMustCompile("15;~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `strconv.ParseInt: parsing "simuhuawei;1449573472;00002": invalid syntax` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "Diff", Filters: []string{}, FieldId: "Diff", Type: utils.MetaDifference, Value: config.NewRSRParsersMustCompile("15;12;2", true, utils.INFIELD_SEP), Mandatory: true}, } expected := int64(1) if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, %T received: <%+v> %T", expected, expected, out, out) } } func TestAgReqParseFieldMetaValueExponent(t *testing.T) { //creater diameter message m := diam.NewRequest(diam.CreditControl, 4, nil) m.NewAVP("Session-Id", avp.Mbit, 0, datatype.UTF8String("simuhuawei;1449573472;00002")) m.NewAVP("Subscription-Id", avp.Mbit, 0, &diam.GroupedAVP{ AVP: []*diam.AVP{ diam.NewAVP(450, avp.Mbit, 0, datatype.Enumerated(2)), // Subscription-Id-Type diam.NewAVP(444, avp.Mbit, 0, datatype.UTF8String("208708000004")), // Subscription-Id-Data diam.NewAVP(avp.ValueDigits, avp.Mbit, 0, datatype.Integer64(20000)), }}) //create diameterDataProvider dP := newDADataProvider(nil, m) data, _ := engine.NewMapStorage() dm := engine.NewDataManager(data) cfg, _ := config.NewDefaultCGRConfig() filterS := engine.NewFilterS(cfg, nil, nil, dm) //pass the data provider to agent request agReq := newAgentRequest(dP, nil, nil, nil, nil, "cgrates.org", "", filterS) tplFlds := []*config.FCTemplate{ &config.FCTemplate{Tag: "ValExp", Filters: []string{}, FieldId: "ValExp", Type: utils.MetaValueExponent, Value: config.NewRSRParsersMustCompile("~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid arguments <[{"Rules":"~*req.Session-Id","AllFiltersMatch":true}]> to *value_exponent` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "ValExp", Filters: []string{}, FieldId: "ValExp", Type: utils.MetaValueExponent, Value: config.NewRSRParsersMustCompile("15;~*req.Session-Id", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `strconv.Atoi: parsing "simuhuawei;1449573472;00002": invalid syntax` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "ValExp", Filters: []string{}, FieldId: "ValExp", Type: utils.MetaValueExponent, Value: config.NewRSRParsersMustCompile("~*req.Session-Id;15", true, utils.INFIELD_SEP), Mandatory: true}, } if _, err := agReq.ParseField(tplFlds[0]); err == nil || err.Error() != `invalid value to *value_exponent` { t.Error(err) } tplFlds = []*config.FCTemplate{ &config.FCTemplate{Tag: "ValExp", Filters: []string{}, FieldId: "ValExp", Type: utils.MetaValueExponent, Value: config.NewRSRParsersMustCompile("2;3", true, utils.INFIELD_SEP), Mandatory: true}, } expected := "2000" if out, err := agReq.ParseField(tplFlds[0]); err != nil { t.Error(err) } else if !reflect.DeepEqual(out, expected) { t.Errorf("expecting: <%+v>, %T received: <%+v> %T", expected, expected, out, out) } }