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.
143 lines
4.6 KiB
143 lines
4.6 KiB
package engine
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
)
|
|
|
|
// TestSQLEngine_ArithmeticOnlyQueryExecution tests the specific fix for queries
|
|
// that contain ONLY arithmetic expressions (no base columns) in the SELECT clause.
|
|
// This was the root issue reported where such queries returned empty values.
|
|
func TestSQLEngine_ArithmeticOnlyQueryExecution(t *testing.T) {
|
|
engine := NewTestSQLEngine()
|
|
|
|
// Test the core functionality: arithmetic-only queries should return data
|
|
tests := []struct {
|
|
name string
|
|
query string
|
|
expectedCols []string
|
|
mustNotBeEmpty bool
|
|
}{
|
|
{
|
|
name: "Basic arithmetic only query",
|
|
query: "SELECT id+user_id, id*2 FROM user_events LIMIT 3",
|
|
expectedCols: []string{"id+user_id", "id*2"},
|
|
mustNotBeEmpty: true,
|
|
},
|
|
{
|
|
name: "With LIMIT and OFFSET - original user issue",
|
|
query: "SELECT id+user_id, id*2 FROM user_events LIMIT 2 OFFSET 1",
|
|
expectedCols: []string{"id+user_id", "id*2"},
|
|
mustNotBeEmpty: true,
|
|
},
|
|
{
|
|
name: "Multiple arithmetic expressions",
|
|
query: "SELECT user_id+100, id-1000 FROM user_events LIMIT 1",
|
|
expectedCols: []string{"user_id+100", "id-1000"},
|
|
mustNotBeEmpty: true,
|
|
},
|
|
}
|
|
|
|
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)
|
|
}
|
|
|
|
// CRITICAL: Verify we got results (the original bug would return empty)
|
|
if tt.mustNotBeEmpty && len(result.Rows) == 0 {
|
|
t.Fatal("CRITICAL BUG: Query returned no rows - arithmetic-only query fix failed!")
|
|
}
|
|
|
|
// Verify column count and names
|
|
if len(result.Columns) != len(tt.expectedCols) {
|
|
t.Errorf("Expected %d columns, got %d", len(tt.expectedCols), len(result.Columns))
|
|
}
|
|
|
|
// CRITICAL: Verify no empty/null values (the original bug symptom)
|
|
if len(result.Rows) > 0 {
|
|
firstRow := result.Rows[0]
|
|
for i, val := range firstRow {
|
|
if val.IsNull() {
|
|
t.Errorf("CRITICAL BUG: Column %d (%s) returned NULL", i, result.Columns[i])
|
|
}
|
|
if val.ToString() == "" {
|
|
t.Errorf("CRITICAL BUG: Column %d (%s) returned empty string", i, result.Columns[i])
|
|
}
|
|
}
|
|
}
|
|
|
|
// Log success
|
|
t.Logf("SUCCESS: %s returned %d rows with calculated values", tt.query, len(result.Rows))
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestSQLEngine_ArithmeticOnlyQueryBugReproduction tests that the original bug
|
|
// (returning empty values) would have failed before our fix
|
|
func TestSQLEngine_ArithmeticOnlyQueryBugReproduction(t *testing.T) {
|
|
engine := NewTestSQLEngine()
|
|
|
|
// This is the EXACT query from the user's bug report
|
|
query := "SELECT id+user_id, id*amount, id*2 FROM user_events LIMIT 10 OFFSET 5"
|
|
|
|
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 (bug would return 0 rows or empty results)
|
|
if len(result.Rows) == 0 {
|
|
t.Fatal("CRITICAL: Query returned no rows - the original bug is NOT fixed!")
|
|
}
|
|
|
|
// 2. Must have expected columns
|
|
expectedColumns := []string{"id+user_id", "id*amount", "id*2"}
|
|
if len(result.Columns) != len(expectedColumns) {
|
|
t.Errorf("Expected %d columns, got %d", len(expectedColumns), len(result.Columns))
|
|
}
|
|
|
|
// 3. Must have calculated values, not empty/null
|
|
for i, row := range result.Rows {
|
|
for j, val := range row {
|
|
if val.IsNull() {
|
|
t.Errorf("Row %d, Column %d (%s) is NULL - original bug not fixed!",
|
|
i, j, result.Columns[j])
|
|
}
|
|
if val.ToString() == "" {
|
|
t.Errorf("Row %d, Column %d (%s) is empty - original bug not fixed!",
|
|
i, j, result.Columns[j])
|
|
}
|
|
}
|
|
}
|
|
|
|
// 4. Verify specific calculations for the OFFSET 5 data
|
|
if len(result.Rows) > 0 {
|
|
firstRow := result.Rows[0]
|
|
// With OFFSET 5, first returned row should be 6th row: id=417224, user_id=7810
|
|
expectedSum := "425034" // 417224 + 7810
|
|
if firstRow[0].ToString() != expectedSum {
|
|
t.Errorf("OFFSET 5 calculation wrong: expected id+user_id=%s, got %s",
|
|
expectedSum, firstRow[0].ToString())
|
|
}
|
|
|
|
expectedDouble := "834448" // 417224 * 2
|
|
if firstRow[2].ToString() != expectedDouble {
|
|
t.Errorf("OFFSET 5 calculation wrong: expected id*2=%s, got %s",
|
|
expectedDouble, firstRow[2].ToString())
|
|
}
|
|
}
|
|
|
|
t.Logf("SUCCESS: Arithmetic-only query with OFFSET works correctly!")
|
|
t.Logf("Query: %s", query)
|
|
t.Logf("Returned %d rows with correct calculations", len(result.Rows))
|
|
}
|