edgex_source.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. // +build edgex
  2. package extensions
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/base64"
  7. "encoding/binary"
  8. "encoding/json"
  9. "fmt"
  10. "github.com/edgexfoundry/go-mod-core-contracts/clients"
  11. "github.com/edgexfoundry/go-mod-core-contracts/clients/coredata"
  12. "github.com/edgexfoundry/go-mod-core-contracts/clients/urlclient/local"
  13. "github.com/edgexfoundry/go-mod-core-contracts/models"
  14. "github.com/edgexfoundry/go-mod-messaging/messaging"
  15. "github.com/edgexfoundry/go-mod-messaging/pkg/types"
  16. "github.com/emqx/kuiper/common"
  17. "github.com/emqx/kuiper/xstream/api"
  18. "strconv"
  19. "strings"
  20. )
  21. type EdgexSource struct {
  22. client messaging.MessageClient
  23. subscribed bool
  24. vdc coredata.ValueDescriptorClient
  25. topic string
  26. valueDescs map[string]string
  27. }
  28. func (es *EdgexSource) Configure(device string, props map[string]interface{}) error {
  29. var protocol = "tcp"
  30. if p, ok := props["protocol"]; ok {
  31. protocol = p.(string)
  32. }
  33. var server = "localhost"
  34. if s, ok := props["server"]; ok {
  35. server = s.(string)
  36. }
  37. var port = 5563
  38. if p, ok := props["port"]; ok {
  39. port = p.(int)
  40. }
  41. if tpc, ok := props["topic"]; ok {
  42. es.topic = tpc.(string)
  43. }
  44. var mbusType = messaging.ZeroMQ
  45. if t, ok := props["type"]; ok {
  46. mbusType = t.(string)
  47. if mbusType != messaging.ZeroMQ && mbusType != messaging.MQTT {
  48. return fmt.Errorf("Specified wrong message type value %s, will use zeromq messagebus.\n", mbusType)
  49. }
  50. }
  51. if serviceServer, ok := props["serviceServer"]; ok {
  52. svr := serviceServer.(string) + clients.ApiValueDescriptorRoute
  53. common.Log.Infof("Connect to value descriptor service at: %s \n", svr)
  54. es.vdc = coredata.NewValueDescriptorClient(local.New(svr))
  55. es.valueDescs = make(map[string]string)
  56. } else {
  57. return fmt.Errorf("The service server cannot be empty.")
  58. }
  59. mbconf := types.MessageBusConfig{SubscribeHost: types.HostInfo{Protocol: protocol, Host: server, Port: port}, Type: mbusType}
  60. var optional = make(map[string]string)
  61. if ops, ok := props["optional"]; ok {
  62. if ops1, ok1 := ops.(map[string]interface{}); ok1 {
  63. for k, v := range ops1 {
  64. if cv, ok := CastToString(v); ok {
  65. optional[k] = cv
  66. } else {
  67. common.Log.Infof("Cannot convert configuration %s: %s to string type.\n", k, v)
  68. }
  69. }
  70. }
  71. mbconf.Optional = optional
  72. }
  73. common.Log.Infof("Use configuration for edgex messagebus %v\n", mbconf)
  74. if client, err := messaging.NewMessageClient(mbconf); err != nil {
  75. return err
  76. } else {
  77. es.client = client
  78. return nil
  79. }
  80. }
  81. func castToString(v interface{}) (result string, ok bool) {
  82. switch v := v.(type) {
  83. case int:
  84. return strconv.Itoa(v), true
  85. case string:
  86. return v, true
  87. case bool:
  88. return strconv.FormatBool(v), true
  89. case float64, float32:
  90. return fmt.Sprintf("%.2f", v), true
  91. default:
  92. return "", false
  93. }
  94. }
  95. func (es *EdgexSource) Open(ctx api.StreamContext, consumer chan<- api.SourceTuple, errCh chan<- error) {
  96. log := ctx.GetLogger()
  97. if err := es.client.Connect(); err != nil {
  98. info := fmt.Errorf("Failed to connect to edgex message bus: " + err.Error())
  99. log.Errorf(info.Error())
  100. errCh <- info
  101. return
  102. }
  103. log.Infof("The connection to edgex messagebus is established successfully.")
  104. messages := make(chan types.MessageEnvelope)
  105. topics := []types.TopicChannel{{Topic: es.topic, Messages: messages}}
  106. err := make(chan error)
  107. if e := es.client.Subscribe(topics, err); e != nil {
  108. log.Errorf("Failed to subscribe to edgex messagebus topic %s.\n", e)
  109. errCh <- e
  110. } else {
  111. es.subscribed = true
  112. log.Infof("Successfully subscribed to edgex messagebus topic %s.", es.topic)
  113. for {
  114. select {
  115. case e1 := <-err:
  116. errCh <- e1
  117. return
  118. case env := <-messages:
  119. if strings.ToLower(env.ContentType) == "application/json" {
  120. e := models.Event{}
  121. if err := e.UnmarshalJSON(env.Payload); err != nil {
  122. len := len(env.Payload)
  123. if len > 200 {
  124. len = 200
  125. }
  126. log.Warnf("payload %s unmarshal fail: %v", env.Payload[0:(len-1)], err)
  127. } else {
  128. result := make(map[string]interface{})
  129. meta := make(map[string]interface{})
  130. log.Debugf("receive message %s from device %s", env.Payload, e.Device)
  131. for _, r := range e.Readings {
  132. if r.Name != "" {
  133. if v, err := es.getValue(r, log); err != nil {
  134. log.Warnf("fail to get value for %s: %v", r.Name, err)
  135. } else {
  136. result[r.Name] = v
  137. }
  138. r_meta := map[string]interface{}{}
  139. r_meta["id"] = r.Id
  140. r_meta["created"] = r.Created
  141. r_meta["modified"] = r.Modified
  142. r_meta["origin"] = r.Origin
  143. r_meta["pushed"] = r.Pushed
  144. r_meta["device"] = r.Device
  145. meta[r.Name] = r_meta
  146. } else {
  147. log.Warnf("The name of readings should not be empty!")
  148. }
  149. }
  150. if len(result) > 0 {
  151. meta["id"] = e.ID
  152. meta["pushed"] = e.Pushed
  153. meta["device"] = e.Device
  154. meta["created"] = e.Created
  155. meta["modified"] = e.Modified
  156. meta["origin"] = e.Origin
  157. meta["correlationid"] = env.CorrelationID
  158. select {
  159. case consumer <- api.NewDefaultSourceTuple(result, meta):
  160. log.Debugf("send data to device node")
  161. case <-ctx.Done():
  162. return
  163. }
  164. } else {
  165. log.Warnf("No readings are processed for the event, so ignore it.")
  166. }
  167. }
  168. } else {
  169. log.Errorf("Unsupported data type %s.", env.ContentType)
  170. }
  171. }
  172. }
  173. }
  174. }
  175. func (es *EdgexSource) getValue(r models.Reading, logger api.Logger) (interface{}, error) {
  176. t, err := es.getType(r.Name, logger)
  177. var ot = t
  178. if err != nil {
  179. return nil, err
  180. }
  181. t = strings.ToUpper(t)
  182. logger.Debugf("name %s with type %s", r.Name, t)
  183. v := r.Value
  184. switch t {
  185. case "BOOL":
  186. if r, err := strconv.ParseBool(v); err != nil {
  187. return nil, err
  188. } else {
  189. return r, nil
  190. }
  191. case "INT8", "INT16", "INT32", "INT64", "UINT8", "UINT16", "UINT32":
  192. if r, err := strconv.Atoi(v); err != nil {
  193. return nil, err
  194. } else {
  195. return r, nil
  196. }
  197. case "UINT64":
  198. if u64, err := strconv.ParseUint(v, 10, 64); err != nil {
  199. return nil, err
  200. } else {
  201. return u64, nil
  202. }
  203. case "FLOAT32", "FLOAT64":
  204. if r.ValueType == "" {
  205. r.ValueType = ot
  206. }
  207. return es.getFloatValue(r.FloatEncoding, r.Value, r.ValueType, logger)
  208. case "STRING":
  209. return v, nil
  210. case "BOOLARRAY":
  211. var val []bool
  212. if e := json.Unmarshal([]byte(v), &val); e == nil {
  213. return val, nil
  214. } else {
  215. return nil, e
  216. }
  217. case "UINT8ARRAY", "UINT16ARRAY", "UINT32ARRAY", "INT8ARRAY", "INT16ARRAY", "INT32ARRAY", "INT64ARRAY":
  218. var val []int
  219. if e := json.Unmarshal([]byte(v), &val); e == nil {
  220. return val, nil
  221. } else {
  222. return nil, e
  223. }
  224. case "UINT64ARRAY":
  225. var val []uint64
  226. if e := json.Unmarshal([]byte(v), &val); e == nil {
  227. return val, nil
  228. } else {
  229. return nil, e
  230. }
  231. case "FLOAT32ARRAY", "FLOAT64ARRAY":
  232. if r.ValueType == "" {
  233. r.ValueType = ot
  234. }
  235. var val1 []string
  236. if e := json.Unmarshal([]byte(v), &val1); e == nil {
  237. ret := []float64{}
  238. for _, v := range val1 {
  239. if fv, err := es.getFloatValue(r.FloatEncoding, v, r.ValueType, logger); err != nil {
  240. return nil, err
  241. } else {
  242. if f, ok := fv.(float64); ok {
  243. ret = append(ret, f)
  244. } else {
  245. return nil, fmt.Errorf("The %v is not a float64 type.", f)
  246. }
  247. }
  248. }
  249. return ret, nil
  250. } else {
  251. var val []float64
  252. ret := []float64{}
  253. if e := json.Unmarshal([]byte(v), &val); e == nil {
  254. for _, v := range val {
  255. ret = append(ret, v)
  256. }
  257. return ret, nil
  258. } else {
  259. return nil, e
  260. }
  261. }
  262. case "BINARY":
  263. return nil, fmt.Errorf("Unsupport for binary type, the value will be ignored.")
  264. default:
  265. logger.Warnf("Not supported type %s, and processed as string value", t)
  266. return v, nil
  267. }
  268. }
  269. func (es *EdgexSource) getFloatValue(FloatEncoding string, Value string, ValueType string, logger api.Logger) (interface{}, error) {
  270. if len(FloatEncoding) == 0 {
  271. if strings.Contains(Value, "=") {
  272. FloatEncoding = models.Base64Encoding
  273. } else {
  274. FloatEncoding = models.ENotation
  275. }
  276. }
  277. switch strings.ToLower(ValueType) {
  278. case strings.ToLower(models.ValueTypeFloat32), strings.ToLower(models.ValueTypeFloat32Array):
  279. var value float64
  280. switch FloatEncoding {
  281. case models.Base64Encoding:
  282. data, err := base64.StdEncoding.DecodeString(Value)
  283. if err != nil {
  284. return false, fmt.Errorf("unable to Base 64 decode float32 value ('%s'): %s", Value, err.Error())
  285. }
  286. var value1 float32
  287. err = binary.Read(bytes.NewReader(data), binary.BigEndian, &value1)
  288. if err != nil {
  289. return false, fmt.Errorf("unable to decode float32 value bytes: %s", err.Error())
  290. }
  291. value = float64(value1)
  292. case models.ENotation:
  293. var err error
  294. var temp float64
  295. temp, err = strconv.ParseFloat(Value, 64)
  296. if err != nil {
  297. return false, fmt.Errorf("unable to parse Float64 eNotation value: %s", err.Error())
  298. }
  299. value = float64(temp)
  300. default:
  301. return false, fmt.Errorf("unkown FloatEncoding for float32 value: %s", FloatEncoding)
  302. }
  303. return value, nil
  304. case strings.ToLower(models.ValueTypeFloat64), strings.ToLower(models.ValueTypeFloat64Array):
  305. var value float64
  306. switch FloatEncoding {
  307. case models.Base64Encoding:
  308. data, err := base64.StdEncoding.DecodeString(Value)
  309. if err != nil {
  310. return false, fmt.Errorf("unable to Base 64 decode float64 value ('%s'): %s", Value, err.Error())
  311. }
  312. err = binary.Read(bytes.NewReader(data), binary.BigEndian, &value)
  313. if err != nil {
  314. return false, fmt.Errorf("unable to decode float64 value bytes: %s", err.Error())
  315. }
  316. return value, nil
  317. case models.ENotation:
  318. var err error
  319. value, err = strconv.ParseFloat(Value, 64)
  320. if err != nil {
  321. return false, fmt.Errorf("unable to parse Float64 eNotation value: %s", err.Error())
  322. }
  323. return value, nil
  324. default:
  325. return false, fmt.Errorf("unkown FloatEncoding for float64 value: %s", FloatEncoding)
  326. }
  327. default:
  328. return nil, fmt.Errorf("unkown value type: %s", ValueType)
  329. }
  330. }
  331. func (es *EdgexSource) fetchAllDataDescriptors() error {
  332. if vdArr, err := es.vdc.ValueDescriptors(context.Background()); err != nil {
  333. return err
  334. } else {
  335. for _, vd := range vdArr {
  336. es.valueDescs[vd.Name] = vd.Type
  337. }
  338. if len(vdArr) == 0 {
  339. common.Log.Infof("Cannot find any value descriptors from value descriptor services.")
  340. } else {
  341. common.Log.Infof("Get %d of value descriptors from service.", len(vdArr))
  342. for i, v := range vdArr {
  343. common.Log.Debugf("%d: %s - %s ", i, v.Name, v.Type)
  344. }
  345. }
  346. }
  347. return nil
  348. }
  349. func (es *EdgexSource) getType(id string, logger api.Logger) (string, error) {
  350. if t, ok := es.valueDescs[id]; ok {
  351. return t, nil
  352. } else {
  353. if e := es.fetchAllDataDescriptors(); e != nil {
  354. return "", e
  355. }
  356. if t, ok := es.valueDescs[id]; ok {
  357. return t, nil
  358. } else {
  359. return "", fmt.Errorf("cannot find type info for %s in value descriptor.", id)
  360. }
  361. }
  362. }
  363. func (es *EdgexSource) Close(ctx api.StreamContext) error {
  364. if es.subscribed {
  365. if e := es.client.Disconnect(); e != nil {
  366. return e
  367. }
  368. }
  369. return nil
  370. }