diff --git a/bob.go b/bob.go index a010263..a8a4470 100644 --- a/bob.go +++ b/bob.go @@ -8,8 +8,10 @@ import ( // ErrEmptyTable is a common database/sql error if a table is empty or no rows is returned by the query. var ErrEmptyTable = errors.New("sql: no rows in result set") + // ErrEmptyTable is a common pgx error if a table is empty or no rows is returned by the query. var ErrEmptyTablePgx = errors.New("no rows in result set") + // ErrDialectNotSupported tells you whether the dialect is supported or not. var ErrDialectNotSupported = errors.New("provided database dialect is not supported") @@ -147,7 +149,7 @@ func Truncate(table string) TruncateBuilder { // Replace("age", 25). // PlaceholderFormat(bob.Question). // ToSql() -// +// // // Another example for PostgreSQL: // sql, args, err = bob. // Upsert("users", bob.PostgreSQL). @@ -157,7 +159,7 @@ func Truncate(table string) TruncateBuilder { // Replace("age", 40). // PlaceholderFormat(bob.Dollar). // ToSql() -// +// // // One more time, for MSSQL / SQL Server: // sql, args, err = bob. // Upsert("users", bob.MSSQL). @@ -169,4 +171,4 @@ func Truncate(table string) TruncateBuilder { // ToSql() func Upsert(table string, dialect int) UpsertBuilder { return BobStmtBuilder.Upsert(table, dialect) -} \ No newline at end of file +} diff --git a/drop.go b/drop.go index 969051d..9d08962 100644 --- a/drop.go +++ b/drop.go @@ -11,7 +11,7 @@ type DropBuilder builder.Builder type dropData struct { TableName string - IfExists bool + IfExists bool } func init() { @@ -45,9 +45,9 @@ func (d *dropData) ToSql() (sqlStr string, args []interface{}, err error) { if d.IfExists { sql.WriteString("IF EXISTS ") } - - sql.WriteString("\""+d.TableName+"\";") + + sql.WriteString("\"" + d.TableName + "\";") sqlStr = sql.String() return -} \ No newline at end of file +} diff --git a/drop_test.go b/drop_test.go index d4c7bdd..ec4785c 100644 --- a/drop_test.go +++ b/drop_test.go @@ -7,7 +7,7 @@ import ( ) func TestDrop(t *testing.T) { - t.Run("should be able to create regular drop query", func (t *testing.T) { + t.Run("should be able to create regular drop query", func(t *testing.T) { sql, _, err := bob.DropTable("users").ToSql() if err != nil { t.Error(err) @@ -37,4 +37,4 @@ func TestDrop(t *testing.T) { t.Error(err) } }) -} \ No newline at end of file +} diff --git a/has_test.go b/has_test.go index 611f999..e4a33a1 100644 --- a/has_test.go +++ b/has_test.go @@ -7,8 +7,6 @@ import ( "github.com/aldy505/bob" ) -// TODO - do more test - func TestHas(t *testing.T) { t.Run("should be able to create a hasTable query", func(t *testing.T) { sql, args, err := bob.HasTable("users").ToSql() diff --git a/rename.go b/rename.go index 38b59f1..396faa7 100644 --- a/rename.go +++ b/rename.go @@ -10,7 +10,7 @@ type RenameBuilder builder.Builder type renameData struct { From string - To string + To string } func init() { @@ -38,6 +38,6 @@ func (d *renameData) ToSql() (sqlStr string, args []interface{}, err error) { if len(d.From) == 0 || d.From == "" || len(d.To) == 0 || d.To == "" { err = errors.New("rename statement must specify a table") } - sqlStr = "RENAME TABLE \""+d.From+"\" TO \""+d.To+"\";" + sqlStr = "RENAME TABLE \"" + d.From + "\" TO \"" + d.To + "\";" return -} \ No newline at end of file +} diff --git a/rename_test.go b/rename_test.go index 05a2492..63e4f8d 100644 --- a/rename_test.go +++ b/rename_test.go @@ -7,7 +7,7 @@ import ( ) func TestRename(t *testing.T) { - t.Run("should be able to create rename query", func (t *testing.T) { + t.Run("should be able to create rename query", func(t *testing.T) { sql, _, err := bob.RenameTable("users", "teachers").ToSql() if err != nil { t.Error(err) @@ -25,4 +25,4 @@ func TestRename(t *testing.T) { t.Error(err) } }) -} \ No newline at end of file +} diff --git a/truncate.go b/truncate.go index a12b9e4..d3cc9e1 100644 --- a/truncate.go +++ b/truncate.go @@ -32,6 +32,6 @@ func (d *truncateData) ToSql() (sqlStr string, args []interface{}, err error) { if len(d.TableName) == 0 || d.TableName == "" { err = errors.New("truncate statement must specify a table") } - sqlStr = "TRUNCATE \""+d.TableName+"\";" + sqlStr = "TRUNCATE \"" + d.TableName + "\";" return -} \ No newline at end of file +} diff --git a/truncate_test.go b/truncate_test.go index 535bc5d..fa556cf 100644 --- a/truncate_test.go +++ b/truncate_test.go @@ -7,7 +7,7 @@ import ( ) func TestTruncate(t *testing.T) { - t.Run("should be able to create truncate query", func (t *testing.T) { + t.Run("should be able to create truncate query", func(t *testing.T) { sql, _, err := bob.Truncate("users").ToSql() if err != nil { t.Error(err) @@ -25,4 +25,4 @@ func TestTruncate(t *testing.T) { t.Error(err) } }) -} \ No newline at end of file +} diff --git a/upsert.go b/upsert.go index 43b3a43..2fb19f2 100644 --- a/upsert.go +++ b/upsert.go @@ -11,12 +11,12 @@ import ( type UpsertBuilder builder.Builder type upsertData struct { - Dialect int - Into string - Columns []string - Values [][]interface{} - Key []interface{} - Replace [][]interface{} + Dialect int + Into string + Columns []string + Values [][]interface{} + Key []interface{} + Replace [][]interface{} Placeholder string } @@ -60,7 +60,7 @@ func (u UpsertBuilder) Key(key ...interface{}) UpsertBuilder { } // Replace sets the column and value respectively for the data to be changed on -// a specific row. +// a specific row. func (u UpsertBuilder) Replace(column interface{}, value interface{}) UpsertBuilder { return builder.Append(u, "Replace", []interface{}{column, value}).(UpsertBuilder) } @@ -105,13 +105,13 @@ func (d *upsertData) ToSql() (sqlStr string, args []interface{}, err error) { err = errors.New("unique key and value must be provided for MS SQL") return } - - sql.WriteString("IF NOT EXISTS (SELECT * FROM \""+d.Into+"\" WHERE \""+d.Key[0].(string)+"\" = ?) ") + + sql.WriteString("IF NOT EXISTS (SELECT * FROM \"" + d.Into + "\" WHERE \"" + d.Key[0].(string) + "\" = ?) ") args = append(args, d.Key[1]) } sql.WriteString("INSERT INTO ") - sql.WriteString("\""+d.Into+"\"") + sql.WriteString("\"" + d.Into + "\"") sql.WriteString(" ") var columns []string @@ -134,7 +134,7 @@ func (d *upsertData) ToSql() (sqlStr string, args []interface{}, err error) { } values = append(values, "("+strings.Join(tempValues, ", ")+")") } - + sql.WriteString(strings.Join(values, ", ")) sql.WriteString(" ") @@ -147,47 +147,47 @@ func (d *upsertData) ToSql() (sqlStr string, args []interface{}, err error) { if d.Dialect == MySQL { // INSERT INTO table (col) VALUES (values) ON DUPLICATE KEY UPDATE col = value - + sql.WriteString("ON DUPLICATE KEY UPDATE ") sql.WriteString(strings.Join(replaces, ", ")) } else if d.Dialect == PostgreSQL || d.Dialect == SQLite { // INSERT INTO players (user_name, age) VALUES('steven', 32) ON CONFLICT(user_name) DO UPDATE SET age=excluded.age; - + if len(d.Key) == 0 { err = errors.New("unique key must be provided for PostgreSQL and SQLite") return } sql.WriteString("ON CONFLICT ") - sql.WriteString("(\""+d.Key[0].(string)+"\") ") + sql.WriteString("(\"" + d.Key[0].(string) + "\") ") sql.WriteString("DO UPDATE SET ") sql.WriteString(strings.Join(replaces, ", ")) } else if d.Dialect == MSSQL { // IF NOT EXISTS (SELECT * FROM dbo.Table1 WHERE ID = @ID) - // INSERT INTO dbo.Table1(ID, Name, ItemName, ItemCatName, ItemQty) - // VALUES(@ID, @Name, @ItemName, @ItemCatName, @ItemQty) - // ELSE - // UPDATE dbo.Table1 - // SET Name = @Name, - // ItemName = @ItemName, - // ItemCatName = @ItemCatName, - // ItemQty = @ItemQty - // WHERE ID = @ID + // INSERT INTO dbo.Table1(ID, Name, ItemName, ItemCatName, ItemQty) + // VALUES(@ID, @Name, @ItemName, @ItemCatName, @ItemQty) + // ELSE + // UPDATE dbo.Table1 + // SET Name = @Name, + // ItemName = @ItemName, + // ItemCatName = @ItemCatName, + // ItemQty = @ItemQty + // WHERE ID = @ID sql.WriteString("ELSE ") - sql.WriteString("UPDATE \""+d.Into+"\" SET ") + sql.WriteString("UPDATE \"" + d.Into + "\" SET ") sql.WriteString(strings.Join(replaces, ", ")) - sql.WriteString(" WHERE \""+d.Key[0].(string)+"\" = ?") + sql.WriteString(" WHERE \"" + d.Key[0].(string) + "\" = ?") args = append(args, d.Key[1]) } else { err = ErrDialectNotSupported return } - + sql.WriteString(";") sqlStr = ReplacePlaceholder(sql.String(), d.Placeholder) return -} \ No newline at end of file +} diff --git a/upsert_test.go b/upsert_test.go index 925b20f..029197e 100644 --- a/upsert_test.go +++ b/upsert_test.go @@ -7,101 +7,101 @@ import ( "github.com/aldy505/bob" ) -func TestUpsert(t *testing.T) { - t.Run("should be able to generate upsert query for mysql", func(t *testing.T) { - sql, args, err := bob. - Upsert("users", bob.MySQL). - Columns("name", "email"). - Values("John Doe", "john@doe.com"). - Replace("name", "John Does"). - ToSql() - if err != nil { - t.Error(err) - } +func TestUpsert_MySQL(t *testing.T) { + sql, args, err := bob. + Upsert("users", bob.MySQL). + Columns("name", "email"). + Values("John Doe", "john@doe.com"). + Replace("name", "John Does"). + ToSql() + if err != nil { + t.Error(err) + } - desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES (?, ?) ON DUPLICATE KEY UPDATE \"name\" = ?;" - desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} + desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES (?, ?) ON DUPLICATE KEY UPDATE \"name\" = ?;" + desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} - if sql != desiredSql { - t.Error("sql is not the same as result: ", sql) - } - if !reflect.DeepEqual(args, desiredArgs) { - t.Error("args is not the same as result: ", args) - } - }) + if sql != desiredSql { + t.Error("sql is not the same as result: ", sql) + } + if !reflect.DeepEqual(args, desiredArgs) { + t.Error("args is not the same as result: ", args) + } +} - t.Run("should be able to generate upsert query for postgres", func(t *testing.T) { - sql, args, err := bob. - Upsert("users", bob.PostgreSQL). - Columns("name", "email"). - Values("John Doe", "john@doe.com"). - Key("email"). - Replace("name", "John Does"). - PlaceholderFormat(bob.Dollar). - ToSql() - if err != nil { - t.Error(err) - } +func TestUpsert_PostgreSQL(t *testing.T) { + sql, args, err := bob. + Upsert("users", bob.PostgreSQL). + Columns("name", "email"). + Values("John Doe", "john@doe.com"). + Key("email"). + Replace("name", "John Does"). + PlaceholderFormat(bob.Dollar). + ToSql() + if err != nil { + t.Error(err) + } - desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES ($1, $2) ON CONFLICT (\"email\") DO UPDATE SET \"name\" = $3;" - desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} + desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES ($1, $2) ON CONFLICT (\"email\") DO UPDATE SET \"name\" = $3;" + desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} - if sql != desiredSql { - t.Error("sql is not the same as result: ", sql) - } - if !reflect.DeepEqual(args, desiredArgs) { - t.Error("args is not the same as result: ", args) - } - }) + if sql != desiredSql { + t.Error("sql is not the same as result: ", sql) + } + if !reflect.DeepEqual(args, desiredArgs) { + t.Error("args is not the same as result: ", args) + } +} - t.Run("should be able to generate upsert query for sqlite", func(t *testing.T) { - sql, args, err := bob. - Upsert("users", bob.SQLite). - Columns("name", "email"). - Values("John Doe", "john@doe.com"). - Key("email"). - Replace("name", "John Does"). - PlaceholderFormat(bob.Question). - ToSql() - if err != nil { - t.Error(err) - } +func TestUpsert_SQLite(t *testing.T) { + sql, args, err := bob. + Upsert("users", bob.SQLite). + Columns("name", "email"). + Values("John Doe", "john@doe.com"). + Key("email"). + Replace("name", "John Does"). + PlaceholderFormat(bob.Question). + ToSql() + if err != nil { + t.Error(err) + } - desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES (?, ?) ON CONFLICT (\"email\") DO UPDATE SET \"name\" = ?;" - desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} + desiredSql := "INSERT INTO \"users\" (\"name\", \"email\") VALUES (?, ?) ON CONFLICT (\"email\") DO UPDATE SET \"name\" = ?;" + desiredArgs := []interface{}{"John Doe", "john@doe.com", "John Does"} - if sql != desiredSql { - t.Error("sql is not the same as result: ", sql) - } - if !reflect.DeepEqual(args, desiredArgs) { - t.Error("args is not the same as result: ", args) - } - }) + if sql != desiredSql { + t.Error("sql is not the same as result: ", sql) + } + if !reflect.DeepEqual(args, desiredArgs) { + t.Error("args is not the same as result: ", args) + } +} - t.Run("should be able to generate upsert query for mssql", func(t *testing.T) { - sql, args, err := bob. - Upsert("users", bob.MSSQL). - Columns("name", "email"). - Values("John Doe", "john@doe.com"). - Key("email", "john@doe.com"). - Replace("name", "John Does"). - PlaceholderFormat(bob.AtP). - ToSql() - if err != nil { - t.Error(err) - } +func TestUpsert_MSSQL(t *testing.T) { + sql, args, err := bob. + Upsert("users", bob.MSSQL). + Columns("name", "email"). + Values("John Doe", "john@doe.com"). + Key("email", "john@doe.com"). + Replace("name", "John Does"). + PlaceholderFormat(bob.AtP). + ToSql() + if err != nil { + t.Error(err) + } - desiredSql := "IF NOT EXISTS (SELECT * FROM \"users\" WHERE \"email\" = @p1) INSERT INTO \"users\" (\"name\", \"email\") VALUES (@p2, @p3) ELSE UPDATE \"users\" SET \"name\" = @p4 WHERE \"email\" = @p5;" - desiredArgs := []interface{}{"john@doe.com", "John Doe", "john@doe.com", "John Does", "john@doe.com"} + desiredSql := "IF NOT EXISTS (SELECT * FROM \"users\" WHERE \"email\" = @p1) INSERT INTO \"users\" (\"name\", \"email\") VALUES (@p2, @p3) ELSE UPDATE \"users\" SET \"name\" = @p4 WHERE \"email\" = @p5;" + desiredArgs := []interface{}{"john@doe.com", "John Doe", "john@doe.com", "John Does", "john@doe.com"} - if sql != desiredSql { - t.Error("sql is not the same as result: ", sql) - } - if !reflect.DeepEqual(args, desiredArgs) { - t.Error("args is not the same as result: ", args) - } - }) + if sql != desiredSql { + t.Error("sql is not the same as result: ", sql) + } + if !reflect.DeepEqual(args, desiredArgs) { + t.Error("args is not the same as result: ", args) + } +} +func TestUpsert_EmitErrors(t *testing.T) { t.Run("should emit error without table name", func(t *testing.T) { _, _, err := bob.Upsert("", bob.MySQL).ToSql() if err == nil && err.Error() != "upsert statement must specify a table" { @@ -152,4 +152,4 @@ func TestUpsert(t *testing.T) { t.Error(err) } }) -} \ No newline at end of file +}