rt_sink_test.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. /*
  2. * ECAL
  3. *
  4. * Copyright 2020 Matthias Ladkau. All rights reserved.
  5. *
  6. * This Source Code Form is subject to the terms of the MIT
  7. * License, If a copy of the MIT License was not distributed with this
  8. * file, You can obtain one at https://opensource.org/licenses/MIT.
  9. */
  10. package interpreter
  11. import (
  12. "fmt"
  13. "testing"
  14. "devt.de/krotik/ecal/scope"
  15. )
  16. func TestEventProcessing(t *testing.T) {
  17. vs := scope.NewScope(scope.GlobalScope)
  18. _, err := UnitTestEvalAndAST(
  19. `
  20. /*
  21. My cool rule
  22. */
  23. sink rule1
  24. kindmatch [ "core.*", "foo.*" ],
  25. scopematch [ "data.write" ],
  26. statematch { "val" : NULL },
  27. priority 10,
  28. suppresses [ "rule2" ]
  29. {
  30. log("rule1 < ", event)
  31. }
  32. `, vs,
  33. `
  34. sink #
  35. My cool rule
  36. identifier: rule1
  37. kindmatch
  38. list
  39. string: 'core.*'
  40. string: 'foo.*'
  41. scopematch
  42. list
  43. string: 'data.write'
  44. statematch
  45. map
  46. kvp
  47. string: 'val'
  48. null
  49. priority
  50. number: 10
  51. suppresses
  52. list
  53. string: 'rule2'
  54. statements
  55. identifier: log
  56. funccall
  57. string: 'rule1 < '
  58. identifier: event
  59. `[1:])
  60. if err != nil {
  61. t.Error(err)
  62. return
  63. }
  64. // Nothing defined in the global scope
  65. if vs.String() != `
  66. GlobalScope {
  67. }`[1:] {
  68. t.Error("Unexpected result: ", vs)
  69. return
  70. }
  71. if res := fmt.Sprint(testprocessor.Rules()["rule1"]); res !=
  72. `Rule:rule1 [My cool rule] (Priority:10 Kind:[core.* foo.*] Scope:[data.write] StateMatch:{"val":null} Suppress:[rule2])` {
  73. t.Error("Unexpected result:", res)
  74. return
  75. }
  76. // Test case 1 - Multiple rules, scope match, priorities and waiting for finish (no errors)
  77. _, err = UnitTestEval(
  78. `
  79. sink rule1
  80. kindmatch [ "web.page.index" ],
  81. scopematch [ "request.read" ],
  82. {
  83. log("rule1 - Handling request: ", event.kind)
  84. addEvent("Rule1Event1", "not_existing", event.state)
  85. addEvent("Rule1Event2", "web.log", event.state)
  86. }
  87. sink rule2
  88. kindmatch [ "web.page.*" ],
  89. priority 1, # Ensure this rule is always executed after rule1
  90. {
  91. log("rule2 - Tracking user:", event.state.user)
  92. if event.state.user == "bar" {
  93. sinkError("User bar was here", [123])
  94. }
  95. }
  96. sink rule3
  97. kindmatch [ "web.log" ],
  98. {
  99. log("rule3 - Logging user:", event.state.user)
  100. }
  101. res := addEventAndWait("request", "web.page.index", {
  102. "user" : "foo"
  103. }, {
  104. "request.read" : true
  105. })
  106. log("ErrorResult:", res, " ", len(res) == 0)
  107. res := addEventAndWait("request", "web.page.index", {
  108. "user" : "bar"
  109. }, {
  110. "request.read" : false
  111. })
  112. log("ErrorResult:", res, " ", res == null)
  113. `, vs)
  114. if err != nil {
  115. t.Error(err)
  116. return
  117. }
  118. if testlogger.String() != `
  119. rule1 - Handling request: web.page.index
  120. rule2 - Tracking user:foo
  121. rule3 - Logging user:foo
  122. ErrorResult:null true
  123. rule2 - Tracking user:bar
  124. ErrorResult:[
  125. {
  126. "errors": {
  127. "rule2": {
  128. "detail": [
  129. 123
  130. ],
  131. "message": "ECAL error in ECALTestRuntime: Error in sink (User bar was here) (Line:17 Pos:13)"
  132. }
  133. },
  134. "event": {
  135. "kind": "web.page.index",
  136. "name": "request",
  137. "state": {
  138. "user": "bar"
  139. }
  140. }
  141. }
  142. ] false`[1:] {
  143. t.Error("Unexpected result:", testlogger.String())
  144. return
  145. }
  146. // Test case 2 - unexpected error
  147. _, err = UnitTestEval(
  148. `
  149. sink rule1
  150. kindmatch [ "test" ],
  151. {
  152. log("rule1 - ", event.kind)
  153. noexitingfunctioncall()
  154. }
  155. err := addEventAndWait("someevent", "test", {})
  156. if err != null {
  157. error(err[0].errors)
  158. }
  159. `, vs)
  160. if err != nil {
  161. t.Error(err)
  162. return
  163. }
  164. if testlogger.String() != `
  165. rule1 - test
  166. error: {
  167. "rule1": {
  168. "detail": null,
  169. "message": "ECAL error in ECALTestRuntime: Unknown construct (Unknown function: noexitingfunctioncall) (Line:6 Pos:9)"
  170. }
  171. }`[1:] {
  172. t.Error("Unexpected result:", testlogger.String())
  173. return
  174. }
  175. // Test case 3 - rule suppression
  176. _, err = UnitTestEval(
  177. `
  178. sink rule1
  179. kindmatch [ "test.event" ],
  180. suppresses [ "rule3" ],
  181. {
  182. log("rule1 - Handling request: ", event.kind)
  183. }
  184. sink rule2
  185. kindmatch [ "test.*" ],
  186. priority 1, # Ensure this rule is always executed after rule1
  187. {
  188. log("rule2 - Handling request: ", event.kind)
  189. }
  190. sink rule3
  191. kindmatch [ "test.*" ],
  192. priority 1, # Ensure this rule is always executed after rule1
  193. {
  194. log("rule3 - Handling request: ", event.kind)
  195. }
  196. err := addEventAndWait("myevent", "test.event", {})
  197. if len(err) > 0 {
  198. error(err[0].errors)
  199. }
  200. `, vs)
  201. if err != nil {
  202. t.Error(err)
  203. return
  204. }
  205. if testlogger.String() != `
  206. rule1 - Handling request: test.event
  207. rule2 - Handling request: test.event`[1:] {
  208. t.Error("Unexpected result:", testlogger.String())
  209. return
  210. }
  211. // Test case 4 - state match
  212. _, err = UnitTestEval(
  213. `
  214. sink rule1
  215. kindmatch [ "test.event", "foo.*" ],
  216. statematch { "a" : null },
  217. {
  218. log("rule1 - Handling request: ", event.kind)
  219. }
  220. sink rule2
  221. kindmatch [ "test.*" ],
  222. priority 1,
  223. statematch { "b" : 1 },
  224. {
  225. log("rule2 - Handling request: ", event.kind)
  226. }
  227. sink rule3
  228. kindmatch [ "test.*" ],
  229. priority 2,
  230. statematch { "c" : 2 },
  231. {
  232. log("rule3 - Handling request: ", event.kind)
  233. }
  234. err := addEventAndWait("myevent", "test.event", {
  235. "a" : "foo",
  236. "b" : 1,
  237. })
  238. if len(err) > 0 {
  239. error(err[0].errors)
  240. }
  241. `, vs)
  242. if err != nil {
  243. t.Error(err)
  244. return
  245. }
  246. if testlogger.String() != `
  247. rule1 - Handling request: test.event
  248. rule2 - Handling request: test.event`[1:] {
  249. t.Error("Unexpected result:", testlogger.String())
  250. return
  251. }
  252. }