You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							190 lines
						
					
					
						
							6.0 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							190 lines
						
					
					
						
							6.0 KiB
						
					
					
				| 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) | |
| 	} | |
| }
 |