From c8217cf2b2695e013d89a548e7733059b2404031 Mon Sep 17 00:00:00 2001 From: Aris Tzoumas Date: Fri, 16 Feb 2024 11:16:01 +0200 Subject: [PATCH] fixup! feat: sqlconnect library --- .github/workflows/test.yaml | 2 +- Makefile | 35 +++------- sqlconnect/def_query_test.go | 64 +++++++++++++++++++ sqlconnect/internal/base/db.go | 2 +- sqlconnect/internal/base/dialect_test.go | 30 +++++++++ sqlconnect/internal/base/schemaadmin.go | 9 ++- sqlconnect/internal/bigquery/db.go | 22 ++++++- sqlconnect/internal/bigquery/dialect_test.go | 30 +++++++++ .../internal/bigquery/integration_test.go | 18 ++++++ sqlconnect/internal/bigquery/schemaadmin.go | 57 +++++++++++++++++ sqlconnect/internal/databricks/config.go | 1 + sqlconnect/internal/databricks/db.go | 3 + .../internal/databricks/dialect_test.go | 30 +++++++++ .../internal/databricks/integration_test.go | 30 +++++++++ .../db_integration_test_scenario.go | 10 +-- sqlconnect/internal/mysql/dialect_test.go | 30 +++++++++ sqlconnect/internal/mysql/integration_test.go | 3 +- .../internal/postgres/integration_test.go | 3 +- sqlconnect/internal/redshift/db.go | 5 +- .../internal/redshift/integration_test.go | 19 ++++++ sqlconnect/internal/snowflake/db.go | 2 +- sqlconnect/internal/snowflake/dialect_test.go | 30 +++++++++ .../internal/snowflake/integration_test.go | 19 ++++++ sqlconnect/internal/trino/db.go | 2 +- sqlconnect/internal/trino/integration_test.go | 19 ++++++ sqlconnect/internal/util/validatehost.go | 1 + sqlconnect/internal/util/validatehost_test.go | 26 ++++++++ sqlconnect/ref_relation_test.go | 60 +++++++++++++++++ sqlconnect/ref_schema_test.go | 16 +++++ 29 files changed, 537 insertions(+), 41 deletions(-) create mode 100644 sqlconnect/def_query_test.go create mode 100644 sqlconnect/internal/base/dialect_test.go create mode 100644 sqlconnect/internal/bigquery/dialect_test.go create mode 100644 sqlconnect/internal/bigquery/integration_test.go create mode 100644 sqlconnect/internal/bigquery/schemaadmin.go create mode 100644 sqlconnect/internal/databricks/dialect_test.go create mode 100644 sqlconnect/internal/databricks/integration_test.go create mode 100644 sqlconnect/internal/mysql/dialect_test.go create mode 100644 sqlconnect/internal/redshift/integration_test.go create mode 100644 sqlconnect/internal/snowflake/dialect_test.go create mode 100644 sqlconnect/internal/snowflake/integration_test.go create mode 100644 sqlconnect/internal/trino/integration_test.go create mode 100644 sqlconnect/internal/util/validatehost_test.go create mode 100644 sqlconnect/ref_relation_test.go create mode 100644 sqlconnect/ref_schema_test.go diff --git a/.github/workflows/test.yaml b/.github/workflows/test.yaml index c1de80c..6236071 100644 --- a/.github/workflows/test.yaml +++ b/.github/workflows/test.yaml @@ -82,7 +82,7 @@ jobs: run: | go install github.com/wadey/gocovmerge@latest gocovmerge */profile.out > profile.out - - uses: codecov/codecov-action@v3 + - uses: codecov/codecov-action@v4 with: fail_ci_if_error: true files: ./profile.out diff --git a/Makefile b/Makefile index 19e20bf..7b74e26 100644 --- a/Makefile +++ b/Makefile @@ -1,4 +1,4 @@ -.PHONY: help default test test-run test-teardown generate lint fmt +.PHONY: help default test test-run generate lint fmt GO=go LDFLAGS?=-s -w @@ -9,47 +9,30 @@ default: lint generate: install-tools $(GO) generate ./... -test: install-tools test-run test-teardown +test: install-tools test-run test-run: ## Run all unit tests ifeq ($(filter 1,$(debug) $(RUNNER_DEBUG)),) - $(eval TEST_CMD = SLOW=0 gotestsum --format pkgname-and-test-fails --) - $(eval TEST_OPTIONS = -p=1 -v -failfast -shuffle=on -coverprofile=profile.out -covermode=count -coverpkg=./... -vet=all --timeout=15m) + $(eval TEST_CMD = gotestsum --format pkgname-and-test-fails --) + $(eval TEST_OPTIONS = -p=1 -v -failfast -shuffle=on -coverprofile=profile.out -covermode=atomic -coverpkg=./... -vet=all --timeout=30m) else $(eval TEST_CMD = SLOW=0 go test) - $(eval TEST_OPTIONS = -p=1 -v -failfast -shuffle=on -coverprofile=profile.out -covermode=count -coverpkg=./... -vet=all --timeout=15m) + $(eval TEST_OPTIONS = -p=1 -v -failfast -shuffle=on -coverprofile=profile.out -covermode=atomic -coverpkg=./... -vet=all --timeout=30m) endif ifdef package ifdef exclude $(eval FILES = `go list ./$(package)/... | egrep -iv '$(exclude)'`) - $(TEST_CMD) -count=1 $(TEST_OPTIONS) $(FILES) && touch $(TESTFILE) || true + $(TEST_CMD) -count=1 $(TEST_OPTIONS) $(FILES) && touch $(TESTFILE) else - $(TEST_CMD) $(TEST_OPTIONS) ./$(package)/... && touch $(TESTFILE) || true + $(TEST_CMD) $(TEST_OPTIONS) ./$(package)/... && touch $(TESTFILE) endif else ifdef exclude $(eval FILES = `go list ./... | egrep -iv '$(exclude)'`) - $(TEST_CMD) -count=1 $(TEST_OPTIONS) $(FILES) && touch $(TESTFILE) || true + $(TEST_CMD) -count=1 $(TEST_OPTIONS) $(FILES) && touch $(TESTFILE) else - $(TEST_CMD) -count=1 $(TEST_OPTIONS) ./... && touch $(TESTFILE) || true + $(TEST_CMD) -count=1 $(TEST_OPTIONS) ./... && touch $(TESTFILE) endif -test-teardown: - @if [ -f "$(TESTFILE)" ]; then \ - echo "Tests passed, tearing down..." ;\ - rm -f $(TESTFILE) ;\ - echo "mode: atomic" > coverage.txt ;\ - find . -name "profile.out" | while read file; do grep -v 'mode: atomic' $${file} >> coverage.txt; rm -f $${file}; done ;\ - else \ - rm -f coverage.txt coverage.html ; find . -name "profile.out" | xargs rm -f ;\ - echo "Tests failed :-(" ;\ - exit 1 ;\ - fi - -coverage: - go tool cover -html=coverage.txt -o coverage.html - -test-with-coverage: test coverage - help: ## Show the available commands @grep -E '^[0-9a-zA-Z_-]+:.*?## .*$$' ./Makefile | sort | awk 'BEGIN {FS = ":.*?## "}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}' diff --git a/sqlconnect/def_query_test.go b/sqlconnect/def_query_test.go new file mode 100644 index 0000000..b4014d0 --- /dev/null +++ b/sqlconnect/def_query_test.go @@ -0,0 +1,64 @@ +package sqlconnect_test + +import ( + "fmt" + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestQueryDef(t *testing.T) { + t.Run("with columns", func(t *testing.T) { + table := sqlconnect.NewRelationRef("table") + q := sqlconnect.QueryDef{ + Table: &table, + Columns: []string{"col1", "col2"}, + Conditions: []*sqlconnect.QueryCondition{ + {Column: "col1", Operator: "=", Value: "'1'"}, + {Column: "col2", Operator: ">", Value: "2"}, + }, + OrderBy: &sqlconnect.QueryOrder{ + Column: "col1", + Order: "ASC", + }, + } + + sql := q.ToSQL(testDialect{}) + expected := `SELECT "col1","col2" FROM "table" WHERE "col1" = '1' AND "col2" > 2 ORDER BY "col1" ASC` + require.Equal(t, expected, sql, "query should be formatted correctly") + }) + + t.Run("without columns", func(t *testing.T) { + table := sqlconnect.NewRelationRef("table") + q := sqlconnect.QueryDef{ + Table: &table, + Conditions: []*sqlconnect.QueryCondition{ + {Column: "col1", Operator: "=", Value: "'1'"}, + {Column: "col2", Operator: ">", Value: "2"}, + }, + } + + sql := q.ToSQL(testDialect{}) + expected := `SELECT * FROM "table" WHERE "col1" = '1' AND "col2" > 2` + require.Equal(t, expected, sql, "query should be formatted correctly") + }) +} + +type testDialect struct{} + +func (d testDialect) FormatTableName(name string) string { + return name +} + +func (d testDialect) QuoteIdentifier(name string) string { + return fmt.Sprintf(`"%s"`, name) +} + +func (d testDialect) QuoteTable(relation sqlconnect.RelationRef) string { + if relation.Schema != "" { + return fmt.Sprintf(`"%s"."%s"`, relation.Schema, relation.Name) + } + return fmt.Sprintf(`"%s"`, relation.Name) +} diff --git a/sqlconnect/internal/base/db.go b/sqlconnect/internal/base/db.go index 04542d2..1e85337 100644 --- a/sqlconnect/internal/base/db.go +++ b/sqlconnect/internal/base/db.go @@ -26,7 +26,7 @@ func NewDB(db *sql.DB, rudderSchema string, opts ...Option) *DB { return "SELECT schema_name FROM information_schema.schemata", "schema_name" }, SchemaExists: func(schema string) string { - return fmt.Sprintf("SELECT EXISTS (SELECT schema_name FROM information_schema.schemata where schema_name = '%[1]s')", schema) + return fmt.Sprintf("SELECT schema_name FROM information_schema.schemata where schema_name = '%[1]s'", schema) }, DropSchema: func(schema string) string { return fmt.Sprintf("DROP SCHEMA %[1]s CASCADE", schema) }, CreateTestTable: func(table string) string { diff --git a/sqlconnect/internal/base/dialect_test.go b/sqlconnect/internal/base/dialect_test.go new file mode 100644 index 0000000..c9a97af --- /dev/null +++ b/sqlconnect/internal/base/dialect_test.go @@ -0,0 +1,30 @@ +package base + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestDialect(t *testing.T) { + var d dialect + t.Run("format table", func(t *testing.T) { + formatted := d.FormatTableName("TaBle") + require.Equal(t, "table", formatted, "table name should be lowercased") + }) + + t.Run("quote identifier", func(t *testing.T) { + quoted := d.QuoteIdentifier("column") + require.Equal(t, `"column"`, quoted, "column name should be quoted with double quotes") + }) + + t.Run("quote table", func(t *testing.T) { + quoted := d.QuoteTable(sqlconnect.NewRelationRef("table")) + require.Equal(t, `"table"`, quoted, "table name should be quoted with double quotes") + + quoted = d.QuoteTable(sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"))) + require.Equal(t, `"schema"."table"`, quoted, "schema and table name should be quoted with double quotes") + }) +} diff --git a/sqlconnect/internal/base/schemaadmin.go b/sqlconnect/internal/base/schemaadmin.go index 91432df..64c561d 100644 --- a/sqlconnect/internal/base/schemaadmin.go +++ b/sqlconnect/internal/base/schemaadmin.go @@ -68,10 +68,15 @@ func (db *DB) ListSchemas(ctx context.Context) ([]sqlconnect.SchemaRef, error) { // SchemaExists returns true if the schema exists func (db *DB) SchemaExists(ctx context.Context, schemaRef sqlconnect.SchemaRef) (bool, error) { - var exists bool - if err := db.QueryRowContext(ctx, db.sqlCommands.SchemaExists(schemaRef.Name)).Scan(&exists); err != nil { + rows, err := db.QueryContext(ctx, db.sqlCommands.SchemaExists(schemaRef.Name)) + if err != nil { return false, fmt.Errorf("querying schema exists: %w", err) } + defer func() { _ = rows.Close() }() + exists := rows.Next() + if err := rows.Err(); err != nil { + return false, fmt.Errorf("iterating schema exists: %w", err) + } return exists, nil } diff --git a/sqlconnect/internal/bigquery/db.go b/sqlconnect/internal/bigquery/db.go index 2d585b5..b9eb7bd 100644 --- a/sqlconnect/internal/bigquery/db.go +++ b/sqlconnect/internal/bigquery/db.go @@ -1,10 +1,12 @@ package bigquery import ( + "context" "database/sql" "encoding/json" "fmt" + "cloud.google.com/go/bigquery" "github.com/samber/lo" "google.golang.org/api/option" @@ -49,7 +51,7 @@ func NewDB(configJSON json.RawMessage) (*DB, error) { } } cmds.TableExists = func(schema, table string) string { - return fmt.Sprintf("SELECT EXISTS (SELECT table_name FROM `%[1]s`.INFORMATION_SCHEMA.TABLES WHERE table_name = '%[1]s'", schema, table) + return fmt.Sprintf("SELECT EXISTS (SELECT table_name FROM `%[1]s`.INFORMATION_SCHEMA.TABLES WHERE table_name = '%[2]s'", schema, table) } return cmds @@ -67,3 +69,21 @@ func init() { type DB struct { *base.DB } + +func (db *DB) WithBigqueryClient(ctx context.Context, f func(*bigquery.Client) error) error { + sqlconn, err := db.Conn(ctx) + if err != nil { + return err + } + defer func() { _ = sqlconn.Close() }() + return sqlconn.Raw(func(driverConn any) error { + if c, ok := driverConn.(bqclient); ok { + return f(c.BigqueryClient()) + } + return fmt.Errorf("invalid driver connection") + }) +} + +type bqclient interface { + BigqueryClient() *bigquery.Client +} diff --git a/sqlconnect/internal/bigquery/dialect_test.go b/sqlconnect/internal/bigquery/dialect_test.go new file mode 100644 index 0000000..3e245e9 --- /dev/null +++ b/sqlconnect/internal/bigquery/dialect_test.go @@ -0,0 +1,30 @@ +package bigquery + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestDialect(t *testing.T) { + var d dialect + t.Run("format table", func(t *testing.T) { + formatted := d.FormatTableName("TaBle") + require.Equal(t, "table", formatted, "table name should be lowercased") + }) + + t.Run("quote identifier", func(t *testing.T) { + quoted := d.QuoteIdentifier("column") + require.Equal(t, "`column`", quoted, "column name should be quoted with backticks") + }) + + t.Run("quote table", func(t *testing.T) { + quoted := d.QuoteTable(sqlconnect.NewRelationRef("table")) + require.Equal(t, "`table`", quoted, "table name should be quoted with backticks") + + quoted = d.QuoteTable(sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"))) + require.Equal(t, "`schema.table`", quoted, "schema and table name should be quoted with backticks") + }) +} diff --git a/sqlconnect/internal/bigquery/integration_test.go b/sqlconnect/internal/bigquery/integration_test.go new file mode 100644 index 0000000..5ec2ada --- /dev/null +++ b/sqlconnect/internal/bigquery/integration_test.go @@ -0,0 +1,18 @@ +package bigquery_test + +import ( + "os" + "strings" + "testing" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/bigquery" + integrationtest "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/integration_test" +) + +func TestBigqueryDB(t *testing.T) { + configJSON, ok := os.LookupEnv("BIGQUERY_TEST_ENVIRONMENT_CREDENTIALS") + if !ok { + t.Skip("skipping bigquery integration test due to lack of a test environment") + } + integrationtest.TestDatabaseScenarios(t, bigquery.DatabaseType, []byte(configJSON), strings.ToLower) +} diff --git a/sqlconnect/internal/bigquery/schemaadmin.go b/sqlconnect/internal/bigquery/schemaadmin.go new file mode 100644 index 0000000..b10b2b8 --- /dev/null +++ b/sqlconnect/internal/bigquery/schemaadmin.go @@ -0,0 +1,57 @@ +package bigquery + +import ( + "context" + "errors" + + "cloud.google.com/go/bigquery" + "google.golang.org/api/googleapi" + "google.golang.org/api/iterator" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +// SchemaExists uses the bigquery client instead of [INFORMATION_SCHEMA.SCHEMATA] due to absence of a region qualifier +// https://cloud.google.com/bigquery/docs/information-schema-datasets-schemata#scope_and_syntax +func (db *DB) SchemaExists(ctx context.Context, schemaRef sqlconnect.SchemaRef) (bool, error) { + var exists bool + if err := db.WithBigqueryClient(ctx, func(c *bigquery.Client) error { + if _, err := c.Dataset(schemaRef.Name).Metadata(ctx); err != nil { + var e *googleapi.Error + if ok := errors.As(err, &e); ok { + if e.Code == 404 { // not found + return nil + } + } + return err + } + exists = true + return nil + }); err != nil { + return false, err + } + return exists, nil +} + +// ListSchemas uses the bigquery client instead of [INFORMATION_SCHEMA.SCHEMATA] due to absence of a region qualifier +// https://cloud.google.com/bigquery/docs/information-schema-datasets-schemata#scope_and_syntax +func (db *DB) ListSchemas(ctx context.Context) ([]sqlconnect.SchemaRef, error) { + var schemas []sqlconnect.SchemaRef + if err := db.WithBigqueryClient(ctx, func(c *bigquery.Client) error { + datasets := c.Datasets(ctx) + for { + var dataset *bigquery.Dataset + dataset, err := datasets.Next() + if err != nil { + if err == iterator.Done { + return nil + } + return err + } + schemas = append(schemas, sqlconnect.SchemaRef{Name: dataset.DatasetID}) + } + }); err != nil { + return nil, err + } + return schemas, nil +} diff --git a/sqlconnect/internal/databricks/config.go b/sqlconnect/internal/databricks/config.go index ca7b869..2a61cb7 100644 --- a/sqlconnect/internal/databricks/config.go +++ b/sqlconnect/internal/databricks/config.go @@ -15,6 +15,7 @@ type Config struct { RetryAttempts int `json:"retryAttempts"` MinRetryWaitTime time.Duration `json:"minRetryWaitTime"` MaxRetryWaitTime time.Duration `json:"maxRetryWaitTime"` + MaxConnIdleTime time.Duration `json:"maxConnIdleTime"` // RudderSchema is used to override the default rudder schema name during tests RudderSchema string `json:"rudderSchema"` diff --git a/sqlconnect/internal/databricks/db.go b/sqlconnect/internal/databricks/db.go index 374673c..1219c0c 100644 --- a/sqlconnect/internal/databricks/db.go +++ b/sqlconnect/internal/databricks/db.go @@ -3,6 +3,7 @@ package databricks import ( "database/sql" "encoding/json" + "fmt" databricks "github.com/databricks/databricks-sql-go" "github.com/samber/lo" @@ -45,6 +46,7 @@ func NewDB(configJson json.RawMessage) (*DB, error) { if err != nil { return nil, err } + db.SetConnMaxIdleTime(config.MaxConnIdleTime) return &DB{ DB: base.NewDB( @@ -55,6 +57,7 @@ func NewDB(configJson json.RawMessage) (*DB, error) { base.WithJsonRowMapper(jsonRowMapper), base.WithSQLCommandsOverride(func(cmds base.SQLCommands) base.SQLCommands { cmds.ListSchemas = func() (string, string) { return "SHOW SCHEMAS", "schema_name" } + cmds.SchemaExists = func(schema string) string { return fmt.Sprintf(`SHOW SCHEMAS LIKE '%s'`, schema) } cmds.ListTables = func(schema string) []lo.Tuple2[string, string] { return []lo.Tuple2[string, string]{ {A: "SHOW TABLES IN " + schema, B: "tableName"}, diff --git a/sqlconnect/internal/databricks/dialect_test.go b/sqlconnect/internal/databricks/dialect_test.go new file mode 100644 index 0000000..33f9866 --- /dev/null +++ b/sqlconnect/internal/databricks/dialect_test.go @@ -0,0 +1,30 @@ +package databricks + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestDialect(t *testing.T) { + var d dialect + t.Run("format table", func(t *testing.T) { + formatted := d.FormatTableName("TaBle") + require.Equal(t, "table", formatted, "table name should be lowercased") + }) + + t.Run("quote identifier", func(t *testing.T) { + quoted := d.QuoteIdentifier("column") + require.Equal(t, "`column`", quoted, "column name should be quoted with backticks") + }) + + t.Run("quote table", func(t *testing.T) { + quoted := d.QuoteTable(sqlconnect.NewRelationRef("table")) + require.Equal(t, "`table`", quoted, "table name should be quoted with backticks") + + quoted = d.QuoteTable(sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"))) + require.Equal(t, "`schema`.`table`", quoted, "schema and table name should be quoted with backticks") + }) +} diff --git a/sqlconnect/internal/databricks/integration_test.go b/sqlconnect/internal/databricks/integration_test.go new file mode 100644 index 0000000..317e255 --- /dev/null +++ b/sqlconnect/internal/databricks/integration_test.go @@ -0,0 +1,30 @@ +package databricks_test + +import ( + "os" + "strings" + "testing" + "time" + + "github.com/stretchr/testify/require" + "github.com/tidwall/sjson" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/databricks" + integrationtest "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/integration_test" +) + +func TestDatabricksDB(t *testing.T) { + configJSON, ok := os.LookupEnv("DATABRICKS_TEST_ENVIRONMENT_CREDENTIALS") + if !ok { + t.Skip("skipping databricks integration test due to lack of a test environment") + } + + configJSON, err := sjson.Set(configJSON, "retryAttempts", 1) + require.NoError(t, err, "failed to set retryAttempts") + configJSON, err = sjson.Set(configJSON, "minRetryWaitTime", time.Second) + require.NoError(t, err, "failed to set minRetryWaitTime") + configJSON, err = sjson.Set(configJSON, "maxRetryWaitTime", time.Minute) + require.NoError(t, err, "failed to set maxRetryWaitTime") + + integrationtest.TestDatabaseScenarios(t, databricks.DatabaseType, []byte(configJSON), strings.ToLower) +} diff --git a/sqlconnect/internal/integration_test/db_integration_test_scenario.go b/sqlconnect/internal/integration_test/db_integration_test_scenario.go index 9a1469d..69a6c3e 100644 --- a/sqlconnect/internal/integration_test/db_integration_test_scenario.go +++ b/sqlconnect/internal/integration_test/db_integration_test_scenario.go @@ -14,8 +14,8 @@ import ( "github.com/rudderlabs/sqlconnect-go/sqlconnect" ) -func TestDatabaseScenarios(t *testing.T, warehouse string, configJSON json.RawMessage) { - schema := sqlconnect.SchemaRef{Name: GenerateTestSchema()} +func TestDatabaseScenarios(t *testing.T, warehouse string, configJSON json.RawMessage, formatfn func(string) string) { + schema := sqlconnect.SchemaRef{Name: GenerateTestSchema(formatfn)} configJSON, err := sjson.SetBytes(configJSON, "rudderSchema", schema.Name) require.NoError(t, err, "it should be able to set the rudder schema") db, err := sqlconnect.NewDB(warehouse, configJSON) @@ -97,7 +97,7 @@ func TestDatabaseScenarios(t *testing.T, warehouse string, configJSON json.RawMe }) t.Run("normal operation", func(t *testing.T) { - otherSchema := sqlconnect.SchemaRef{Name: GenerateTestSchema()} + otherSchema := sqlconnect.SchemaRef{Name: GenerateTestSchema(formatfn)} err := db.CreateSchema(ctx, otherSchema) require.NoError(t, err, "it should be able to create a schema") err = db.DropSchema(ctx, otherSchema) @@ -112,6 +112,6 @@ func TestDatabaseScenarios(t *testing.T, warehouse string, configJSON json.RawMe }) } -func GenerateTestSchema() string { - return fmt.Sprintf("tsqlcon_%s_%d", rand.String(12), time.Now().Unix()) +func GenerateTestSchema(formatfn func(string) string) string { + return formatfn(fmt.Sprintf("tsqlcon_%s_%d", rand.String(12), time.Now().Unix())) } diff --git a/sqlconnect/internal/mysql/dialect_test.go b/sqlconnect/internal/mysql/dialect_test.go new file mode 100644 index 0000000..28ffc84 --- /dev/null +++ b/sqlconnect/internal/mysql/dialect_test.go @@ -0,0 +1,30 @@ +package mysql + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestDialect(t *testing.T) { + var d dialect + t.Run("format table", func(t *testing.T) { + formatted := d.FormatTableName("TaBle") + require.Equal(t, "table", formatted, "table name should be lowercased") + }) + + t.Run("quote identifier", func(t *testing.T) { + quoted := d.QuoteIdentifier("column") + require.Equal(t, "`column`", quoted, "column name should be quoted with backticks") + }) + + t.Run("quote table", func(t *testing.T) { + quoted := d.QuoteTable(sqlconnect.NewRelationRef("table")) + require.Equal(t, "`table`", quoted, "table name should be quoted with backticks") + + quoted = d.QuoteTable(sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"))) + require.Equal(t, "`schema`.`table`", quoted, "schema and table name should be quoted with backticks") + }) +} diff --git a/sqlconnect/internal/mysql/integration_test.go b/sqlconnect/internal/mysql/integration_test.go index 770987e..066d12f 100644 --- a/sqlconnect/internal/mysql/integration_test.go +++ b/sqlconnect/internal/mysql/integration_test.go @@ -3,6 +3,7 @@ package mysql_test import ( "encoding/json" "strconv" + "strings" "testing" "github.com/ory/dockertest/v3" @@ -33,5 +34,5 @@ func TestMysqlDB(t *testing.T) { configJSON, err := json.Marshal(config) require.NoError(t, err, "it should be able to marshal config to json") - integrationtest.TestDatabaseScenarios(t, mysql.DatabaseType, configJSON) + integrationtest.TestDatabaseScenarios(t, mysql.DatabaseType, configJSON, strings.ToLower) } diff --git a/sqlconnect/internal/postgres/integration_test.go b/sqlconnect/internal/postgres/integration_test.go index e916510..75ce8ab 100644 --- a/sqlconnect/internal/postgres/integration_test.go +++ b/sqlconnect/internal/postgres/integration_test.go @@ -3,6 +3,7 @@ package postgres_test import ( "encoding/json" "strconv" + "strings" "testing" "github.com/ory/dockertest/v3" @@ -33,5 +34,5 @@ func TestPostgresDB(t *testing.T) { configJSON, err := json.Marshal(config) require.NoError(t, err, "it should be able to marshal config to json") - integrationtest.TestDatabaseScenarios(t, postgres.DatabaseType, configJSON) + integrationtest.TestDatabaseScenarios(t, postgres.DatabaseType, configJSON, strings.ToLower) } diff --git a/sqlconnect/internal/redshift/db.go b/sqlconnect/internal/redshift/db.go index 042284f..1bf3ab7 100644 --- a/sqlconnect/internal/redshift/db.go +++ b/sqlconnect/internal/redshift/db.go @@ -35,8 +35,11 @@ func NewDB(credentialsJSON json.RawMessage) (*DB, error) { lo.Ternary(config.RudderSchema != "", config.RudderSchema, defaultRudderSchema), base.WithColumnTypeMappings(columnTypeMappings), base.WithSQLCommandsOverride(func(cmds base.SQLCommands) base.SQLCommands { + cmds.ListSchemas = func() (string, string) { + return "SELECT schema_name FROM svv_redshift_schemas", "schema_name" + } cmds.SchemaExists = func(schema string) string { - return fmt.Sprintf("SELECT has_schema_privilege((SELECT current_user), '%[1]s', 'usage')", schema) + return fmt.Sprintf("SELECT schema_name FROM svv_redshift_schemas WHERE schema_name = '%[1]s'", schema) } return cmds }), diff --git a/sqlconnect/internal/redshift/integration_test.go b/sqlconnect/internal/redshift/integration_test.go new file mode 100644 index 0000000..3ad3ea6 --- /dev/null +++ b/sqlconnect/internal/redshift/integration_test.go @@ -0,0 +1,19 @@ +package redshift_test + +import ( + "os" + "strings" + "testing" + + integrationtest "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/integration_test" + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/redshift" +) + +func TestRedshiftDB(t *testing.T) { + configJSON, ok := os.LookupEnv("REDSHIFT_TEST_ENVIRONMENT_CREDENTIALS") + if !ok { + t.Skip("skipping redshift integration test due to lack of a test environment") + } + + integrationtest.TestDatabaseScenarios(t, redshift.DatabaseType, []byte(configJSON), strings.ToLower) +} diff --git a/sqlconnect/internal/snowflake/db.go b/sqlconnect/internal/snowflake/db.go index 6662e8b..8bc26a8 100644 --- a/sqlconnect/internal/snowflake/db.go +++ b/sqlconnect/internal/snowflake/db.go @@ -41,10 +41,10 @@ func NewDB(configJSON json.RawMessage) (*DB, error) { base.WithColumnTypeMappings(columnTypeMappings), base.WithJsonRowMapper(jsonRowMapper), base.WithSQLCommandsOverride(func(cmds base.SQLCommands) base.SQLCommands { + cmds.ListSchemas = func() (string, string) { return "SHOW TERSE SCHEMAS", "name" } cmds.SchemaExists = func(schema string) string { return fmt.Sprintf("SHOW TERSE SCHEMAS LIKE '%[1]s'", schema) } - cmds.ListSchemas = func() (string, string) { return "SHOW TERSE SCHEMAS", "schema_name" } cmds.ListTables = func(schema string) []lo.Tuple2[string, string] { return []lo.Tuple2[string, string]{ {A: fmt.Sprintf("SHOW TERSE TABLES IN SCHEMA %[1]s", schema), B: "name"}, diff --git a/sqlconnect/internal/snowflake/dialect_test.go b/sqlconnect/internal/snowflake/dialect_test.go new file mode 100644 index 0000000..ec66800 --- /dev/null +++ b/sqlconnect/internal/snowflake/dialect_test.go @@ -0,0 +1,30 @@ +package snowflake + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestDialect(t *testing.T) { + var d dialect + t.Run("format table", func(t *testing.T) { + formatted := d.FormatTableName("TaBle") + require.Equal(t, "TABLE", formatted, "table name should be uppercased") + }) + + t.Run("quote identifier", func(t *testing.T) { + quoted := d.QuoteIdentifier("column") + require.Equal(t, `"column"`, quoted, "column name should be quoted with double quotes") + }) + + t.Run("quote table", func(t *testing.T) { + quoted := d.QuoteTable(sqlconnect.NewRelationRef("table")) + require.Equal(t, `"table"`, quoted, "table name should be quoted with double quotes") + + quoted = d.QuoteTable(sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"))) + require.Equal(t, `"schema"."table"`, quoted, "schema and table name should be quoted with double quotes") + }) +} diff --git a/sqlconnect/internal/snowflake/integration_test.go b/sqlconnect/internal/snowflake/integration_test.go new file mode 100644 index 0000000..8c45744 --- /dev/null +++ b/sqlconnect/internal/snowflake/integration_test.go @@ -0,0 +1,19 @@ +package snowflake_test + +import ( + "os" + "strings" + "testing" + + integrationtest "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/integration_test" + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/snowflake" +) + +func TestSnowflakeDB(t *testing.T) { + configJSON, ok := os.LookupEnv("SNOWFLAKE_TEST_ENVIRONMENT_CREDENTIALS") + if !ok { + t.Skip("skipping snowflake integration test due to lack of a test environment") + } + + integrationtest.TestDatabaseScenarios(t, snowflake.DatabaseType, []byte(configJSON), strings.ToUpper) +} diff --git a/sqlconnect/internal/trino/db.go b/sqlconnect/internal/trino/db.go index 1b72c31..9cdf78a 100644 --- a/sqlconnect/internal/trino/db.go +++ b/sqlconnect/internal/trino/db.go @@ -38,7 +38,7 @@ func NewDB(configJSON json.RawMessage) (*DB, error) { base.WithSQLCommandsOverride(func(cmds base.SQLCommands) base.SQLCommands { cmds.ListTables = func(schema string) []lo.Tuple2[string, string] { return []lo.Tuple2[string, string]{ - {A: fmt.Sprintf("SHOW TABLES FROM %s", schema), B: "tableName"}, + {A: fmt.Sprintf("SHOW TABLES FROM %[1]s", schema), B: "tableName"}, } } cmds.ListTablesWithPrefix = func(schema, prefix string) []lo.Tuple2[string, string] { diff --git a/sqlconnect/internal/trino/integration_test.go b/sqlconnect/internal/trino/integration_test.go new file mode 100644 index 0000000..69c65b8 --- /dev/null +++ b/sqlconnect/internal/trino/integration_test.go @@ -0,0 +1,19 @@ +package trino_test + +import ( + "os" + "strings" + "testing" + + integrationtest "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/integration_test" + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/trino" +) + +func TestTrinoDB(t *testing.T) { + configJSON, ok := os.LookupEnv("TRINO_TEST_ENVIRONMENT_CREDENTIALS") + if !ok { + t.Skip("skipping trino integration test due to lack of a test environment") + } + + integrationtest.TestDatabaseScenarios(t, trino.DatabaseType, []byte(configJSON), strings.ToLower) +} diff --git a/sqlconnect/internal/util/validatehost.go b/sqlconnect/internal/util/validatehost.go index d6f79e2..240759c 100644 --- a/sqlconnect/internal/util/validatehost.go +++ b/sqlconnect/internal/util/validatehost.go @@ -5,6 +5,7 @@ import ( "net" ) +// ValidateHost checks if the hostname is resolvable and that it doesn't correspond to localhost. func ValidateHost(hostname string) error { addrs, err := net.LookupHost(hostname) if err != nil { diff --git a/sqlconnect/internal/util/validatehost_test.go b/sqlconnect/internal/util/validatehost_test.go new file mode 100644 index 0000000..fc2b38b --- /dev/null +++ b/sqlconnect/internal/util/validatehost_test.go @@ -0,0 +1,26 @@ +package util_test + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect/internal/util" +) + +func TestValidateHost(t *testing.T) { + t.Run("valid host", func(t *testing.T) { + err := util.ValidateHost("github.com") + require.NoError(t, err) + }) + + t.Run("invalid host", func(t *testing.T) { + err := util.ValidateHost("!@#$.$%^") + require.Error(t, err) + }) + + t.Run("localhost", func(t *testing.T) { + err := util.ValidateHost("localhost") + require.Error(t, err) + }) +} diff --git a/sqlconnect/ref_relation_test.go b/sqlconnect/ref_relation_test.go new file mode 100644 index 0000000..592c04c --- /dev/null +++ b/sqlconnect/ref_relation_test.go @@ -0,0 +1,60 @@ +package sqlconnect_test + +import ( + "encoding/json" + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestRelationRef(t *testing.T) { + t.Run("name", func(t *testing.T) { + ref := sqlconnect.NewRelationRef("table") + require.Equal(t, sqlconnect.RelationRef{Name: "table", Type: "table"}, ref) + require.Equal(t, "table", ref.String()) + + refJSON, _ := json.Marshal(ref) + var ref1 sqlconnect.RelationRef + err := ref1.UnmarshalJSON(refJSON) + require.NoError(t, err) + require.Equal(t, ref, ref1) + }) + + t.Run("name and schema", func(t *testing.T) { + ref := sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema")) + require.Equal(t, sqlconnect.RelationRef{Name: "table", Schema: "schema", Type: "table"}, ref) + require.Equal(t, "schema.table", ref.String()) + + refJSON, _ := json.Marshal(ref) + var ref1 sqlconnect.RelationRef + err := ref1.UnmarshalJSON(refJSON) + require.NoError(t, err) + require.Equal(t, ref, ref1) + }) + + t.Run("name and schema and catalog", func(t *testing.T) { + ref := sqlconnect.NewRelationRef("table", sqlconnect.WithSchema("schema"), sqlconnect.WithCatalog("catalog")) + require.Equal(t, sqlconnect.RelationRef{Name: "table", Schema: "schema", Catalog: "catalog", Type: "table"}, ref) + require.Equal(t, "catalog.schema.table", ref.String()) + + refJSON, _ := json.Marshal(ref) + var ref1 sqlconnect.RelationRef + err := ref1.UnmarshalJSON(refJSON) + require.NoError(t, err) + require.Equal(t, ref, ref1) + }) + + t.Run("view instead of table", func(t *testing.T) { + ref := sqlconnect.NewRelationRef("view", sqlconnect.WithRelationType(sqlconnect.ViewRelation)) + require.Equal(t, sqlconnect.RelationRef{Name: "view", Type: "view"}, ref) + }) + + t.Run("unmarshal without a type", func(t *testing.T) { + var ref sqlconnect.RelationRef + err := ref.UnmarshalJSON([]byte(`{"name":"table"}`)) + require.NoError(t, err) + require.Equal(t, sqlconnect.NewRelationRef("table"), ref) + }) +} diff --git a/sqlconnect/ref_schema_test.go b/sqlconnect/ref_schema_test.go new file mode 100644 index 0000000..19df4e0 --- /dev/null +++ b/sqlconnect/ref_schema_test.go @@ -0,0 +1,16 @@ +package sqlconnect_test + +import ( + "testing" + + "github.com/stretchr/testify/require" + + "github.com/rudderlabs/sqlconnect-go/sqlconnect" +) + +func TestSchemaRef(t *testing.T) { + t.Run("string", func(t *testing.T) { + s := sqlconnect.SchemaRef{Name: "schema"} + require.Equal(t, "schema", s.String(), "schema name should be returned") + }) +}