/* 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 Affero 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 Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see */ package agents import ( "encoding/xml" "fmt" "io" "net/http" "net/http/httputil" "strconv" "strings" "github.com/antchfx/xmlquery" "github.com/cgrates/cgrates/utils" ) // newHADataProvider constructs a DataProvider func newHADataProvider(reqPayload string, req *http.Request) (dP utils.DataProvider, err error) { switch reqPayload { default: return nil, fmt.Errorf("unsupported decoder type <%s>", reqPayload) case utils.MetaUrl: return newHTTPUrlDP(req) case utils.MetaXml: return newHTTPXmlDP(req) } } func newHTTPUrlDP(req *http.Request) (dP utils.DataProvider, err error) { dP = &httpUrlDP{req: req, cache: utils.MapStorage{}} return } // httpUrlDP implements utils.DataProvider, serving as url data decoder // decoded data is only searched once and cached type httpUrlDP struct { req *http.Request cache utils.MapStorage } // String is part of utils.DataProvider interface // when called, it will display the already parsed values out of cache func (hU *httpUrlDP) String() string { byts, _ := httputil.DumpRequest(hU.req, true) return string(byts) } // FieldAsInterface is part of utils.DataProvider interface func (hU *httpUrlDP) FieldAsInterface(fldPath []string) (data any, err error) { if len(fldPath) != 1 { return nil, utils.ErrNotFound } if data, err = hU.cache.FieldAsInterface(fldPath); err != nil { if err != utils.ErrNotFound { // item found in cache return } err = nil // cancel previous err } else { return // data found in cache } data = hU.req.FormValue(fldPath[0]) hU.cache.Set(fldPath, data) return } // FieldAsString is part of utils.DataProvider interface func (hU *httpUrlDP) FieldAsString(fldPath []string) (data string, err error) { var valIface any valIface, err = hU.FieldAsInterface(fldPath) if err != nil { return } return utils.IfaceAsString(valIface), nil } func newHTTPXmlDP(req *http.Request) (dP utils.DataProvider, err error) { byteData, err := io.ReadAll(req.Body) if err != nil { return nil, err } //convert the byteData into a xmlquery Node doc, err := xmlquery.Parse(strings.NewReader(string(byteData))) if err != nil { return nil, err } dP = &httpXmlDP{xmlDoc: doc, cache: utils.MapStorage{}, addr: req.RemoteAddr} return } // httpXmlDP implements utils.DataProvider, serving as xml data decoder // decoded data is only searched once and cached type httpXmlDP struct { cache utils.MapStorage xmlDoc *xmlquery.Node addr string } // String is part of utils.DataProvider interface // when called, it will display the already parsed values out of cache func (hU *httpXmlDP) String() string { return hU.xmlDoc.OutputXML(true) } // FieldAsInterface is part of utils.DataProvider interface func (hU *httpXmlDP) FieldAsInterface(fldPath []string) (data any, err error) { //if path is missing return here error because if it arrived in xmlquery library will panic if len(fldPath) == 0 { return nil, fmt.Errorf("Empty path") } if data, err = hU.cache.FieldAsInterface(fldPath); err == nil || err != utils.ErrNotFound { // item found in cache return } err = nil // cancel previous err var slctrStr string workPath := make([]string, len(fldPath)) copy(workPath, fldPath) for i := range workPath { if sIdx := strings.Index(workPath[i], "["); sIdx != -1 { slctrStr = workPath[i][sIdx:] if slctrStr[len(slctrStr)-1:] != "]" { return nil, fmt.Errorf("filter rule <%s> needs to end in ]", slctrStr) } workPath[i] = workPath[i][:sIdx] if slctrStr[1:2] != "@" { i, err := strconv.Atoi(slctrStr[1 : len(slctrStr)-1]) if err != nil { return nil, err } slctrStr = "[" + strconv.Itoa(i+1) + "]" } workPath[i] = workPath[i] + slctrStr } } //convert workPath to HierarchyPath hrPath := utils.HierarchyPath(workPath) var elmnt *xmlquery.Node elmnt, err = xmlquery.Query(hU.xmlDoc, hrPath.AsString("/", false)) if err != nil { return nil, err } if elmnt == nil { return } //add the content in data and cache it data = elmnt.InnerText() hU.cache.Set(fldPath, data) return } // FieldAsString is part of utils.DataProvider interface func (hU *httpXmlDP) FieldAsString(fldPath []string) (data string, err error) { var valIface any valIface, err = hU.FieldAsInterface(fldPath) if err != nil { return } return utils.IfaceAsString(valIface), nil } // httpAgentReplyEncoder will encode []*engine.NMElement // and write content to http writer type httpAgentReplyEncoder interface { Encode(*utils.OrderedNavigableMap) error } // newHAReplyEncoder constructs a httpAgentReqDecoder based on encoder type func newHAReplyEncoder(encType string, w http.ResponseWriter) (rE httpAgentReplyEncoder, err error) { switch encType { default: return nil, fmt.Errorf("unsupported encoder type <%s>", encType) case utils.MetaXml: return newHAXMLEncoder(w) case utils.MetaTextPlain: return newHATextPlainEncoder(w) } } func newHAXMLEncoder(w http.ResponseWriter) (xE httpAgentReplyEncoder, err error) { return &haXMLEncoder{w: w}, nil } type haXMLEncoder struct { w http.ResponseWriter } // Encode implements httpAgentReplyEncoder func (xE *haXMLEncoder) Encode(nM *utils.OrderedNavigableMap) (err error) { var xmlElmnts []*utils.XMLElement if xmlElmnts, err = utils.NMAsXMLElements(nM); err != nil { return } if len(xmlElmnts) == 0 { return } var xmlOut []byte if xmlOut, err = xml.MarshalIndent(xmlElmnts, "", " "); err != nil { return } if _, err = xE.w.Write([]byte(xml.Header)); err != nil { return } _, err = xE.w.Write(xmlOut) return } func newHATextPlainEncoder(w http.ResponseWriter) (xE httpAgentReplyEncoder, err error) { return &haTextPlainEncoder{w: w}, nil } type haTextPlainEncoder struct { w http.ResponseWriter } // Encode implements httpAgentReplyEncoder func (xE *haTextPlainEncoder) Encode(nM *utils.OrderedNavigableMap) (err error) { var str, nmPath string msgFields := make(map[string]string) // work around to NMap issue for el := nM.GetFirstElement(); el != nil; el = el.Next() { path := el.Value nmIt, _ := nM.Field(path) path = path[:len(path)-1] // remove the last index nmPath = strings.Join(path, utils.NestingSep) val := nmIt.String() msgFields[utils.ConcatenatedKey(nmPath, val)] = val } for key, val := range msgFields { str += fmt.Sprintf("%s=%s\n", strings.Split(key, utils.InInFieldSep)[0], val) } _, err = xE.w.Write([]byte(str)) return }