edgex_source.go 9.0 KB

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