rest_sink.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. package sinks
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/emqx/kuiper/xstream/api"
  7. "io/ioutil"
  8. "net"
  9. "net/http"
  10. "net/url"
  11. "strings"
  12. "time"
  13. )
  14. type RestSink struct {
  15. method string
  16. url string
  17. headers map[string]string
  18. bodyType string
  19. timeout int64
  20. sendSingle bool
  21. client *http.Client
  22. }
  23. var methodsMap = map[string]bool{"GET": true, "HEAD": true, "POST": true, "PUT": true, "DELETE": true, "PATCH": true}
  24. var bodyTypeMap = map[string]string{"none": "", "text": "text/plain", "json": "application/json", "html": "text/html", "xml": "application/xml", "javascript": "application/javascript", "form": ""}
  25. func (ms *RestSink) Configure(ps map[string]interface{}) error {
  26. temp, ok := ps["method"]
  27. if ok {
  28. ms.method, ok = temp.(string)
  29. if !ok {
  30. return fmt.Errorf("rest sink property method %v is not a string", temp)
  31. }
  32. ms.method = strings.ToUpper(strings.Trim(ms.method, ""))
  33. } else {
  34. ms.method = "GET"
  35. }
  36. if _, ok = methodsMap[ms.method]; !ok {
  37. return fmt.Errorf("invalid property method: %s", ms.method)
  38. }
  39. switch ms.method {
  40. case "GET", "HEAD":
  41. ms.bodyType = "none"
  42. default:
  43. ms.bodyType = "json"
  44. }
  45. temp, ok = ps["url"]
  46. if !ok {
  47. return fmt.Errorf("rest sink is missing property url")
  48. }
  49. ms.url, ok = temp.(string)
  50. if !ok {
  51. return fmt.Errorf("rest sink property url %v is not a string", temp)
  52. }
  53. ms.url = strings.Trim(ms.url, "")
  54. temp, ok = ps["headers"]
  55. if ok {
  56. ms.headers, ok = temp.(map[string]string)
  57. if !ok {
  58. return fmt.Errorf("rest sink property headers %v is not a map[string][]string", temp)
  59. }
  60. }
  61. temp, ok = ps["bodyType"]
  62. if ok {
  63. ms.bodyType, ok = temp.(string)
  64. if !ok {
  65. return fmt.Errorf("rest sink property bodyType %v is not a string", temp)
  66. }
  67. ms.bodyType = strings.ToLower(strings.Trim(ms.bodyType, ""))
  68. }
  69. if _, ok = bodyTypeMap[ms.bodyType]; !ok {
  70. return fmt.Errorf("invalid property bodyType: %s, should be \"none\" or \"form\"", ms.bodyType)
  71. }
  72. temp, ok = ps["timeout"]
  73. if !ok {
  74. ms.timeout = 5000
  75. } else {
  76. to, ok := temp.(float64)
  77. if !ok {
  78. return fmt.Errorf("rest sink property timeout %v is not a number", temp)
  79. }
  80. ms.timeout = int64(to)
  81. }
  82. temp, ok = ps["sendSingle"]
  83. if !ok {
  84. ms.sendSingle = false
  85. } else {
  86. ms.sendSingle, ok = temp.(bool)
  87. if !ok {
  88. return fmt.Errorf("rest sink property sendSingle %v is not a bool", temp)
  89. }
  90. }
  91. return nil
  92. }
  93. func (ms *RestSink) Open(ctx api.StreamContext) error {
  94. logger := ctx.GetLogger()
  95. ms.client = &http.Client{Timeout: time.Duration(ms.timeout) * time.Millisecond}
  96. logger.Infof("open rest sink with configuration: {method: %s, url: %s, bodyType: %s, timeout: %d,header: %v, sendSingle: %v", ms.method, ms.url, ms.bodyType, ms.timeout, ms.headers, ms.sendSingle)
  97. timeout := 1 * time.Second
  98. if u, err := url.Parse(ms.url); err != nil {
  99. return err
  100. } else {
  101. _, err := net.DialTimeout("tcp", u.Host, timeout)
  102. if err != nil {
  103. logger.Errorf("Target web server unreachable: %s", err)
  104. return err
  105. } else {
  106. logger.Infof("Target web server is available.")
  107. }
  108. }
  109. return nil
  110. }
  111. type MultiErrors []error
  112. func (me MultiErrors) AddError(err error) MultiErrors {
  113. me = append(me, err)
  114. return me
  115. }
  116. func (me MultiErrors) Error() string {
  117. s := make([]string, len(me))
  118. for i, v := range me {
  119. s = append(s, fmt.Sprintf("Error %d with info %s. \n", i, v))
  120. }
  121. return strings.Join(s, " ")
  122. }
  123. func (ms *RestSink) Collect(ctx api.StreamContext, item interface{}) error {
  124. logger := ctx.GetLogger()
  125. v, ok := item.([]byte)
  126. if !ok {
  127. logger.Warnf("rest sink receive non []byte data: %v", item)
  128. }
  129. logger.Debugf("rest sink receive %s", item)
  130. return ms.send(v, logger)
  131. }
  132. func (ms *RestSink) send(v interface{}, logger api.Logger) error {
  133. var req *http.Request
  134. var err error
  135. switch ms.bodyType {
  136. case "none":
  137. req, err = http.NewRequest(ms.method, ms.url, nil)
  138. if err != nil {
  139. return fmt.Errorf("fail to create request: %v", err)
  140. }
  141. case "json", "text", "javascript", "html", "xml":
  142. var body = &(bytes.Buffer{})
  143. switch t := v.(type) {
  144. case []byte:
  145. body = bytes.NewBuffer(t)
  146. default:
  147. return fmt.Errorf("invalid content: %v", v)
  148. }
  149. req, err = http.NewRequest(ms.method, ms.url, body)
  150. if err != nil {
  151. return fmt.Errorf("fail to create request: %v", err)
  152. }
  153. req.Header.Set("Content-Type", bodyTypeMap[ms.bodyType])
  154. case "form":
  155. form := url.Values{}
  156. im, err := convertToMap(v, ms.sendSingle)
  157. if err != nil {
  158. return err
  159. }
  160. for key, value := range im {
  161. var vstr string
  162. switch value.(type) {
  163. case []interface{}, map[string]interface{}:
  164. if temp, err := json.Marshal(value); err != nil {
  165. return fmt.Errorf("fail to parse fomr value: %v", err)
  166. } else {
  167. vstr = string(temp)
  168. }
  169. default:
  170. vstr = fmt.Sprintf("%v", value)
  171. }
  172. form.Set(key, vstr)
  173. }
  174. body := ioutil.NopCloser(strings.NewReader(form.Encode()))
  175. req, err = http.NewRequest(ms.method, ms.url, body)
  176. if err != nil {
  177. return fmt.Errorf("fail to create request: %v", err)
  178. }
  179. req.Header.Set("Content-Type", "application/x-www-form-urlencoded;param=value")
  180. default:
  181. return fmt.Errorf("unsupported body type %s", ms.bodyType)
  182. }
  183. if len(ms.headers) > 0 {
  184. for k, v := range ms.headers {
  185. req.Header.Set(k, v)
  186. }
  187. }
  188. logger.Debugf("do request: %s %s with %s", ms.method, ms.url, req.Body)
  189. resp, err := ms.client.Do(req)
  190. if err != nil {
  191. return fmt.Errorf("rest sink fails to send out the data")
  192. } else {
  193. logger.Debugf("rest sink got response %v", resp)
  194. if resp.StatusCode < 200 || resp.StatusCode > 299 {
  195. return fmt.Errorf("rest sink fails to err http return code: %d.", resp.StatusCode)
  196. }
  197. }
  198. return nil
  199. }
  200. func convertToMap(v interface{}, sendSingle bool) (map[string]interface{}, error) {
  201. switch t := v.(type) {
  202. case []byte:
  203. r := make(map[string]interface{})
  204. if err := json.Unmarshal(t, &r); err != nil {
  205. if sendSingle {
  206. return nil, fmt.Errorf("fail to decode content: %v", err)
  207. } else {
  208. r["result"] = string(t)
  209. }
  210. }
  211. return r, nil
  212. default:
  213. return nil, fmt.Errorf("invalid content: %v", v)
  214. }
  215. return nil, fmt.Errorf("invalid content: %v", v)
  216. }
  217. func (ms *RestSink) Close(ctx api.StreamContext) error {
  218. logger := ctx.GetLogger()
  219. logger.Infof("Closing rest sink")
  220. return nil
  221. }