-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement Go 1.23 iterator function for iterating rows
- Loading branch information
Showing
11 changed files
with
573 additions
and
74 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
//go:build go1.23 | ||
|
||
package constructsql | ||
|
||
import ( | ||
"errors" | ||
"iter" | ||
"log" | ||
) | ||
|
||
// IterateRows returns an iterator over the rows of a database query and scans them to T. | ||
// It returns a single-use iterator. | ||
func IterateRows[T any](rows Rows, err error) iter.Seq2[T, error] { | ||
if err != nil { | ||
return func(yield func(T, error) bool) { | ||
var result T | ||
yield(result, err) | ||
} | ||
} | ||
|
||
return func(yield func(T, error) bool) { | ||
var err error | ||
var iteratorClosed bool | ||
|
||
defer func() { | ||
closeErr := rows.Close() | ||
if iteratorClosed && closeErr != nil { | ||
log.Printf("constructsql: Error closing rows after function for loop body returned false: %v", closeErr) | ||
return | ||
} | ||
|
||
err = errors.Join(err, closeErr) | ||
if err != nil { | ||
var result T | ||
yield(result, err) | ||
} | ||
}() | ||
|
||
var value T | ||
for rows.Next() { | ||
value, err = scanRow[T](rows) | ||
if err != nil { | ||
return | ||
} | ||
if !yield(value, nil) { | ||
iteratorClosed = true | ||
return | ||
} | ||
} | ||
|
||
if err = rows.Err(); err != nil { | ||
return | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,258 @@ | ||
//go:build go1.23 | ||
|
||
package constructsql_test | ||
|
||
import ( | ||
"errors" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
|
||
"github.com/networkteam/construct/v2/constructsql" | ||
) | ||
|
||
func TestIterateRows(t *testing.T) { | ||
t.Run("iterate rows without error", func(t *testing.T) { | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
}, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
count++ | ||
} | ||
assert.Equal(t, 1, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate empty rows without error", func(t *testing.T) { | ||
rows := mockRows{ | ||
rows: []mockRow{}, | ||
} | ||
|
||
count := 0 | ||
for range constructsql.IterateRows[user](&rows, nil) { | ||
count++ | ||
} | ||
assert.Equal(t, 0, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with early break", func(t *testing.T) { | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
{ | ||
scanJSON: []byte(`{"id": 2, "name": "test"}`), | ||
}, | ||
}, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else { | ||
break | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 1, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with early break and close error", func(t *testing.T) { | ||
closeErr := errors.New("some error on close") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
{ | ||
scanJSON: []byte(`{"id": 2, "name": "test"}`), | ||
}, | ||
}, | ||
closeErr: closeErr, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else { | ||
break | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 1, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with initial error", func(t *testing.T) { | ||
rows := mockRows{} | ||
initialErr := errors.New("some initial error") | ||
count := 0 | ||
for _, err := range constructsql.IterateRows[user](&rows, initialErr) { | ||
assert.ErrorIs(t, err, initialErr) | ||
count++ | ||
} | ||
assert.Equal(t, 1, count) | ||
|
||
assert.False(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with scan error", func(t *testing.T) { | ||
scanErr := errors.New("some scan error") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
{ | ||
scanErr: scanErr, | ||
}, | ||
}, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else if count == 1 { | ||
assert.ErrorIs(t, err, scanErr) | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 2, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with iterate error", func(t *testing.T) { | ||
iterateErr := errors.New("some iterate error") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
}, | ||
iterateErr: iterateErr, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else if count == 1 { | ||
assert.ErrorIs(t, err, iterateErr) | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 2, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with close error", func(t *testing.T) { | ||
closeErr := errors.New("some error on close") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
}, | ||
closeErr: closeErr, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else if count == 1 { | ||
assert.ErrorIs(t, err, closeErr) | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 2, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with scan and close error", func(t *testing.T) { | ||
scanErr := errors.New("some scan error") | ||
closeErr := errors.New("some error on close") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
{ | ||
scanErr: scanErr, | ||
}, | ||
}, | ||
closeErr: closeErr, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else if count == 1 { | ||
assert.ErrorIs(t, err, scanErr) | ||
assert.ErrorIs(t, err, closeErr) | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 2, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
|
||
t.Run("iterate rows with iterate and close error", func(t *testing.T) { | ||
iterateErr := errors.New("some iterate error") | ||
closeErr := errors.New("some error on close") | ||
rows := mockRows{ | ||
rows: []mockRow{ | ||
{ | ||
scanJSON: []byte(`{"id": 1, "name": "test"}`), | ||
}, | ||
}, | ||
iterateErr: iterateErr, | ||
closeErr: closeErr, | ||
} | ||
|
||
count := 0 | ||
for record, err := range constructsql.IterateRows[user](&rows, nil) { | ||
if count == 0 { | ||
require.NoError(t, err) | ||
assert.Equal(t, user{ID: 1, Name: "test"}, record) | ||
} else if count == 1 { | ||
assert.ErrorIs(t, err, iterateErr) | ||
assert.ErrorIs(t, err, closeErr) | ||
} | ||
count++ | ||
} | ||
assert.Equal(t, 2, count) | ||
|
||
assert.True(t, rows.closed) | ||
}) | ||
} |
Oops, something went wrong.