aboutsummaryrefslogtreecommitdiff
path: root/weed/query/engine/string_concatenation_test.go
diff options
context:
space:
mode:
Diffstat (limited to 'weed/query/engine/string_concatenation_test.go')
-rw-r--r--weed/query/engine/string_concatenation_test.go190
1 files changed, 190 insertions, 0 deletions
diff --git a/weed/query/engine/string_concatenation_test.go b/weed/query/engine/string_concatenation_test.go
new file mode 100644
index 000000000..c4843bef6
--- /dev/null
+++ b/weed/query/engine/string_concatenation_test.go
@@ -0,0 +1,190 @@
+package engine
+
+import (
+ "context"
+ "testing"
+)
+
+// TestSQLEngine_StringConcatenationWithLiterals tests string concatenation with || operator
+// This covers the user's reported issue where string literals were being lost
+func TestSQLEngine_StringConcatenationWithLiterals(t *testing.T) {
+ engine := NewTestSQLEngine()
+
+ tests := []struct {
+ name string
+ query string
+ expectedCols []string
+ validateFirst func(t *testing.T, row []string)
+ }{
+ {
+ name: "Simple concatenation with literals",
+ query: "SELECT 'test' || action || 'end' FROM user_events LIMIT 1",
+ expectedCols: []string{"'test'||action||'end'"},
+ validateFirst: func(t *testing.T, row []string) {
+ expected := "testloginend" // action="login" from first row
+ if row[0] != expected {
+ t.Errorf("Expected %s, got %s", expected, row[0])
+ }
+ },
+ },
+ {
+ name: "User's original complex concatenation",
+ query: "SELECT 'test' || action || 'xxx' || action || ' ~~~ ' || status FROM user_events LIMIT 1",
+ expectedCols: []string{"'test'||action||'xxx'||action||'~~~'||status"},
+ validateFirst: func(t *testing.T, row []string) {
+ // First row: action="login", status="active"
+ expected := "testloginxxxlogin ~~~ active"
+ if row[0] != expected {
+ t.Errorf("Expected %s, got %s", expected, row[0])
+ }
+ },
+ },
+ {
+ name: "Mixed columns and literals",
+ query: "SELECT status || '=' || action, 'prefix:' || user_type FROM user_events LIMIT 1",
+ expectedCols: []string{"status||'='||action", "'prefix:'||user_type"},
+ validateFirst: func(t *testing.T, row []string) {
+ // First row: status="active", action="login", user_type="premium"
+ if row[0] != "active=login" {
+ t.Errorf("Expected 'active=login', got %s", row[0])
+ }
+ if row[1] != "prefix:premium" {
+ t.Errorf("Expected 'prefix:premium', got %s", row[1])
+ }
+ },
+ },
+ {
+ name: "Concatenation with spaces in literals",
+ query: "SELECT ' [ ' || status || ' ] ' FROM user_events LIMIT 2",
+ expectedCols: []string{"'['||status||']'"},
+ validateFirst: func(t *testing.T, row []string) {
+ expected := " [ active ] " // status="active" from first row
+ if row[0] != expected {
+ t.Errorf("Expected '%s', got '%s'", expected, row[0])
+ }
+ },
+ },
+ }
+
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ result, err := engine.ExecuteSQL(context.Background(), tt.query)
+ if err != nil {
+ t.Fatalf("Query failed: %v", err)
+ }
+ if result.Error != nil {
+ t.Fatalf("Query returned error: %v", result.Error)
+ }
+
+ // Verify we got results
+ if len(result.Rows) == 0 {
+ t.Fatal("Query returned no rows")
+ }
+
+ // Verify column count
+ if len(result.Columns) != len(tt.expectedCols) {
+ t.Errorf("Expected %d columns, got %d", len(tt.expectedCols), len(result.Columns))
+ }
+
+ // Check column names
+ for i, expectedCol := range tt.expectedCols {
+ if i < len(result.Columns) && result.Columns[i] != expectedCol {
+ t.Logf("Expected column %d to be '%s', got '%s'", i, expectedCol, result.Columns[i])
+ // Don't fail on column name formatting differences, just log
+ }
+ }
+
+ // Validate first row
+ if tt.validateFirst != nil {
+ firstRow := result.Rows[0]
+ stringRow := make([]string, len(firstRow))
+ for i, val := range firstRow {
+ stringRow[i] = val.ToString()
+ }
+ tt.validateFirst(t, stringRow)
+ }
+
+ // Log results for debugging
+ t.Logf("Query: %s", tt.query)
+ t.Logf("Columns: %v", result.Columns)
+ for i, row := range result.Rows {
+ values := make([]string, len(row))
+ for j, val := range row {
+ values[j] = val.ToString()
+ }
+ t.Logf("Row %d: %v", i, values)
+ }
+ })
+ }
+}
+
+// TestSQLEngine_StringConcatenationBugReproduction tests the exact user query that was failing
+func TestSQLEngine_StringConcatenationBugReproduction(t *testing.T) {
+ engine := NewTestSQLEngine()
+
+ // This is the EXACT query from the user that was showing incorrect results
+ query := "SELECT UPPER(status), id*2, 'test' || action || 'xxx' || action || ' ~~~ ' || status FROM user_events LIMIT 2"
+
+ result, err := engine.ExecuteSQL(context.Background(), query)
+ if err != nil {
+ t.Fatalf("Query failed: %v", err)
+ }
+ if result.Error != nil {
+ t.Fatalf("Query returned error: %v", result.Error)
+ }
+
+ // Key assertions that would fail with the original bug:
+
+ // 1. Must return rows
+ if len(result.Rows) != 2 {
+ t.Errorf("Expected 2 rows, got %d", len(result.Rows))
+ }
+
+ // 2. Must have 3 columns
+ expectedColumns := 3
+ if len(result.Columns) != expectedColumns {
+ t.Errorf("Expected %d columns, got %d", expectedColumns, len(result.Columns))
+ }
+
+ // 3. Verify the complex concatenation works correctly
+ if len(result.Rows) >= 1 {
+ firstRow := result.Rows[0]
+
+ // Column 0: UPPER(status) should be "ACTIVE"
+ upperStatus := firstRow[0].ToString()
+ if upperStatus != "ACTIVE" {
+ t.Errorf("Expected UPPER(status)='ACTIVE', got '%s'", upperStatus)
+ }
+
+ // Column 1: id*2 should be calculated correctly
+ idTimes2 := firstRow[1].ToString()
+ if idTimes2 != "164920" { // id=82460 * 2
+ t.Errorf("Expected id*2=164920, got '%s'", idTimes2)
+ }
+
+ // Column 2: Complex concatenation should include all parts
+ concatenated := firstRow[2].ToString()
+
+ // Should be: "test" + "login" + "xxx" + "login" + " ~~~ " + "active" = "testloginxxxlogin ~~~ active"
+ expected := "testloginxxxlogin ~~~ active"
+ if concatenated != expected {
+ t.Errorf("String concatenation failed. Expected '%s', got '%s'", expected, concatenated)
+ }
+
+ // CRITICAL: Must not be the buggy result like "viewviewpending"
+ if concatenated == "loginloginactive" || concatenated == "viewviewpending" || concatenated == "clickclickfailed" {
+ t.Errorf("CRITICAL BUG: String concatenation returned buggy result '%s' - string literals are being lost!", concatenated)
+ }
+ }
+
+ t.Logf("✅ SUCCESS: Complex string concatenation works correctly!")
+ t.Logf("Query: %s", query)
+
+ for i, row := range result.Rows {
+ values := make([]string, len(row))
+ for j, val := range row {
+ values[j] = val.ToString()
+ }
+ t.Logf("Row %d: %v", i, values)
+ }
+}