mirror of https://github.com/VinGarcia/ksql.git
Add three new modifiers and fix a couple bugs
This commit adds these modifiers: 1. skipUpdates (with unit tests) 2. skipInserts (with unit tests) 3. skipQueries (no tests yet) (and not really working yet) And handles two situations previously not considered: 1. Updates with no attributes will now return a properly formatted error instead of returning a syntax error. 2. Inserts with no values will now work on SQLite, Postgres and SQLServerpull/29/head
parent
0d73ac9a18
commit
f41edb427d
|
@ -8,6 +8,7 @@ import (
|
||||||
|
|
||||||
// ErrRecordNotFound ...
|
// ErrRecordNotFound ...
|
||||||
var ErrRecordNotFound error = fmt.Errorf("ksql: the query returned no results: %w", sql.ErrNoRows)
|
var ErrRecordNotFound error = fmt.Errorf("ksql: the query returned no results: %w", sql.ErrNoRows)
|
||||||
|
var ErrNoValuesToUpdate error = fmt.Errorf("ksql: the input struct contains no values to update")
|
||||||
|
|
||||||
// ErrAbortIteration ...
|
// ErrAbortIteration ...
|
||||||
var ErrAbortIteration error = fmt.Errorf("ksql: abort iteration, should only be used inside QueryChunks function")
|
var ErrAbortIteration error = fmt.Errorf("ksql: abort iteration, should only be used inside QueryChunks function")
|
||||||
|
|
|
@ -9,10 +9,21 @@ import (
|
||||||
var modifiers sync.Map
|
var modifiers sync.Map
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// These are the builtin modifiers
|
// These are the builtin modifiers:
|
||||||
|
|
||||||
|
// This one is useful for serializing/desserializing structs:
|
||||||
modifiers.Store("json", jsonModifier)
|
modifiers.Store("json", jsonModifier)
|
||||||
|
|
||||||
|
// This next two are useful for the UpdatedAt and Created fields respectively:
|
||||||
|
// They only work on time.Time attributes and will set the attribute to time.Now().
|
||||||
modifiers.Store("timeNowUTC", timeNowUTCModifier)
|
modifiers.Store("timeNowUTC", timeNowUTCModifier)
|
||||||
modifiers.Store("timeNowUTC/skipUpdates", timeNowUTCSkipUpdatesModifier)
|
modifiers.Store("timeNowUTC/skipUpdates", timeNowUTCSkipUpdatesModifier)
|
||||||
|
|
||||||
|
// These are mostly example modifiers and they are also used
|
||||||
|
// to test the feature of skipping updates, inserts and queries.
|
||||||
|
modifiers.Store("skipUpdates", skipUpdatesModifier)
|
||||||
|
modifiers.Store("skipInserts", skipInsertsModifier)
|
||||||
|
modifiers.Store("skipQueries", skipQueriesModifier)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RegisterAttrModifier allow users to add custom modifiers on startup
|
// RegisterAttrModifier allow users to add custom modifiers on startup
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
package modifiers
|
||||||
|
|
||||||
|
var skipInsertsModifier = AttrModifier{
|
||||||
|
SkipOnInsert: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
var skipUpdatesModifier = AttrModifier{
|
||||||
|
SkipOnUpdate: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
var skipQueriesModifier = AttrModifier{
|
||||||
|
SkipOnQuery: true,
|
||||||
|
}
|
26
ksql.go
26
ksql.go
|
@ -258,8 +258,8 @@ func (c DB) QueryOne(
|
||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
|
|
||||||
if !rows.Next() {
|
if !rows.Next() {
|
||||||
if rows.Err() != nil {
|
if err := rows.Err(); err != nil {
|
||||||
return rows.Err()
|
return err
|
||||||
}
|
}
|
||||||
return ErrRecordNotFound
|
return ErrRecordNotFound
|
||||||
}
|
}
|
||||||
|
@ -709,11 +709,15 @@ func buildInsertQuery(
|
||||||
|
|
||||||
columnNames := []string{}
|
columnNames := []string{}
|
||||||
for col := range recordMap {
|
for col := range recordMap {
|
||||||
|
if info.ByName(col).Modifier.SkipOnInsert {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
columnNames = append(columnNames, col)
|
columnNames = append(columnNames, col)
|
||||||
}
|
}
|
||||||
|
|
||||||
params = make([]interface{}, len(recordMap))
|
params = make([]interface{}, len(columnNames))
|
||||||
valuesQuery := make([]string, len(recordMap))
|
valuesQuery := make([]string, len(columnNames))
|
||||||
for i, col := range columnNames {
|
for i, col := range columnNames {
|
||||||
recordValue := recordMap[col]
|
recordValue := recordMap[col]
|
||||||
params[i] = recordValue
|
params[i] = recordValue
|
||||||
|
@ -770,6 +774,16 @@ func buildInsertQuery(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if len(columnNames) == 0 && dialect.DriverName() != "mysql" {
|
||||||
|
query = fmt.Sprintf(
|
||||||
|
"INSERT INTO %s%s DEFAULT VALUES%s",
|
||||||
|
dialect.Escape(table.name),
|
||||||
|
outputQuery,
|
||||||
|
returningQuery,
|
||||||
|
)
|
||||||
|
return query, params, scanValues, nil
|
||||||
|
}
|
||||||
|
|
||||||
// Note that the outputQuery and the returningQuery depend
|
// Note that the outputQuery and the returningQuery depend
|
||||||
// on the selected driver, thus, they might be empty strings.
|
// on the selected driver, thus, they might be empty strings.
|
||||||
query = fmt.Sprintf(
|
query = fmt.Sprintf(
|
||||||
|
@ -807,6 +821,10 @@ func buildUpdateQuery(
|
||||||
numNonIDArgs := numAttrs - len(idFieldNames)
|
numNonIDArgs := numAttrs - len(idFieldNames)
|
||||||
whereArgs := args[numNonIDArgs:]
|
whereArgs := args[numNonIDArgs:]
|
||||||
|
|
||||||
|
if numNonIDArgs == 0 {
|
||||||
|
return "", nil, ErrNoValuesToUpdate
|
||||||
|
}
|
||||||
|
|
||||||
err = validateIfAllIdsArePresent(idFieldNames, recordMap)
|
err = validateIfAllIdsArePresent(idFieldNames, recordMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", nil, err
|
return "", nil, err
|
||||||
|
|
238
test_adapters.go
238
test_adapters.go
|
@ -896,6 +896,33 @@ func InsertTest(
|
||||||
tt.AssertNoErr(t, err)
|
tt.AssertNoErr(t, err)
|
||||||
tt.AssertEqual(t, inserted.Age, 5455)
|
tt.AssertEqual(t, inserted.Age, 5455)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("should work and retrieve the ID for structs with no attributes", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type tsUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipInserts"`
|
||||||
|
}
|
||||||
|
u := tsUser{
|
||||||
|
Name: "Letícia",
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &u)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, u.ID, 0)
|
||||||
|
|
||||||
|
var untaggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name *string `ksql:"name"`
|
||||||
|
}
|
||||||
|
err = c.QueryOne(ctx, &untaggedUser, `FROM users WHERE id = `+c.dialect.Placeholder(0), u.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, untaggedUser.Name, (*string)(nil))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("composite key tables", func(t *testing.T) {
|
t.Run("composite key tables", func(t *testing.T) {
|
||||||
|
@ -1714,6 +1741,24 @@ func PatchTest(
|
||||||
tt.AssertNotEqual(t, err, nil)
|
tt.AssertNotEqual(t, err, nil)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("should report error if the struct has no fields to update", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
err = c.Update(ctx, usersTable, struct {
|
||||||
|
ID uint `ksql:"id"` // ID fields are not updated
|
||||||
|
Name string `ksql:"name,skipUpdates"` // the skipUpdate modifier should rule this one out
|
||||||
|
Age *int `ksql:"age"` // Age is a nil pointer so it would not be updated
|
||||||
|
}{
|
||||||
|
ID: 1,
|
||||||
|
Name: "some name",
|
||||||
|
})
|
||||||
|
tt.AssertEqual(t, err, ErrNoValuesToUpdate)
|
||||||
|
})
|
||||||
|
|
||||||
t.Run("should report error if the id is missing", func(t *testing.T) {
|
t.Run("should report error if the id is missing", func(t *testing.T) {
|
||||||
t.Run("with a single primary key", func(t *testing.T) {
|
t.Run("with a single primary key", func(t *testing.T) {
|
||||||
db, closer := newDBAdapter(t)
|
db, closer := newDBAdapter(t)
|
||||||
|
@ -2749,6 +2794,199 @@ func ModifiersTest(
|
||||||
tt.AssertEqual(t, taggedUser.CreatedAt, tt.ParseTime(t, "2000-08-05T14:00:00Z"))
|
tt.AssertEqual(t, taggedUser.CreatedAt, tt.ParseTime(t, "2000-08-05T14:00:00Z"))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("skipInserts modifier", func(t *testing.T) {
|
||||||
|
t.Run("should ignore the field during insertions", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type tsUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipInserts"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
u := tsUser{
|
||||||
|
Name: "Letícia",
|
||||||
|
Age: 22,
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &u)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, u.ID, 0)
|
||||||
|
|
||||||
|
var untaggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name *string `ksql:"name"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
err = c.QueryOne(ctx, &untaggedUser, `FROM users WHERE id = `+c.dialect.Placeholder(0), u.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, untaggedUser.Name, (*string)(nil))
|
||||||
|
tt.AssertEqual(t, untaggedUser.Age, 22)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("should have no effect on updates", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type userWithNoTags struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
untaggedUser := userWithNoTags{
|
||||||
|
Name: "Laurinha Ribeiro",
|
||||||
|
Age: 11,
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &untaggedUser)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, untaggedUser.ID, 0)
|
||||||
|
|
||||||
|
type taggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipInserts"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
u := taggedUser{
|
||||||
|
ID: untaggedUser.ID,
|
||||||
|
Name: "Laura Ribeiro",
|
||||||
|
Age: 12,
|
||||||
|
}
|
||||||
|
err = c.Patch(ctx, usersTable, u)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
|
||||||
|
var untaggedUser2 userWithNoTags
|
||||||
|
err = c.QueryOne(ctx, &untaggedUser2, "FROM users WHERE id = "+c.dialect.Placeholder(0), u.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, untaggedUser2.Name, "Laura Ribeiro")
|
||||||
|
tt.AssertEqual(t, untaggedUser2.Age, 12)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("should not alter the value on queries", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type userWithNoTags struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name"`
|
||||||
|
}
|
||||||
|
untaggedUser := userWithNoTags{
|
||||||
|
Name: "Marta Ribeiro",
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &untaggedUser)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, untaggedUser.ID, 0)
|
||||||
|
|
||||||
|
var taggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipInserts"`
|
||||||
|
}
|
||||||
|
err = c.QueryOne(ctx, &taggedUser, "FROM users WHERE id = "+c.dialect.Placeholder(0), untaggedUser.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, taggedUser.ID, untaggedUser.ID)
|
||||||
|
tt.AssertEqual(t, taggedUser.Name, "Marta Ribeiro")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("skipUpdates modifier", func(t *testing.T) {
|
||||||
|
t.Run("should set the field on insertion", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type tsUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipUpdates"`
|
||||||
|
}
|
||||||
|
u := tsUser{
|
||||||
|
Name: "Letícia",
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &u)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, u.ID, 0)
|
||||||
|
|
||||||
|
var untaggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name"`
|
||||||
|
}
|
||||||
|
err = c.QueryOne(ctx, &untaggedUser, `FROM users WHERE id = `+c.dialect.Placeholder(0), u.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, untaggedUser.Name, "Letícia")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("should be ignored on updates", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type userWithNoTags struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
untaggedUser := userWithNoTags{
|
||||||
|
Name: "Laurinha Ribeiro",
|
||||||
|
Age: 11,
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &untaggedUser)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, untaggedUser.ID, 0)
|
||||||
|
|
||||||
|
type taggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipUpdates"`
|
||||||
|
Age int `ksql:"age"`
|
||||||
|
}
|
||||||
|
u := taggedUser{
|
||||||
|
ID: untaggedUser.ID,
|
||||||
|
Name: "Laura Ribeiro",
|
||||||
|
Age: 12,
|
||||||
|
}
|
||||||
|
err = c.Patch(ctx, usersTable, u)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
|
||||||
|
var untaggedUser2 userWithNoTags
|
||||||
|
err = c.QueryOne(ctx, &untaggedUser2, "FROM users WHERE id = "+c.dialect.Placeholder(0), u.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, untaggedUser2.Name, "Laurinha Ribeiro")
|
||||||
|
tt.AssertEqual(t, untaggedUser2.Age, 12)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("should not alter the value on queries", func(t *testing.T) {
|
||||||
|
db, closer := newDBAdapter(t)
|
||||||
|
defer closer.Close()
|
||||||
|
|
||||||
|
c := newTestDB(db, driver)
|
||||||
|
|
||||||
|
type userWithNoTags struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name"`
|
||||||
|
}
|
||||||
|
untaggedUser := userWithNoTags{
|
||||||
|
Name: "Marta Ribeiro",
|
||||||
|
}
|
||||||
|
err := c.Insert(ctx, usersTable, &untaggedUser)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertNotEqual(t, untaggedUser.ID, 0)
|
||||||
|
|
||||||
|
var taggedUser struct {
|
||||||
|
ID uint `ksql:"id"`
|
||||||
|
Name string `ksql:"name,skipUpdates"`
|
||||||
|
}
|
||||||
|
err = c.QueryOne(ctx, &taggedUser, "FROM users WHERE id = "+c.dialect.Placeholder(0), untaggedUser.ID)
|
||||||
|
tt.AssertNoErr(t, err)
|
||||||
|
tt.AssertEqual(t, taggedUser.ID, untaggedUser.ID)
|
||||||
|
tt.AssertEqual(t, taggedUser.Name, "Marta Ribeiro")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue