project_test.go 38 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561
  1. package plans
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "github.com/emqx/kuiper/common"
  7. "github.com/emqx/kuiper/xsql"
  8. "github.com/emqx/kuiper/xstream/contexts"
  9. "reflect"
  10. "strings"
  11. "testing"
  12. )
  13. func TestProjectPlan_Apply1(t *testing.T) {
  14. var tests = []struct {
  15. sql string
  16. data *xsql.Tuple
  17. result []map[string]interface{}
  18. }{
  19. {
  20. sql: "SELECT a FROM test",
  21. data: &xsql.Tuple{
  22. Emitter: "test",
  23. Message: xsql.Message{
  24. "a": "val_a",
  25. },
  26. },
  27. result: []map[string]interface{}{{
  28. "a": "val_a",
  29. }},
  30. },
  31. {
  32. sql: "SELECT b FROM test",
  33. data: &xsql.Tuple{
  34. Emitter: "test",
  35. Message: xsql.Message{
  36. "a": "val_a",
  37. },
  38. },
  39. result: []map[string]interface{}{{}},
  40. },
  41. {
  42. sql: "SELECT ts FROM test",
  43. data: &xsql.Tuple{
  44. Emitter: "test",
  45. Message: xsql.Message{
  46. "a": "val_a",
  47. "ts": common.TimeFromUnixMilli(1568854573431),
  48. },
  49. },
  50. result: []map[string]interface{}{{
  51. "ts": "2019-09-19T00:56:13.431Z",
  52. }},
  53. },
  54. //Schemaless may return a message without selecting column
  55. {
  56. sql: "SELECT ts FROM test",
  57. data: &xsql.Tuple{
  58. Emitter: "test",
  59. Message: xsql.Message{
  60. "a": "val_a",
  61. "ts2": common.TimeFromUnixMilli(1568854573431),
  62. },
  63. },
  64. result: []map[string]interface{}{{}},
  65. },
  66. {
  67. sql: "SELECT A FROM test",
  68. data: &xsql.Tuple{
  69. Emitter: "test",
  70. Message: xsql.Message{
  71. "a": "val_a",
  72. },
  73. },
  74. result: []map[string]interface{}{{
  75. "A": "val_a",
  76. }},
  77. },
  78. {
  79. sql: `SELECT "value" FROM test`,
  80. data: &xsql.Tuple{
  81. Emitter: "test",
  82. Message: xsql.Message{},
  83. },
  84. result: []map[string]interface{}{{
  85. DEFAULT_FIELD_NAME_PREFIX + "0": "value",
  86. }},
  87. },
  88. {
  89. sql: `SELECT 3.4 FROM test`,
  90. data: &xsql.Tuple{
  91. Emitter: "test",
  92. Message: xsql.Message{},
  93. },
  94. result: []map[string]interface{}{{
  95. DEFAULT_FIELD_NAME_PREFIX + "0": 3.4,
  96. }},
  97. },
  98. {
  99. sql: `SELECT 5 FROM test`,
  100. data: &xsql.Tuple{
  101. Emitter: "test",
  102. Message: xsql.Message{},
  103. },
  104. result: []map[string]interface{}{{
  105. DEFAULT_FIELD_NAME_PREFIX + "0": 5.0,
  106. }},
  107. },
  108. {
  109. sql: `SELECT a, "value" AS b FROM test`,
  110. data: &xsql.Tuple{
  111. Emitter: "test",
  112. Message: xsql.Message{
  113. "a": "val_a",
  114. },
  115. },
  116. result: []map[string]interface{}{{
  117. "a": "val_a",
  118. "b": "value",
  119. }},
  120. },
  121. {
  122. sql: `SELECT a, "value" AS b, 3.14 as Pi, 0 as Zero FROM test`,
  123. data: &xsql.Tuple{
  124. Emitter: "test",
  125. Message: xsql.Message{
  126. "a": "val_a",
  127. },
  128. },
  129. result: []map[string]interface{}{{
  130. "a": "val_a",
  131. "b": "value",
  132. "Pi": 3.14,
  133. "Zero": 0.0,
  134. }},
  135. },
  136. {
  137. sql: `SELECT a->b AS ab FROM test`,
  138. data: &xsql.Tuple{
  139. Emitter: "test",
  140. Message: xsql.Message{
  141. "a": map[string]interface{}{"b": "hello"},
  142. },
  143. },
  144. result: []map[string]interface{}{{
  145. "ab": "hello",
  146. }},
  147. },
  148. {
  149. sql: `SELECT a->b AS ab FROM test`,
  150. data: &xsql.Tuple{
  151. Emitter: "test",
  152. Message: xsql.Message{
  153. "name": "name",
  154. },
  155. },
  156. result: []map[string]interface{}{{}},
  157. },
  158. {
  159. sql: `SELECT a->b AS ab FROM test`,
  160. data: &xsql.Tuple{
  161. Emitter: "test",
  162. Message: xsql.Message{
  163. "a": "commonstring",
  164. },
  165. },
  166. result: []map[string]interface{}{{}},
  167. },
  168. {
  169. sql: `SELECT a[0]->b AS ab FROM test`,
  170. data: &xsql.Tuple{
  171. Emitter: "test",
  172. Message: xsql.Message{
  173. "a": []interface{}{
  174. map[string]interface{}{"b": "hello1"},
  175. map[string]interface{}{"b": "hello2"},
  176. },
  177. },
  178. },
  179. result: []map[string]interface{}{{
  180. "ab": "hello1",
  181. }},
  182. },
  183. {
  184. sql: `SELECT a[0]->b AS ab FROM test`,
  185. data: &xsql.Tuple{
  186. Emitter: "test",
  187. Message: xsql.Message{
  188. "a": []map[string]interface{}{
  189. {"b": "hello1"},
  190. {"b": "hello2"},
  191. },
  192. },
  193. },
  194. result: []map[string]interface{}{{
  195. "ab": "hello1",
  196. }},
  197. },
  198. {
  199. sql: `SELECT a[2:4] AS ab FROM test`,
  200. data: &xsql.Tuple{
  201. Emitter: "test",
  202. Message: xsql.Message{
  203. "a": []map[string]interface{}{
  204. {"b": "hello1"},
  205. {"b": "hello2"},
  206. {"b": "hello3"},
  207. {"b": "hello4"},
  208. {"b": "hello5"},
  209. },
  210. },
  211. },
  212. result: []map[string]interface{}{{
  213. "ab": []interface{}{
  214. map[string]interface{}{"b": "hello3"},
  215. map[string]interface{}{"b": "hello4"},
  216. },
  217. }},
  218. },
  219. {
  220. sql: `SELECT a->c->d AS f1 FROM test`,
  221. data: &xsql.Tuple{
  222. Emitter: "test",
  223. Message: xsql.Message{
  224. "a": map[string]interface{}{
  225. "b": "hello",
  226. "c": map[string]interface{}{
  227. "d": 35.2,
  228. },
  229. },
  230. },
  231. },
  232. result: []map[string]interface{}{{
  233. "f1": 35.2,
  234. }},
  235. },
  236. {
  237. sql: `SELECT a->c->d AS f1 FROM test`,
  238. data: &xsql.Tuple{
  239. Emitter: "test",
  240. Message: xsql.Message{
  241. "a": map[string]interface{}{
  242. "b": "hello",
  243. "c": map[string]interface{}{
  244. "e": 35.2,
  245. },
  246. },
  247. },
  248. },
  249. result: []map[string]interface{}{{}},
  250. },
  251. {
  252. sql: `SELECT a->c->d AS f1 FROM test`,
  253. data: &xsql.Tuple{
  254. Emitter: "test",
  255. Message: xsql.Message{
  256. "a": map[string]interface{}{
  257. "b": "hello",
  258. },
  259. },
  260. },
  261. result: []map[string]interface{}{{}},
  262. },
  263. //The int type is not supported yet, the json parser returns float64 for int values
  264. {
  265. sql: `SELECT a->c->d AS f1 FROM test`,
  266. data: &xsql.Tuple{
  267. Emitter: "test",
  268. Message: xsql.Message{
  269. "a": map[string]interface{}{
  270. "b": "hello",
  271. "c": map[string]interface{}{
  272. "d": float64(35),
  273. },
  274. },
  275. },
  276. },
  277. result: []map[string]interface{}{{
  278. "f1": float64(35),
  279. }},
  280. },
  281. {
  282. sql: "SELECT a FROM test",
  283. data: &xsql.Tuple{
  284. Emitter: "test",
  285. Message: xsql.Message{},
  286. },
  287. result: []map[string]interface{}{
  288. {},
  289. },
  290. },
  291. {
  292. sql: "SELECT * FROM test",
  293. data: &xsql.Tuple{
  294. Emitter: "test",
  295. Message: xsql.Message{},
  296. },
  297. result: []map[string]interface{}{
  298. {},
  299. },
  300. },
  301. {
  302. sql: `SELECT * FROM test`,
  303. data: &xsql.Tuple{
  304. Emitter: "test",
  305. Message: xsql.Message{
  306. "a": map[string]interface{}{
  307. "b": "hello",
  308. "c": map[string]interface{}{
  309. "d": 35.2,
  310. },
  311. },
  312. },
  313. },
  314. result: []map[string]interface{}{{
  315. "a": map[string]interface{}{
  316. "b": "hello",
  317. "c": map[string]interface{}{
  318. "d": 35.2,
  319. },
  320. },
  321. }},
  322. },
  323. {
  324. sql: `SELECT * FROM test`,
  325. data: &xsql.Tuple{
  326. Emitter: "test",
  327. Message: xsql.Message{
  328. "a": "val1",
  329. "b": 3.14,
  330. },
  331. },
  332. result: []map[string]interface{}{{
  333. "a": "val1",
  334. "b": 3.14,
  335. }},
  336. },
  337. {
  338. sql: `SELECT 3*4 AS f1 FROM test`,
  339. data: &xsql.Tuple{
  340. Emitter: "test",
  341. Message: xsql.Message{},
  342. },
  343. result: []map[string]interface{}{{
  344. "f1": float64(12),
  345. }},
  346. },
  347. {
  348. sql: `SELECT 4.5*2 AS f1 FROM test`,
  349. data: &xsql.Tuple{
  350. Emitter: "test",
  351. Message: xsql.Message{},
  352. },
  353. result: []map[string]interface{}{{
  354. "f1": float64(9),
  355. }},
  356. },
  357. }
  358. fmt.Printf("The test bucket size is %d.\n\n", len(tests))
  359. contextLogger := common.Log.WithField("rule", "TestProjectPlan_Apply1")
  360. ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
  361. for i, tt := range tests {
  362. stmt, _ := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
  363. pp := &ProjectPlan{Fields: stmt.Fields}
  364. pp.isTest = true
  365. result := pp.Apply(ctx, tt.data)
  366. var mapRes []map[string]interface{}
  367. if v, ok := result.([]byte); ok {
  368. err := json.Unmarshal(v, &mapRes)
  369. if err != nil {
  370. t.Errorf("Failed to parse the input into map.\n")
  371. continue
  372. }
  373. //fmt.Printf("%t\n", mapRes["rengine_field_0"])
  374. if !reflect.DeepEqual(tt.result, mapRes) {
  375. t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
  376. }
  377. } else {
  378. t.Errorf("%d. The returned result is not type of []byte\n", i)
  379. }
  380. }
  381. }
  382. func TestProjectPlan_MultiInput(t *testing.T) {
  383. var tests = []struct {
  384. sql string
  385. data interface{}
  386. result []map[string]interface{}
  387. }{
  388. {
  389. sql: "SELECT * FROM tbl WHERE abc*2+3 > 12 AND abc < 20",
  390. data: &xsql.Tuple{
  391. Emitter: "tbl",
  392. Message: xsql.Message{
  393. "abc": int64(6),
  394. },
  395. },
  396. result: []map[string]interface{}{{
  397. "abc": float64(6), //json marshall problem
  398. }},
  399. },
  400. {
  401. sql: "SELECT abc FROM tbl WHERE abc*2+3 > 12 OR def = \"hello\"",
  402. data: &xsql.Tuple{
  403. Emitter: "tbl",
  404. Message: xsql.Message{
  405. "abc": int64(34),
  406. "def": "hello",
  407. },
  408. },
  409. result: []map[string]interface{}{{
  410. "abc": float64(34),
  411. }},
  412. },
  413. {
  414. sql: "SELECT id1 FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  415. data: xsql.WindowTuplesSet{
  416. xsql.WindowTuples{
  417. Emitter: "src1",
  418. Tuples: []xsql.Tuple{
  419. {
  420. Emitter: "src1",
  421. Message: xsql.Message{"id1": 1, "f1": "v1"},
  422. }, {
  423. Emitter: "src1",
  424. Message: xsql.Message{"id1": 2, "f1": "v2"},
  425. }, {
  426. Emitter: "src1",
  427. Message: xsql.Message{"id1": 3, "f1": "v1"},
  428. },
  429. },
  430. },
  431. },
  432. result: []map[string]interface{}{{
  433. "id1": float64(1),
  434. }, {
  435. "id1": float64(2),
  436. }, {
  437. "id1": float64(3),
  438. }},
  439. },
  440. {
  441. sql: "SELECT id1 FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  442. data: xsql.WindowTuplesSet{
  443. xsql.WindowTuples{
  444. Emitter: "src1",
  445. Tuples: []xsql.Tuple{
  446. {
  447. Emitter: "src1",
  448. Message: xsql.Message{"id1": 1, "f1": "v1"},
  449. }, {
  450. Emitter: "src1",
  451. Message: xsql.Message{"id2": 2, "f1": "v2"},
  452. }, {
  453. Emitter: "src1",
  454. Message: xsql.Message{"id1": 3, "f1": "v1"},
  455. },
  456. },
  457. },
  458. },
  459. result: []map[string]interface{}{{
  460. "id1": float64(1),
  461. }, {}, {
  462. "id1": float64(3),
  463. }},
  464. },
  465. {
  466. sql: "SELECT * FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  467. data: xsql.WindowTuplesSet{
  468. xsql.WindowTuples{
  469. Emitter: "src1",
  470. Tuples: []xsql.Tuple{
  471. {
  472. Emitter: "src1",
  473. Message: xsql.Message{"id1": 1, "f1": "v1"},
  474. }, {
  475. Emitter: "src1",
  476. Message: xsql.Message{"id1": 2, "f1": "v2"},
  477. }, {
  478. Emitter: "src1",
  479. Message: xsql.Message{"id1": 3, "f1": "v1"},
  480. },
  481. },
  482. },
  483. },
  484. result: []map[string]interface{}{{
  485. "id1": float64(1),
  486. "f1": "v1",
  487. }, {
  488. "id1": float64(2),
  489. "f1": "v2",
  490. }, {
  491. "id1": float64(3),
  492. "f1": "v1",
  493. }},
  494. },
  495. {
  496. sql: "SELECT * FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  497. data: xsql.WindowTuplesSet{
  498. xsql.WindowTuples{
  499. Emitter: "src1",
  500. Tuples: []xsql.Tuple{
  501. {
  502. Emitter: "src1",
  503. Message: xsql.Message{"id1": 1, "f1": "v1"},
  504. }, {
  505. Emitter: "src1",
  506. Message: xsql.Message{"id2": 2, "f2": "v2"},
  507. }, {
  508. Emitter: "src1",
  509. Message: xsql.Message{"id1": 3, "f1": "v1"},
  510. },
  511. },
  512. },
  513. },
  514. result: []map[string]interface{}{{
  515. "id1": float64(1),
  516. "f1": "v1",
  517. }, {
  518. "id2": float64(2),
  519. "f2": "v2",
  520. }, {
  521. "id1": float64(3),
  522. "f1": "v1",
  523. }},
  524. },
  525. {
  526. sql: "SELECT src1.* FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  527. data: xsql.WindowTuplesSet{
  528. xsql.WindowTuples{
  529. Emitter: "src1",
  530. Tuples: []xsql.Tuple{
  531. {
  532. Emitter: "src1",
  533. Message: xsql.Message{"id1": 1, "f1": "v1"},
  534. }, {
  535. Emitter: "src1",
  536. Message: xsql.Message{"id1": 2, "f1": "v2"},
  537. }, {
  538. Emitter: "src1",
  539. Message: xsql.Message{"id1": 3, "f1": "v1"},
  540. },
  541. },
  542. },
  543. },
  544. result: []map[string]interface{}{{
  545. "id1": float64(1),
  546. "f1": "v1",
  547. }, {
  548. "id1": float64(2),
  549. "f1": "v2",
  550. }, {
  551. "id1": float64(3),
  552. "f1": "v1",
  553. }},
  554. },
  555. {
  556. sql: "SELECT id1 FROM src1 left join src2 on src1.id1 = src2.id2 WHERE src1.f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  557. data: xsql.JoinTupleSets{
  558. xsql.JoinTuple{
  559. Tuples: []xsql.Tuple{
  560. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  561. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  562. },
  563. },
  564. xsql.JoinTuple{
  565. Tuples: []xsql.Tuple{
  566. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  567. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  568. },
  569. },
  570. xsql.JoinTuple{
  571. Tuples: []xsql.Tuple{
  572. {Emitter: "src1", Message: xsql.Message{"id1": 3, "f1": "v1"}},
  573. },
  574. },
  575. },
  576. result: []map[string]interface{}{{
  577. "id1": float64(1),
  578. }, {
  579. "id1": float64(2),
  580. }, {
  581. "id1": float64(3),
  582. }},
  583. },
  584. {
  585. sql: "SELECT id1 FROM src1 left join src2 on src1.id1 = src2.id2 WHERE src1.f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
  586. data: xsql.JoinTupleSets{
  587. xsql.JoinTuple{
  588. Tuples: []xsql.Tuple{
  589. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  590. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  591. },
  592. },
  593. xsql.JoinTuple{
  594. Tuples: []xsql.Tuple{
  595. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  596. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  597. },
  598. },
  599. xsql.JoinTuple{
  600. Tuples: []xsql.Tuple{
  601. {Emitter: "src1", Message: xsql.Message{"id2": 3, "f1": "v1"}},
  602. },
  603. },
  604. },
  605. result: []map[string]interface{}{{
  606. "id1": float64(1),
  607. }, {
  608. "id1": float64(2),
  609. }, {}},
  610. },
  611. {
  612. sql: "SELECT abc FROM tbl group by abc",
  613. data: xsql.GroupedTuplesSet{
  614. {
  615. &xsql.Tuple{
  616. Emitter: "tbl",
  617. Message: xsql.Message{
  618. "abc": int64(6),
  619. "def": "hello",
  620. },
  621. },
  622. },
  623. },
  624. result: []map[string]interface{}{{
  625. "abc": float64(6),
  626. }},
  627. },
  628. {
  629. sql: "SELECT abc FROM tbl group by abc",
  630. data: xsql.GroupedTuplesSet{
  631. {
  632. &xsql.Tuple{
  633. Emitter: "tbl",
  634. Message: xsql.Message{
  635. "def": "hello",
  636. },
  637. },
  638. },
  639. },
  640. result: []map[string]interface{}{{}},
  641. },
  642. {
  643. sql: "SELECT id1 FROM src1 GROUP BY TUMBLINGWINDOW(ss, 10), f1",
  644. data: xsql.GroupedTuplesSet{
  645. {
  646. &xsql.Tuple{
  647. Emitter: "src1",
  648. Message: xsql.Message{"id1": 1, "f1": "v1"},
  649. },
  650. &xsql.Tuple{
  651. Emitter: "src1",
  652. Message: xsql.Message{"id1": 3, "f1": "v1"},
  653. },
  654. },
  655. {
  656. &xsql.Tuple{
  657. Emitter: "src1",
  658. Message: xsql.Message{"id1": 2, "f1": "v2"},
  659. },
  660. },
  661. },
  662. result: []map[string]interface{}{{
  663. "id1": float64(1),
  664. }, {
  665. "id1": float64(2),
  666. }},
  667. },
  668. {
  669. sql: "SELECT id1 FROM src1 GROUP BY TUMBLINGWINDOW(ss, 10), f1",
  670. data: xsql.GroupedTuplesSet{
  671. {
  672. &xsql.Tuple{
  673. Emitter: "src1",
  674. Message: xsql.Message{"id1": 1, "f1": "v1"},
  675. },
  676. &xsql.Tuple{
  677. Emitter: "src1",
  678. Message: xsql.Message{"id1": 3, "f1": "v1"},
  679. },
  680. },
  681. {
  682. &xsql.Tuple{
  683. Emitter: "src1",
  684. Message: xsql.Message{"id2": 2, "f1": "v2"},
  685. },
  686. },
  687. },
  688. result: []map[string]interface{}{{
  689. "id1": float64(1),
  690. }, {}},
  691. },
  692. {
  693. sql: "SELECT src2.id2 FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
  694. data: xsql.GroupedTuplesSet{
  695. {
  696. &xsql.JoinTuple{
  697. Tuples: []xsql.Tuple{
  698. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  699. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  700. },
  701. },
  702. },
  703. {
  704. &xsql.JoinTuple{
  705. Tuples: []xsql.Tuple{
  706. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  707. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  708. },
  709. },
  710. },
  711. {
  712. &xsql.JoinTuple{
  713. Tuples: []xsql.Tuple{
  714. {Emitter: "src1", Message: xsql.Message{"id1": 3, "f1": "v1"}},
  715. },
  716. },
  717. },
  718. },
  719. result: []map[string]interface{}{{
  720. "id2": float64(2),
  721. }, {
  722. "id2": float64(4),
  723. }, {}},
  724. },
  725. {
  726. sql: "SELECT src1.*, f2 FROM src1 left join src2 GROUP BY TUMBLINGWINDOW(ss, 10)",
  727. data: xsql.JoinTupleSets{
  728. xsql.JoinTuple{
  729. Tuples: []xsql.Tuple{
  730. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  731. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  732. },
  733. },
  734. xsql.JoinTuple{
  735. Tuples: []xsql.Tuple{
  736. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  737. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  738. },
  739. },
  740. xsql.JoinTuple{
  741. Tuples: []xsql.Tuple{
  742. {Emitter: "src1", Message: xsql.Message{"id1": 3, "f1": "v1"}},
  743. },
  744. },
  745. },
  746. result: []map[string]interface{}{{
  747. "id1": float64(1),
  748. "f1": "v1",
  749. "f2": "w2",
  750. }, {
  751. "id1": float64(2),
  752. "f1": "v2",
  753. "f2": "w3",
  754. }, {
  755. "id1": float64(3),
  756. "f1": "v1",
  757. }},
  758. },
  759. {
  760. sql: "SELECT * FROM src1 left join src2 GROUP BY TUMBLINGWINDOW(ss, 10)",
  761. data: xsql.JoinTupleSets{
  762. xsql.JoinTuple{
  763. Tuples: []xsql.Tuple{
  764. {Emitter: "src1", Message: xsql.Message{"id": 1, "f1": "v1"}},
  765. {Emitter: "src2", Message: xsql.Message{"id": 2, "f2": "w2"}},
  766. },
  767. },
  768. xsql.JoinTuple{
  769. Tuples: []xsql.Tuple{
  770. {Emitter: "src1", Message: xsql.Message{"id": 2, "f1": "v2"}},
  771. {Emitter: "src2", Message: xsql.Message{"id": 4, "f2": "w3"}},
  772. },
  773. },
  774. xsql.JoinTuple{
  775. Tuples: []xsql.Tuple{
  776. {Emitter: "src1", Message: xsql.Message{"id": 3, "f1": "v1"}},
  777. },
  778. },
  779. },
  780. result: []map[string]interface{}{{
  781. "id": float64(1),
  782. "f1": "v1",
  783. "f2": "w2",
  784. }, {
  785. "id": float64(2),
  786. "f1": "v2",
  787. "f2": "w3",
  788. }, {
  789. "id": float64(3),
  790. "f1": "v1",
  791. }},
  792. },
  793. {
  794. sql: "SELECT src1.* FROM src1 GROUP BY TUMBLINGWINDOW(ss, 10), f1",
  795. data: xsql.GroupedTuplesSet{
  796. {
  797. &xsql.Tuple{
  798. Emitter: "src1",
  799. Message: xsql.Message{"id1": 1, "f1": "v1"},
  800. },
  801. &xsql.Tuple{
  802. Emitter: "src1",
  803. Message: xsql.Message{"id1": 3, "f1": "v1"},
  804. },
  805. },
  806. {
  807. &xsql.Tuple{
  808. Emitter: "src1",
  809. Message: xsql.Message{"id1": 2, "f1": "v2"},
  810. },
  811. },
  812. },
  813. result: []map[string]interface{}{{
  814. "id1": float64(1),
  815. "f1": "v1",
  816. }, {
  817. "id1": float64(2),
  818. "f1": "v2",
  819. }},
  820. },
  821. {
  822. sql: "SELECT src2.id2, src1.* FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
  823. data: xsql.GroupedTuplesSet{
  824. {
  825. &xsql.JoinTuple{
  826. Tuples: []xsql.Tuple{
  827. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  828. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  829. },
  830. },
  831. },
  832. {
  833. &xsql.JoinTuple{
  834. Tuples: []xsql.Tuple{
  835. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  836. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  837. },
  838. },
  839. },
  840. {
  841. &xsql.JoinTuple{
  842. Tuples: []xsql.Tuple{
  843. {Emitter: "src1", Message: xsql.Message{"id1": 3, "f1": "v1"}},
  844. },
  845. },
  846. },
  847. },
  848. result: []map[string]interface{}{{
  849. "id2": float64(2),
  850. "id1": float64(1),
  851. "f1": "v1",
  852. }, {
  853. "id2": float64(4),
  854. "id1": float64(2),
  855. "f1": "v2",
  856. }, {
  857. "id1": float64(3),
  858. "f1": "v1",
  859. }},
  860. },
  861. {
  862. sql: "SELECT src2.id2, src1.* FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
  863. data: xsql.GroupedTuplesSet{
  864. {
  865. &xsql.JoinTuple{
  866. Tuples: []xsql.Tuple{
  867. {Emitter: "src1", Message: xsql.Message{"id1": 1, "f1": "v1"}},
  868. {Emitter: "src2", Message: xsql.Message{"id2": 2, "f2": "w2"}},
  869. },
  870. },
  871. },
  872. {
  873. &xsql.JoinTuple{
  874. Tuples: []xsql.Tuple{
  875. {Emitter: "src1", Message: xsql.Message{"id1": 2, "f1": "v2"}},
  876. {Emitter: "src2", Message: xsql.Message{"id2": 4, "f2": "w3"}},
  877. },
  878. },
  879. },
  880. {
  881. &xsql.JoinTuple{
  882. Tuples: []xsql.Tuple{
  883. {Emitter: "src1", Message: xsql.Message{"id1": 3, "f1": "v1"}},
  884. },
  885. },
  886. },
  887. },
  888. result: []map[string]interface{}{{
  889. "id2": float64(2),
  890. "id1": float64(1),
  891. "f1": "v1",
  892. }, {
  893. "id2": float64(4),
  894. "id1": float64(2),
  895. "f1": "v2",
  896. }, {
  897. "id1": float64(3),
  898. "f1": "v1",
  899. }},
  900. },
  901. }
  902. fmt.Printf("The test bucket size is %d.\n\n", len(tests))
  903. contextLogger := common.Log.WithField("rule", "TestProjectPlan_MultiInput")
  904. ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
  905. for i, tt := range tests {
  906. stmt, _ := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
  907. pp := &ProjectPlan{Fields: stmt.Fields}
  908. pp.isTest = true
  909. result := pp.Apply(ctx, tt.data)
  910. var mapRes []map[string]interface{}
  911. if v, ok := result.([]byte); ok {
  912. err := json.Unmarshal(v, &mapRes)
  913. if err != nil {
  914. t.Errorf("Failed to parse the input into map.\n")
  915. continue
  916. }
  917. //fmt.Printf("%t\n", mapRes["rengine_field_0"])
  918. if !reflect.DeepEqual(tt.result, mapRes) {
  919. t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
  920. }
  921. } else {
  922. t.Errorf("The returned result is not type of []byte\n")
  923. }
  924. }
  925. }
  926. func TestProjectPlan_Funcs(t *testing.T) {
  927. var tests = []struct {
  928. sql string
  929. data interface{}
  930. result []map[string]interface{}
  931. }{
  932. {
  933. sql: "SELECT round(a) as r FROM test",
  934. data: &xsql.Tuple{
  935. Emitter: "test",
  936. Message: xsql.Message{
  937. "a": 47.5,
  938. },
  939. },
  940. result: []map[string]interface{}{{
  941. "r": float64(48),
  942. }},
  943. }, {
  944. sql: "SELECT round(a) as r FROM test GROUP BY TumblingWindow(ss, 10)",
  945. data: xsql.WindowTuplesSet{
  946. xsql.WindowTuples{
  947. Emitter: "test",
  948. Tuples: []xsql.Tuple{
  949. {
  950. Emitter: "src1",
  951. Message: xsql.Message{"a": 53.1},
  952. }, {
  953. Emitter: "src1",
  954. Message: xsql.Message{"a": 27.4},
  955. }, {
  956. Emitter: "src1",
  957. Message: xsql.Message{"a": 123123.7},
  958. },
  959. },
  960. },
  961. },
  962. result: []map[string]interface{}{{
  963. "r": float64(53),
  964. }, {
  965. "r": float64(27),
  966. }, {
  967. "r": float64(123124),
  968. }},
  969. }, {
  970. sql: "SELECT round(a) as r FROM test GROUP BY TumblingWindow(ss, 10)",
  971. data: xsql.WindowTuplesSet{
  972. xsql.WindowTuples{
  973. Emitter: "test",
  974. Tuples: []xsql.Tuple{
  975. {
  976. Emitter: "src1",
  977. Message: xsql.Message{"a": 53.1},
  978. }, {
  979. Emitter: "src1",
  980. Message: xsql.Message{"a": 27.4},
  981. }, {
  982. Emitter: "src1",
  983. Message: xsql.Message{"a": 123123.7},
  984. },
  985. },
  986. },
  987. },
  988. result: []map[string]interface{}{{
  989. "r": float64(53),
  990. }, {
  991. "r": float64(27),
  992. }, {
  993. "r": float64(123124),
  994. }},
  995. }, {
  996. sql: "SELECT round(a) as r FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
  997. data: xsql.JoinTupleSets{
  998. xsql.JoinTuple{
  999. Tuples: []xsql.Tuple{
  1000. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 65.55}},
  1001. {Emitter: "test1", Message: xsql.Message{"id": 1, "b": 12}},
  1002. },
  1003. },
  1004. xsql.JoinTuple{
  1005. Tuples: []xsql.Tuple{
  1006. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 73.499}},
  1007. {Emitter: "test1", Message: xsql.Message{"id": 2, "b": 34}},
  1008. },
  1009. },
  1010. xsql.JoinTuple{
  1011. Tuples: []xsql.Tuple{
  1012. {Emitter: "test", Message: xsql.Message{"id": 3, "a": 88.88}},
  1013. {Emitter: "test1", Message: xsql.Message{"id": 3, "b": 6}},
  1014. },
  1015. },
  1016. },
  1017. result: []map[string]interface{}{{
  1018. "r": float64(66),
  1019. }, {
  1020. "r": float64(73),
  1021. }, {
  1022. "r": float64(89),
  1023. }},
  1024. }, {
  1025. sql: "SELECT CONCAT(test.id, test.a, test1.b) as concat FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
  1026. data: xsql.JoinTupleSets{
  1027. xsql.JoinTuple{
  1028. Tuples: []xsql.Tuple{
  1029. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 65.55}},
  1030. {Emitter: "test1", Message: xsql.Message{"id": 1, "b": 12}},
  1031. },
  1032. },
  1033. xsql.JoinTuple{
  1034. Tuples: []xsql.Tuple{
  1035. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 73.499}},
  1036. {Emitter: "test1", Message: xsql.Message{"id": 2, "b": 34}},
  1037. },
  1038. },
  1039. xsql.JoinTuple{
  1040. Tuples: []xsql.Tuple{
  1041. {Emitter: "test", Message: xsql.Message{"id": 3, "a": 88.88}},
  1042. {Emitter: "test1", Message: xsql.Message{"id": 3, "b": 6}},
  1043. },
  1044. },
  1045. },
  1046. result: []map[string]interface{}{{
  1047. "concat": "165.5512",
  1048. }, {
  1049. "concat": "273.49934",
  1050. }, {
  1051. "concat": "388.886",
  1052. }},
  1053. }, {
  1054. sql: "SELECT count(a) as r FROM test",
  1055. data: &xsql.Tuple{
  1056. Emitter: "test",
  1057. Message: xsql.Message{
  1058. "a": 47.5,
  1059. },
  1060. },
  1061. result: []map[string]interface{}{{
  1062. "r": float64(1),
  1063. }},
  1064. }, {
  1065. sql: "SELECT meta(test.device) as d FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
  1066. data: xsql.JoinTupleSets{
  1067. xsql.JoinTuple{
  1068. Tuples: []xsql.Tuple{
  1069. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 65.55}, Metadata: xsql.Metadata{"device": "devicea"}},
  1070. {Emitter: "test1", Message: xsql.Message{"id": 1, "b": 12}},
  1071. },
  1072. },
  1073. xsql.JoinTuple{
  1074. Tuples: []xsql.Tuple{
  1075. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 73.499}, Metadata: xsql.Metadata{"device": "deviceb"}},
  1076. {Emitter: "test1", Message: xsql.Message{"id": 2, "b": 34}},
  1077. },
  1078. },
  1079. xsql.JoinTuple{
  1080. Tuples: []xsql.Tuple{
  1081. {Emitter: "test", Message: xsql.Message{"id": 3, "a": 88.88}, Metadata: xsql.Metadata{"device": "devicec"}},
  1082. {Emitter: "test1", Message: xsql.Message{"id": 3, "b": 6}},
  1083. },
  1084. },
  1085. },
  1086. result: []map[string]interface{}{{
  1087. "d": "devicea",
  1088. }, {
  1089. "d": "deviceb",
  1090. }, {
  1091. "d": "devicec",
  1092. }},
  1093. },
  1094. }
  1095. fmt.Printf("The test bucket size is %d.\n\n", len(tests))
  1096. contextLogger := common.Log.WithField("rule", "TestProjectPlan_Funcs")
  1097. ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
  1098. for i, tt := range tests {
  1099. stmt, err := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
  1100. if err != nil {
  1101. t.Error(err)
  1102. }
  1103. pp := &ProjectPlan{Fields: stmt.Fields, IsAggregate: xsql.IsAggStatement(stmt)}
  1104. pp.isTest = true
  1105. result := pp.Apply(ctx, tt.data)
  1106. var mapRes []map[string]interface{}
  1107. if v, ok := result.([]byte); ok {
  1108. err := json.Unmarshal(v, &mapRes)
  1109. if err != nil {
  1110. t.Errorf("Failed to parse the input into map.\n")
  1111. continue
  1112. }
  1113. //fmt.Printf("%t\n", mapRes["rengine_field_0"])
  1114. if !reflect.DeepEqual(tt.result, mapRes) {
  1115. t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
  1116. }
  1117. } else {
  1118. t.Errorf("%d. The returned result is not type of []byte\n", i)
  1119. }
  1120. }
  1121. }
  1122. func TestProjectPlan_AggFuncs(t *testing.T) {
  1123. var tests = []struct {
  1124. sql string
  1125. data interface{}
  1126. result []map[string]interface{}
  1127. }{
  1128. {
  1129. sql: "SELECT count(*) as c, round(a) as r FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
  1130. data: xsql.GroupedTuplesSet{
  1131. {
  1132. &xsql.JoinTuple{
  1133. Tuples: []xsql.Tuple{
  1134. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1135. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1136. },
  1137. },
  1138. &xsql.JoinTuple{
  1139. Tuples: []xsql.Tuple{
  1140. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.51}},
  1141. {Emitter: "src2", Message: xsql.Message{"id": 5, "color": "w2"}},
  1142. },
  1143. },
  1144. },
  1145. {
  1146. &xsql.JoinTuple{
  1147. Tuples: []xsql.Tuple{
  1148. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 89.03}},
  1149. {Emitter: "src2", Message: xsql.Message{"id": 2, "color": "w1"}},
  1150. },
  1151. },
  1152. &xsql.JoinTuple{
  1153. Tuples: []xsql.Tuple{
  1154. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 14.6}},
  1155. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w1"}},
  1156. },
  1157. },
  1158. },
  1159. },
  1160. result: []map[string]interface{}{{
  1161. "c": float64(2),
  1162. "r": float64(122),
  1163. }, {
  1164. "c": float64(2),
  1165. "r": float64(89),
  1166. }},
  1167. },
  1168. {
  1169. sql: "SELECT avg(a) as avg FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
  1170. data: xsql.GroupedTuplesSet{
  1171. {
  1172. &xsql.JoinTuple{
  1173. Tuples: []xsql.Tuple{
  1174. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1175. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1176. },
  1177. },
  1178. &xsql.JoinTuple{
  1179. Tuples: []xsql.Tuple{
  1180. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 68.54}},
  1181. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1182. },
  1183. },
  1184. &xsql.JoinTuple{
  1185. Tuples: []xsql.Tuple{
  1186. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 98.31}},
  1187. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w2"}},
  1188. },
  1189. },
  1190. &xsql.JoinTuple{
  1191. Tuples: []xsql.Tuple{
  1192. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.54}},
  1193. {Emitter: "src2", Message: xsql.Message{"id": 5, "color": "w2"}},
  1194. },
  1195. },
  1196. },
  1197. {
  1198. &xsql.JoinTuple{
  1199. Tuples: []xsql.Tuple{
  1200. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 89.03}},
  1201. {Emitter: "src2", Message: xsql.Message{"id": 2, "color": "w1"}},
  1202. },
  1203. },
  1204. &xsql.JoinTuple{
  1205. Tuples: []xsql.Tuple{
  1206. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 14.6}},
  1207. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w1"}},
  1208. },
  1209. },
  1210. },
  1211. },
  1212. result: []map[string]interface{}{{
  1213. "avg": 116.68,
  1214. }, {
  1215. "avg": 51.815,
  1216. }},
  1217. },
  1218. {
  1219. sql: "SELECT max(a) as max FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
  1220. data: xsql.GroupedTuplesSet{
  1221. {
  1222. &xsql.JoinTuple{
  1223. Tuples: []xsql.Tuple{
  1224. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1225. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1226. },
  1227. },
  1228. &xsql.JoinTuple{
  1229. Tuples: []xsql.Tuple{
  1230. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 68.55}},
  1231. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1232. },
  1233. },
  1234. &xsql.JoinTuple{
  1235. Tuples: []xsql.Tuple{
  1236. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.51}},
  1237. {Emitter: "src2", Message: xsql.Message{"id": 5, "color": "w2"}},
  1238. },
  1239. },
  1240. },
  1241. {
  1242. &xsql.JoinTuple{
  1243. Tuples: []xsql.Tuple{
  1244. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 89.03}},
  1245. {Emitter: "src2", Message: xsql.Message{"id": 2, "color": "w1"}},
  1246. },
  1247. },
  1248. &xsql.JoinTuple{
  1249. Tuples: []xsql.Tuple{
  1250. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 14.6}},
  1251. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w1"}},
  1252. },
  1253. },
  1254. },
  1255. },
  1256. result: []map[string]interface{}{{
  1257. "max": 177.51,
  1258. }, {
  1259. "max": 89.03,
  1260. }},
  1261. },
  1262. {
  1263. sql: "SELECT min(a) as min FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
  1264. data: xsql.JoinTupleSets{
  1265. xsql.JoinTuple{
  1266. Tuples: []xsql.Tuple{
  1267. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1268. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1269. },
  1270. },
  1271. xsql.JoinTuple{
  1272. Tuples: []xsql.Tuple{
  1273. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 68.55}},
  1274. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1275. },
  1276. },
  1277. xsql.JoinTuple{
  1278. Tuples: []xsql.Tuple{
  1279. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.51}},
  1280. {Emitter: "src2", Message: xsql.Message{"id": 5, "color": "w2"}},
  1281. },
  1282. },
  1283. },
  1284. result: []map[string]interface{}{{
  1285. "min": 68.55,
  1286. }},
  1287. }, {
  1288. sql: "SELECT sum(a) as sum FROM test GROUP BY TumblingWindow(ss, 10)",
  1289. data: xsql.WindowTuplesSet{
  1290. xsql.WindowTuples{
  1291. Emitter: "test",
  1292. Tuples: []xsql.Tuple{
  1293. {
  1294. Emitter: "src1",
  1295. Message: xsql.Message{"a": 53},
  1296. }, {
  1297. Emitter: "src1",
  1298. Message: xsql.Message{"a": 27},
  1299. }, {
  1300. Emitter: "src1",
  1301. Message: xsql.Message{"a": 123123},
  1302. },
  1303. },
  1304. },
  1305. },
  1306. result: []map[string]interface{}{{
  1307. "sum": float64(123203),
  1308. }},
  1309. }, {
  1310. sql: "SELECT sum(a) as sum FROM test GROUP BY TumblingWindow(ss, 10)",
  1311. data: xsql.WindowTuplesSet{
  1312. xsql.WindowTuples{
  1313. Emitter: "test",
  1314. Tuples: []xsql.Tuple{
  1315. {
  1316. Emitter: "src1",
  1317. Message: xsql.Message{"a": 53},
  1318. }, {
  1319. Emitter: "src1",
  1320. Message: xsql.Message{"a": 27},
  1321. }, {
  1322. Emitter: "src1",
  1323. Message: xsql.Message{"a": 123123},
  1324. },
  1325. },
  1326. },
  1327. },
  1328. result: []map[string]interface{}{{
  1329. "sum": float64(123203),
  1330. }},
  1331. },
  1332. {
  1333. sql: "SELECT count(*) as c, meta(test1.device) as d FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
  1334. data: xsql.GroupedTuplesSet{
  1335. {
  1336. &xsql.JoinTuple{
  1337. Tuples: []xsql.Tuple{
  1338. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1339. {Emitter: "test1", Message: xsql.Message{"id": 1, "color": "w2"}, Metadata: xsql.Metadata{"device": "devicea"}},
  1340. },
  1341. },
  1342. &xsql.JoinTuple{
  1343. Tuples: []xsql.Tuple{
  1344. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.51}},
  1345. {Emitter: "test1", Message: xsql.Message{"id": 5, "color": "w2"}, Metadata: xsql.Metadata{"device": "deviceb"}},
  1346. },
  1347. },
  1348. },
  1349. {
  1350. &xsql.JoinTuple{
  1351. Tuples: []xsql.Tuple{
  1352. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 89.03}},
  1353. {Emitter: "test1", Message: xsql.Message{"id": 2, "color": "w1"}, Metadata: xsql.Metadata{"device": "devicec"}},
  1354. },
  1355. },
  1356. &xsql.JoinTuple{
  1357. Tuples: []xsql.Tuple{
  1358. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 14.6}},
  1359. {Emitter: "test1", Message: xsql.Message{"id": 4, "color": "w1"}, Metadata: xsql.Metadata{"device": "deviced"}},
  1360. },
  1361. },
  1362. },
  1363. },
  1364. result: []map[string]interface{}{{
  1365. "c": float64(2),
  1366. "d": "devicea",
  1367. }, {
  1368. "c": float64(2),
  1369. "d": "devicec",
  1370. }},
  1371. },
  1372. }
  1373. fmt.Printf("The test bucket size is %d.\n\n", len(tests))
  1374. contextLogger := common.Log.WithField("rule", "TestProjectPlan_AggFuncs")
  1375. ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
  1376. for i, tt := range tests {
  1377. stmt, err := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
  1378. if err != nil {
  1379. t.Error(err)
  1380. }
  1381. pp := &ProjectPlan{Fields: stmt.Fields, IsAggregate: true}
  1382. pp.isTest = true
  1383. result := pp.Apply(ctx, tt.data)
  1384. var mapRes []map[string]interface{}
  1385. if v, ok := result.([]byte); ok {
  1386. err := json.Unmarshal(v, &mapRes)
  1387. if err != nil {
  1388. t.Errorf("Failed to parse the input into map.\n")
  1389. continue
  1390. }
  1391. //fmt.Printf("%t\n", mapRes["rengine_field_0"])
  1392. if !reflect.DeepEqual(tt.result, mapRes) {
  1393. t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
  1394. }
  1395. } else {
  1396. t.Errorf("%d. %q\n\nThe returned result is not type of []byte: %#v\n", i, tt.sql, result)
  1397. }
  1398. }
  1399. }
  1400. func TestProjectPlanError(t *testing.T) {
  1401. var tests = []struct {
  1402. sql string
  1403. data interface{}
  1404. result interface{}
  1405. }{
  1406. {
  1407. sql: "SELECT a FROM test",
  1408. data: errors.New("an error from upstream"),
  1409. result: errors.New("an error from upstream"),
  1410. }, {
  1411. sql: "SELECT a * 5 FROM test",
  1412. data: &xsql.Tuple{
  1413. Emitter: "test",
  1414. Message: xsql.Message{
  1415. "a": "val_a",
  1416. },
  1417. },
  1418. result: errors.New("run Select error: invalid operation string(val_a) * int64(5)"),
  1419. }, {
  1420. sql: `SELECT a[0]->b AS ab FROM test`,
  1421. data: &xsql.Tuple{
  1422. Emitter: "test",
  1423. Message: xsql.Message{
  1424. "a": "common string",
  1425. },
  1426. },
  1427. result: errors.New("run Select error: invalid operation string(common string) [] *xsql.BracketEvalResult(&{0 0})"),
  1428. }, {
  1429. sql: `SELECT round(a) as r FROM test`,
  1430. data: &xsql.Tuple{
  1431. Emitter: "test",
  1432. Message: xsql.Message{
  1433. "a": "common string",
  1434. },
  1435. },
  1436. result: errors.New("run Select error: call func round error: only float64 & int type are supported"),
  1437. }, {
  1438. sql: `SELECT round(a) as r FROM test`,
  1439. data: &xsql.Tuple{
  1440. Emitter: "test",
  1441. Message: xsql.Message{
  1442. "abc": "common string",
  1443. },
  1444. },
  1445. result: errors.New("run Select error: call func round error: only float64 & int type are supported"),
  1446. }, {
  1447. sql: "SELECT avg(a) as avg FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
  1448. data: xsql.GroupedTuplesSet{
  1449. {
  1450. &xsql.JoinTuple{
  1451. Tuples: []xsql.Tuple{
  1452. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 122.33}},
  1453. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1454. },
  1455. },
  1456. &xsql.JoinTuple{
  1457. Tuples: []xsql.Tuple{
  1458. {Emitter: "test", Message: xsql.Message{"id": 1, "a": 68.54}},
  1459. {Emitter: "src2", Message: xsql.Message{"id": 1, "color": "w2"}},
  1460. },
  1461. },
  1462. &xsql.JoinTuple{
  1463. Tuples: []xsql.Tuple{
  1464. {Emitter: "test", Message: xsql.Message{"id": 4, "a": "dde"}},
  1465. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w2"}},
  1466. },
  1467. },
  1468. &xsql.JoinTuple{
  1469. Tuples: []xsql.Tuple{
  1470. {Emitter: "test", Message: xsql.Message{"id": 5, "a": 177.54}},
  1471. {Emitter: "src2", Message: xsql.Message{"id": 5, "color": "w2"}},
  1472. },
  1473. },
  1474. },
  1475. {
  1476. &xsql.JoinTuple{
  1477. Tuples: []xsql.Tuple{
  1478. {Emitter: "test", Message: xsql.Message{"id": 2, "a": 89.03}},
  1479. {Emitter: "src2", Message: xsql.Message{"id": 2, "color": "w1"}},
  1480. },
  1481. },
  1482. &xsql.JoinTuple{
  1483. Tuples: []xsql.Tuple{
  1484. {Emitter: "test", Message: xsql.Message{"id": 4, "a": 14.6}},
  1485. {Emitter: "src2", Message: xsql.Message{"id": 4, "color": "w1"}},
  1486. },
  1487. },
  1488. },
  1489. },
  1490. result: errors.New("run Select error: call func avg error: requires float64 but found string(dde)"),
  1491. }, {
  1492. sql: "SELECT sum(a) as sum FROM test GROUP BY TumblingWindow(ss, 10)",
  1493. data: xsql.WindowTuplesSet{
  1494. xsql.WindowTuples{
  1495. Emitter: "test",
  1496. Tuples: []xsql.Tuple{
  1497. {
  1498. Emitter: "src1",
  1499. Message: xsql.Message{"a": 53},
  1500. }, {
  1501. Emitter: "src1",
  1502. Message: xsql.Message{"a": "ddd"},
  1503. }, {
  1504. Emitter: "src1",
  1505. Message: xsql.Message{"a": 123123},
  1506. },
  1507. },
  1508. },
  1509. },
  1510. result: errors.New("run Select error: call func sum error: requires int but found string(ddd)"),
  1511. },
  1512. }
  1513. fmt.Printf("The test bucket size is %d.\n\n", len(tests))
  1514. contextLogger := common.Log.WithField("rule", "TestProjectPlanError")
  1515. ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
  1516. for i, tt := range tests {
  1517. stmt, _ := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
  1518. pp := &ProjectPlan{Fields: stmt.Fields, IsAggregate: xsql.IsAggStatement(stmt)}
  1519. pp.isTest = true
  1520. result := pp.Apply(ctx, tt.data)
  1521. if !reflect.DeepEqual(tt.result, result) {
  1522. t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, result)
  1523. }
  1524. }
  1525. }