1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093 |
- package plans
- import (
- "encoding/json"
- "fmt"
- "github.com/emqx/kuiper/common"
- "github.com/emqx/kuiper/xsql"
- "github.com/emqx/kuiper/xstream/contexts"
- "reflect"
- "strings"
- "testing"
- )
- func TestProjectPlan_Apply1(t *testing.T) {
- var tests = []struct {
- sql string
- data *xsql.Tuple
- result []map[string]interface{}
- }{
- {
- sql: "SELECT a FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val_a",
- },
- },
- result: []map[string]interface{}{{
- "a": "val_a",
- }},
- },
- {
- sql: "SELECT ts FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val_a",
- "ts" : common.TimeFromUnixMilli(1568854573431),
- },
- },
- result: []map[string]interface{}{{
- "ts": "2019-09-19T00:56:13.000000431Z",
- }},
- },
- {
- sql: "SELECT A FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val_a",
- },
- },
- result: []map[string]interface{}{{
- "A": "val_a",
- }},
- },
- {
- sql: `SELECT "value" FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{{
- DEFAULT_FIELD_NAME_PREFIX + "0" : "value",
- }},
- },
- {
- sql: `SELECT 3.4 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{{
- DEFAULT_FIELD_NAME_PREFIX + "0" : 3.4,
- }},
- },
- {
- sql: `SELECT 5 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{{
- DEFAULT_FIELD_NAME_PREFIX + "0": 5.0,
- }},
- },
- {
- sql: `SELECT a, "value" AS b FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val_a",
- },
- },
- result: []map[string]interface{}{{
- "a" : "val_a",
- "b" : "value",
- }},
- },
- {
- sql: `SELECT a, "value" AS b, 3.14 as Pi, 0 as Zero FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val_a",
- },
- },
- result: []map[string]interface{}{{
- "a" : "val_a",
- "b" : "value",
- "Pi" : 3.14,
- "Zero" : 0.0,
- }},
- },
- {
- sql: `SELECT a->b AS ab FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : map[string]interface{}{"b" : "hello"},
- },
- },
- result: []map[string]interface{}{{
- "ab" : "hello",
- }},
- },
- {
- sql: `SELECT a[0]->b AS ab FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : []interface{}{
- map[string]interface{}{"b" : "hello1"},
- map[string]interface{}{"b" : "hello2"},
- },
- },
- },
- result: []map[string]interface{}{{
- "ab" : "hello1",
- }},
- },
- {
- sql: `SELECT a->c->d AS f1 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : map[string]interface{}{
- "b" : "hello",
- "c" : map[string]interface{}{
- "d": 35.2,
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "f1" : 35.2,
- }},
- },
- //The int type is not supported yet, the json parser returns float64 for int values
- {
- sql: `SELECT a->c->d AS f1 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : map[string]interface{}{
- "b" : "hello",
- "c" : map[string]interface{}{
- "d": float64(35),
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "f1" : float64(35),
- }},
- },
- {
- sql: "SELECT a FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{
- {},
- },
- },
- {
- sql: "SELECT * FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{
- {},
- },
- },
- {
- sql: `SELECT * FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : map[string]interface{}{
- "b" : "hello",
- "c" : map[string]interface{}{
- "d": 35.2,
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "a" : map[string]interface{} {
- "b" : "hello",
- "c" : map[string]interface{} {
- "d" : 35.2,
- },
- },
- }},
- },
- {
- sql: `SELECT * FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a" : "val1",
- "b" : 3.14,
- },
- },
- result: []map[string]interface{}{{
- "a" : "val1",
- "b" : 3.14,
- }},
- },
- {
- sql: `SELECT 3*4 AS f1 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{{
- "f1" : float64(12),
- }},
- },
- {
- sql: `SELECT 4.5*2 AS f1 FROM test`,
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- },
- },
- result: []map[string]interface{}{{
- "f1" : float64(9),
- }},
- },
- }
- fmt.Printf("The test bucket size is %d.\n\n", len(tests))
- contextLogger := common.Log.WithField("rule", "TestProjectPlan_Apply1")
- ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
- for i, tt := range tests {
- stmt, _ := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
- pp := &ProjectPlan{Fields:stmt.Fields}
- pp.isTest = true
- result := pp.Apply(ctx, tt.data)
- var mapRes []map[string]interface{}
- if v, ok := result.([]byte); ok {
- err := json.Unmarshal(v, &mapRes)
- if err != nil {
- t.Errorf("Failed to parse the input into map.\n")
- continue
- }
- //fmt.Printf("%t\n", mapRes["rengine_field_0"])
- if !reflect.DeepEqual(tt.result, mapRes) {
- t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
- }
- } else {
- t.Errorf("The returned result is not type of []byte\n")
- }
- }
- }
- func TestProjectPlan_MultiInput(t *testing.T) {
- var tests = []struct {
- sql string
- data interface{}
- result []map[string]interface{}
- }{
- {
- sql: "SELECT * FROM tbl WHERE abc*2+3 > 12 AND abc < 20",
- data: &xsql.Tuple{
- Emitter: "tbl",
- Message: xsql.Message{
- "abc" : int64(6),
- },
- },
- result: []map[string]interface{}{{
- "abc" : float64(6), //json marshall problem
- }},
- },
- {
- sql: "SELECT abc FROM tbl WHERE abc*2+3 > 12 OR def = \"hello\"",
- data: &xsql.Tuple{
- Emitter: "tbl",
- Message: xsql.Message{
- "abc" : int64(34),
- "def" : "hello",
- },
- },
- result: []map[string]interface{}{{
- "abc" : float64(34),
- }},
- },
- {
- sql: "SELECT id1 FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.WindowTuplesSet{
- xsql.WindowTuples{
- Emitter:"src1",
- Tuples:[]xsql.Tuple{
- {
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 1, "f1" : "v1", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 2, "f1" : "v2", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 3, "f1" : "v1", },
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id1" : float64(1),
- },{
- "id1" : float64(2),
- },{
- "id1" : float64(3),
- }},
- },
- {
- sql: "SELECT * FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.WindowTuplesSet{
- xsql.WindowTuples{
- Emitter:"src1",
- Tuples:[]xsql.Tuple{
- {
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 1, "f1" : "v1", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 2, "f1" : "v2", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 3, "f1" : "v1", },
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id1" : float64(1),
- "f1" : "v1",
- },{
- "id1" : float64(2),
- "f1" : "v2",
- },{
- "id1" : float64(3),
- "f1" : "v1",
- }},
- },
- {
- sql: "SELECT src1.* FROM src1 WHERE f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.WindowTuplesSet{
- xsql.WindowTuples{
- Emitter:"src1",
- Tuples:[]xsql.Tuple{
- {
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 1, "f1" : "v1", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 2, "f1" : "v2", },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 3, "f1" : "v1", },
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id1" : float64(1),
- "f1" : "v1",
- },{
- "id1" : float64(2),
- "f1" : "v2",
- },{
- "id1" : float64(3),
- "f1" : "v1",
- }},
- },
- {
- sql: "SELECT id1 FROM src1 left join src2 on src1.id1 = src2.id2 WHERE src1.f1 = \"v1\" GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 2, "f2" : "w2",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 4, "f2" : "w3",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 3, "f1" : "v1",},},
- },
- },
- },
- result: []map[string]interface{}{{
- "id1" : float64(1),
- },{
- "id1" : float64(2),
- },{
- "id1" : float64(3),
- }},
- },
- {
- sql: "SELECT abc FROM tbl group by abc",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.Tuple{
- Emitter: "tbl",
- Message: xsql.Message{
- "abc" : int64(6),
- "def" : "hello",
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "abc" : float64(6),
- },},
- },
- {
- sql: "SELECT id1 FROM src1 GROUP BY TUMBLINGWINDOW(ss, 10), f1",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 1, "f1" : "v1", },
- },
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 3, "f1" : "v1", },
- },
- },
- {
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 2, "f1" : "v2", },
- },
- },
- },
- result: []map[string]interface{}{{
- "id1": float64(1),
- },{
- "id1": float64(2),
- },},
- },
- {
- sql: "SELECT src2.id2 FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 2, "f2" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 4, "f2" : "w3",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 3, "f1" : "v1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id2": float64(2),
- }, {
- "id2": float64(4),
- },{
- },},
- },
- {
- sql: "SELECT src1.*, f2 FROM src1 left join src2 GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 2, "f2" : "w2",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 4, "f2" : "w3",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 3, "f1" : "v1",},},
- },
- },
- },
- result: []map[string]interface{}{{
- "id1" : float64(1),
- "f1" : "v1",
- "f2" : "w2",
- },{
- "id1" : float64(2),
- "f1" : "v2",
- "f2" : "w3",
- },{
- "id1" : float64(3),
- "f1" : "v1",
- }},
- },
- {
- sql: "SELECT * FROM src1 left join src2 GROUP BY TUMBLINGWINDOW(ss, 10)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 2, "f2" : "w2",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 4, "f2" : "w3",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id" : 3, "f1" : "v1",},},
- },
- },
- },
- result: []map[string]interface{}{{
- "id" : float64(1),
- "f1" : "v1",
- "f2" : "w2",
- },{
- "id" : float64(2),
- "f1" : "v2",
- "f2" : "w3",
- },{
- "id" : float64(3),
- "f1" : "v1",
- }},
- },
- {
- sql: "SELECT src1.* FROM src1 GROUP BY TUMBLINGWINDOW(ss, 10), f1",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 1, "f1" : "v1", },
- },
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 3, "f1" : "v1", },
- },
- },
- {
- &xsql.Tuple{
- Emitter: "src1",
- Message: xsql.Message{ "id1" : 2, "f1" : "v2", },
- },
- },
- },
- result: []map[string]interface{}{{
- "id1": float64(1),
- "f1": "v1",
- },{
- "id1": float64(2),
- "f1": "v2",
- },},
- },
- {
- sql: "SELECT src2.id2, src1.* FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 2, "f2" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 4, "f2" : "w3",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 3, "f1" : "v1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id2": float64(2),
- "id1": float64(1),
- "f1" : "v1",
- }, {
- "id2": float64(4),
- "id1": float64(2),
- "f1" : "v2",
- },{
- "id1": float64(3),
- "f1" : "v1",
- },},
- },
- {
- sql: "SELECT src2.id2, src1.* FROM src1 left join src2 on src1.id1 = src2.id2 GROUP BY src2.f2, TUMBLINGWINDOW(ss, 10)",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 1, "f1" : "v1",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 2, "f2" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 2, "f1" : "v2",},},
- {Emitter: "src2", Message: xsql.Message{ "id2" : 4, "f2" : "w3",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "src1", Message: xsql.Message{ "id1" : 3, "f1" : "v1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "id2": float64(2),
- "id1": float64(1),
- "f1" : "v1",
- }, {
- "id2": float64(4),
- "id1": float64(2),
- "f1" : "v2",
- },{
- "id1": float64(3),
- "f1" : "v1",
- },},
- },
- }
- fmt.Printf("The test bucket size is %d.\n\n", len(tests))
- contextLogger := common.Log.WithField("rule", "TestProjectPlan_MultiInput")
- ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
- for i, tt := range tests {
- stmt, _ := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
- pp := &ProjectPlan{Fields:stmt.Fields}
- pp.isTest = true
- result := pp.Apply(ctx, tt.data)
- var mapRes []map[string]interface{}
- if v, ok := result.([]byte); ok {
- err := json.Unmarshal(v, &mapRes)
- if err != nil {
- t.Errorf("Failed to parse the input into map.\n")
- continue
- }
- //fmt.Printf("%t\n", mapRes["rengine_field_0"])
- if !reflect.DeepEqual(tt.result, mapRes) {
- t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
- }
- } else {
- t.Errorf("The returned result is not type of []byte\n")
- }
- }
- }
- func TestProjectPlan_Funcs(t *testing.T) {
- var tests = []struct {
- sql string
- data interface{}
- result []map[string]interface{}
- }{
- {
- sql: "SELECT round(a) as r FROM test",
- data: &xsql.Tuple{
- Emitter: "test",
- Message: xsql.Message{
- "a": 47.5,
- },
- },
- result: []map[string]interface{}{{
- "r": float64(48),
- }},
- }, {
- sql: "SELECT round(a) as r FROM test GROUP BY TumblingWindow(ss, 10)",
- data: xsql.WindowTuplesSet{
- xsql.WindowTuples{
- Emitter:"test",
- Tuples:[]xsql.Tuple{
- {
- Emitter: "src1",
- Message: xsql.Message{ "a": 53.1 },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "a": 27.4 },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "a": 123123.7 },
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "r": float64(53),
- },{
- "r": float64(27),
- },{
- "r": float64(123124),
- }},
- }, {
- sql: "SELECT round(a) as r FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 1, "a": 65.55},},
- {Emitter: "test1", Message: xsql.Message{ "id": 1, "b": 12},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 2, "a": 73.499},},
- {Emitter: "test1", Message: xsql.Message{ "id": 2, "b": 34},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 3, "a": 88.88},},
- {Emitter: "test1", Message: xsql.Message{ "id": 3, "b": 6},},
- },
- },
- },
- result: []map[string]interface{}{{
- "r": float64(66),
- },{
- "r": float64(73),
- },{
- "r": float64(89),
- }},
- }, {
- 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)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 1, "a": 65.55},},
- {Emitter: "test1", Message: xsql.Message{ "id": 1, "b": 12},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 2, "a": 73.499},},
- {Emitter: "test1", Message: xsql.Message{ "id": 2, "b": 34},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id": 3, "a": 88.88},},
- {Emitter: "test1", Message: xsql.Message{ "id": 3, "b": 6},},
- },
- },
- },
- result: []map[string]interface{}{{
- "concat": "165.5512",
- },{
- "concat": "273.49934",
- },{
- "concat": "388.886",
- }},
- },
- }
- fmt.Printf("The test bucket size is %d.\n\n", len(tests))
- contextLogger := common.Log.WithField("rule", "TestProjectPlan_Funcs")
- ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
- for i, tt := range tests {
- stmt, err := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
- if err != nil{
- t.Error(err)
- }
- pp := &ProjectPlan{Fields:stmt.Fields}
- pp.isTest = true
- result := pp.Apply(ctx, tt.data)
- var mapRes []map[string]interface{}
- if v, ok := result.([]byte); ok {
- err := json.Unmarshal(v, &mapRes)
- if err != nil {
- t.Errorf("Failed to parse the input into map.\n")
- continue
- }
- //fmt.Printf("%t\n", mapRes["rengine_field_0"])
- if !reflect.DeepEqual(tt.result, mapRes) {
- t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
- }
- } else {
- t.Errorf("The returned result is not type of []byte\n")
- }
- }
- }
- func TestProjectPlan_AggFuncs(t *testing.T) {
- var tests = []struct {
- sql string
- data interface{}
- result []map[string]interface{}
- }{
- {
- 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",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 122.33,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 5, "a": 177.51,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 5, "color" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 2, "a": 89.03,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 2, "color" : "w1",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 4, "a": 14.6,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 4, "color" : "w1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "c": float64(2),
- "r": float64(122),
- },{
- "c": float64(2),
- "r": float64(89),
- }},
- },
- {
- sql: "SELECT avg(a) as avg FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 122.33,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 68.54,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 4, "a": 98.31,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 4, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 5, "a": 177.54,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 5, "color" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 2, "a": 89.03,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 2, "color" : "w1",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 4, "a": 14.6,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 4, "color" : "w1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "avg": 116.68,
- },{
- "avg": 51.815,
- }},
- },
- {
- sql: "SELECT max(a) as max FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10), test1.color",
- data: xsql.GroupedTuplesSet{
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 122.33,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 68.55,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 5, "a": 177.51,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 5, "color" : "w2",},},
- },
- },
- },
- {
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 2, "a": 89.03,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 2, "color" : "w1",},},
- },
- },
- &xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 4, "a": 14.6,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 4, "color" : "w1",},},
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "max": 177.51,
- },{
- "max": 89.03,
- }},
- },
- {
- sql: "SELECT min(a) as min FROM test Inner Join test1 on test.id = test1.id GROUP BY TumblingWindow(ss, 10)",
- data: xsql.JoinTupleSets{
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 122.33,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 1, "a": 68.55,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 1, "color" : "w2",},},
- },
- },
- xsql.JoinTuple{
- Tuples: []xsql.Tuple{
- {Emitter: "test", Message: xsql.Message{ "id" : 5, "a": 177.51,},},
- {Emitter: "src2", Message: xsql.Message{ "id" : 5, "color" : "w2",},},
- },
- },
- },
- result: []map[string]interface{}{{
- "min": 68.55,
- }},
- },{
- sql: "SELECT sum(a) as sum FROM test GROUP BY TumblingWindow(ss, 10)",
- data: xsql.WindowTuplesSet{
- xsql.WindowTuples{
- Emitter:"test",
- Tuples:[]xsql.Tuple{
- {
- Emitter: "src1",
- Message: xsql.Message{ "a": 53 },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "a": 27 },
- },{
- Emitter: "src1",
- Message: xsql.Message{ "a": 123123 },
- },
- },
- },
- },
- result: []map[string]interface{}{{
- "sum": float64(123203),
- }},
- },
- }
- fmt.Printf("The test bucket size is %d.\n\n", len(tests))
- contextLogger := common.Log.WithField("rule", "TestProjectPlan_AggFuncs")
- ctx := contexts.WithValue(contexts.Background(), contexts.LoggerKey, contextLogger)
- for i, tt := range tests {
- stmt, err := xsql.NewParser(strings.NewReader(tt.sql)).Parse()
- if err != nil{
- t.Error(err)
- }
- pp := &ProjectPlan{Fields:stmt.Fields, IsAggregate: true}
- pp.isTest = true
- result := pp.Apply(ctx, tt.data)
- var mapRes []map[string]interface{}
- if v, ok := result.([]byte); ok {
- err := json.Unmarshal(v, &mapRes)
- if err != nil {
- t.Errorf("Failed to parse the input into map.\n")
- continue
- }
- //fmt.Printf("%t\n", mapRes["rengine_field_0"])
- if !reflect.DeepEqual(tt.result, mapRes) {
- t.Errorf("%d. %q\n\nresult mismatch:\n\nexp=%#v\n\ngot=%#v\n\n", i, tt.sql, tt.result, mapRes)
- }
- } else {
- t.Errorf("The returned result is not type of []byte\n")
- }
- }
- }
|