Replace all ocurrencies of `KissORM` for `KissSQL`

pull/2/head
Vinícius Garcia 2021-03-08 11:18:52 -03:00
parent 0bd858efee
commit 568c61bdba
15 changed files with 212 additions and 212 deletions

View File

@ -1,9 +1,9 @@
# KissORM # KissSQL
Welcome to the KissORM project, the Keep It Stupid Simple - ORM. Welcome to the KissSQL project, the Keep It Stupid Simple sql package.
This ORM was created to be used by any developer efficiently and safely. This package was created to be used by any developer efficiently and safely.
The goals were: The goals were:
- To be easy to use - To be easy to use
@ -19,11 +19,11 @@ Currently we only support 2 Drivers:
- `"postgres"` - `"postgres"`
- `"sqlite3"` - `"sqlite3"`
### Why KissORM? ### Why KissSQL?
> Note: If you want numbers see our Benchmark section below > Note: If you want numbers see our Benchmark section below
KissORM was created to fill a hole between the complexity KissSQL was created to fill a hole between the complexity
we find in the tools I've seen so far, namely: we find in the tools I've seen so far, namely:
- ORMs such as `GORM` that do a lot and have literally hundreds - ORMs such as `GORM` that do a lot and have literally hundreds
@ -43,9 +43,9 @@ So the goal was to be high level enough that it would
avoid the complications from the `sql` package and avoid the complications from the `sql` package and
at the same time to be simple enough to avoid at the same time to be simple enough to avoid
the big learning curve and complexity of the hundreds the big learning curve and complexity of the hundreds
of functions offered by more complete ORMs. of functions offered by ORMs.
That said, KissORM attempts to apply the Kiss principle, That said, KissSQL attempts to apply the Kiss principle,
in order to save development time for your team, i.e.: in order to save development time for your team, i.e.:
- Less time spent learning (few methods to learn) - Less time spent learning (few methods to learn)
@ -59,8 +59,8 @@ The current interface is as follows and we plan to keep
it with as little functions as possible, so don't expect many additions: it with as little functions as possible, so don't expect many additions:
```go ```go
// ORMProvider describes the public behavior of this ORM // SQLProvider describes the public behavior of this ORM
type ORMProvider interface { type SQLProvider interface {
Insert(ctx context.Context, records ...interface{}) error Insert(ctx context.Context, records ...interface{}) error
Delete(ctx context.Context, ids ...interface{}) error Delete(ctx context.Context, ids ...interface{}) error
Update(ctx context.Context, records ...interface{}) error Update(ctx context.Context, records ...interface{}) error
@ -70,7 +70,7 @@ type ORMProvider interface {
QueryChunks(ctx context.Context, parser ChunkParser) error QueryChunks(ctx context.Context, parser ChunkParser) error
Exec(ctx context.Context, query string, params ...interface{}) error Exec(ctx context.Context, query string, params ...interface{}) error
Transaction(ctx context.Context, fn func(ORMProvider) error) error Transaction(ctx context.Context, fn func(SQLProvider) error) error
} }
``` ```
@ -87,24 +87,24 @@ import (
"fmt" "fmt"
_ "github.com/mattn/go-sqlite3" _ "github.com/mattn/go-sqlite3"
"github.com/vingarcia/kissorm" "github.com/vingarcia/kisssql"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
) )
type User struct { type User struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
// This field will be saved as JSON in the database // This field will be saved as JSON in the database
Address Address `kissorm:"address,json"` Address Address `kisssql:"address,json"`
} }
type PartialUpdateUser struct { type PartialUpdateUser struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
Address *Address `kissorm:"address,json"` Address *Address `kisssql:"address,json"`
} }
type Address struct { type Address struct {
@ -114,7 +114,7 @@ type Address struct {
func main() { func main() {
ctx := context.Background() ctx := context.Background()
db, err := kissorm.New("sqlite3", "/tmp/hello.sqlite", 1, "users") db, err := kisssql.New("sqlite3", "/tmp/hello.sqlite", 1, "users")
if err != nil { if err != nil {
panic(err.Error()) panic(err.Error())
} }
@ -178,8 +178,8 @@ func main() {
// Partial update technique 1: // Partial update technique 1:
err = db.Update(ctx, struct { err = db.Update(ctx, struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
}{ID: cris.ID, Age: 28}) }{ID: cris.ID, Age: 28})
if err != nil { if err != nil {
panic(err.Error()) panic(err.Error())
@ -216,9 +216,9 @@ func main() {
This library has a few helper functions for helping your tests: This library has a few helper functions for helping your tests:
- `kissorm.FillStructWith(struct interface{}, dbRow map[string]interface{}) error` - `kisssql.FillStructWith(struct interface{}, dbRow map[string]interface{}) error`
- `kissorm.FillSliceWith(structSlice interface{}, dbRows []map[string]interface{}) error` - `kisssql.FillSliceWith(structSlice interface{}, dbRows []map[string]interface{}) error`
- `kissorm.StructToMap(struct interface{}) (map[string]interface{}, error)` - `kisssql.StructToMap(struct interface{}) (map[string]interface{}, error)`
If you want to see examples (we have examples for all the public functions) just If you want to see examples (we have examples for all the public functions) just
read the example tests available on our [example service](./examples/example_service) read the example tests available on our [example service](./examples/example_service)
@ -232,16 +232,16 @@ $ make bench TIME=3s
go test -bench=. -benchtime=3s go test -bench=. -benchtime=3s
goos: linux goos: linux
goarch: amd64 goarch: amd64
pkg: github.com/vingarcia/kissorm pkg: github.com/vingarcia/kisssql
cpu: Intel(R) Core(TM) i5-3210M CPU @ 2.50GHz cpu: Intel(R) Core(TM) i5-3210M CPU @ 2.50GHz
BenchmarkInsert/kissorm-setup/insert-one-4 4302 776648 ns/op BenchmarkInsert/kisssql-setup/insert-one-4 4302 776648 ns/op
BenchmarkInsert/sqlx-setup/insert-one-4 4716 762358 ns/op BenchmarkInsert/sqlx-setup/insert-one-4 4716 762358 ns/op
BenchmarkQuery/kissorm-setup/single-row-4 12204 293858 ns/op BenchmarkQuery/kisssql-setup/single-row-4 12204 293858 ns/op
BenchmarkQuery/kissorm-setup/multiple-rows-4 11145 323571 ns/op BenchmarkQuery/kisssql-setup/multiple-rows-4 11145 323571 ns/op
BenchmarkQuery/sqlx-setup/single-row-4 12440 290937 ns/op BenchmarkQuery/sqlx-setup/single-row-4 12440 290937 ns/op
BenchmarkQuery/sqlx-setup/multiple-rows-4 10000 310314 ns/op BenchmarkQuery/sqlx-setup/multiple-rows-4 10000 310314 ns/op
PASS PASS
ok github.com/vingarcia/kissorm 34.251s ok github.com/vingarcia/kisssql 34.251s
``` ```
### TODO List ### TODO List

View File

@ -1,4 +1,4 @@
package kissorm_test package kisssql_test
import ( import (
"context" "context"
@ -9,16 +9,16 @@ import (
"github.com/jmoiron/sqlx" "github.com/jmoiron/sqlx"
_ "github.com/lib/pq" _ "github.com/lib/pq"
"github.com/vingarcia/kissorm" "github.com/vingarcia/kisssql"
) )
func BenchmarkInsert(b *testing.B) { func BenchmarkInsert(b *testing.B) {
ctx := context.Background() ctx := context.Background()
driver := "postgres" driver := "postgres"
connStr := "host=localhost port=5432 user=postgres password=postgres dbname=kissorm sslmode=disable" connStr := "host=localhost port=5432 user=postgres password=postgres dbname=kisssql sslmode=disable"
kissormDB, err := kissorm.New(driver, connStr, kissorm.Config{ kisssqlDB, err := kisssql.New(driver, connStr, kisssql.Config{
MaxOpenConns: 1, MaxOpenConns: 1,
TableName: "users", TableName: "users",
}) })
@ -27,12 +27,12 @@ func BenchmarkInsert(b *testing.B) {
} }
type User struct { type User struct {
ID int `kissorm:"id" db:"id"` ID int `kisssql:"id" db:"id"`
Name string `kissorm:"name" db:"name"` Name string `kisssql:"name" db:"name"`
Age int `kissorm:"age" db:"age"` Age int `kisssql:"age" db:"age"`
} }
b.Run("kissorm-setup", func(b *testing.B) { b.Run("kisssql-setup", func(b *testing.B) {
err := recreateTable(connStr) err := recreateTable(connStr)
if err != nil { if err != nil {
b.Fatalf("error creating table: %s", err.Error()) b.Fatalf("error creating table: %s", err.Error())
@ -40,7 +40,7 @@ func BenchmarkInsert(b *testing.B) {
b.Run("insert-one", func(b *testing.B) { b.Run("insert-one", func(b *testing.B) {
for i := 0; i < b.N; i++ { for i := 0; i < b.N; i++ {
err := kissormDB.Insert(ctx, &User{ err := kisssqlDB.Insert(ctx, &User{
Name: strconv.Itoa(i), Name: strconv.Itoa(i),
Age: i, Age: i,
}) })
@ -88,9 +88,9 @@ func BenchmarkQuery(b *testing.B) {
ctx := context.Background() ctx := context.Background()
driver := "postgres" driver := "postgres"
connStr := "host=localhost port=5432 user=postgres password=postgres dbname=kissorm sslmode=disable" connStr := "host=localhost port=5432 user=postgres password=postgres dbname=kisssql sslmode=disable"
kissormDB, err := kissorm.New(driver, connStr, kissorm.Config{ kisssqlDB, err := kisssql.New(driver, connStr, kisssql.Config{
MaxOpenConns: 1, MaxOpenConns: 1,
TableName: "users", TableName: "users",
}) })
@ -99,12 +99,12 @@ func BenchmarkQuery(b *testing.B) {
} }
type User struct { type User struct {
ID int `kissorm:"id" db:"id"` ID int `kisssql:"id" db:"id"`
Name string `kissorm:"name" db:"name"` Name string `kisssql:"name" db:"name"`
Age int `kissorm:"age" db:"age"` Age int `kisssql:"age" db:"age"`
} }
b.Run("kissorm-setup", func(b *testing.B) { b.Run("kisssql-setup", func(b *testing.B) {
err := recreateTable(connStr) err := recreateTable(connStr)
if err != nil { if err != nil {
b.Fatalf("error creating table: %s", err.Error()) b.Fatalf("error creating table: %s", err.Error())
@ -118,7 +118,7 @@ func BenchmarkQuery(b *testing.B) {
b.Run("single-row", func(b *testing.B) { b.Run("single-row", func(b *testing.B) {
for i := 0; i < b.N; i++ { for i := 0; i < b.N; i++ {
var user User var user User
err := kissormDB.QueryOne(ctx, &user, `SELECT * FROM users OFFSET $1 LIMIT 1`, i%100) err := kisssqlDB.QueryOne(ctx, &user, `SELECT * FROM users OFFSET $1 LIMIT 1`, i%100)
if err != nil { if err != nil {
b.Fatalf("query error: %s", err.Error()) b.Fatalf("query error: %s", err.Error())
} }
@ -128,7 +128,7 @@ func BenchmarkQuery(b *testing.B) {
b.Run("multiple-rows", func(b *testing.B) { b.Run("multiple-rows", func(b *testing.B) {
for i := 0; i < b.N; i++ { for i := 0; i < b.N; i++ {
var users []User var users []User
err := kissormDB.Query(ctx, &users, `SELECT * FROM users OFFSET $1 LIMIT 10`, i%90) err := kisssqlDB.Query(ctx, &users, `SELECT * FROM users OFFSET $1 LIMIT 10`, i%90)
if err != nil { if err != nil {
b.Fatalf("query error: %s", err.Error()) b.Fatalf("query error: %s", err.Error())
} }

View File

@ -1,4 +1,4 @@
package kissorm package kisssql
import ( import (
"context" "context"
@ -9,13 +9,13 @@ import (
) )
// ErrRecordNotFound ... // ErrRecordNotFound ...
var ErrRecordNotFound error = errors.Wrap(sql.ErrNoRows, "kissorm: the query returned no results") var ErrRecordNotFound error = errors.Wrap(sql.ErrNoRows, "kisssql: the query returned no results")
// ErrAbortIteration ... // ErrAbortIteration ...
var ErrAbortIteration error = fmt.Errorf("kissorm: abort iteration, should only be used inside QueryChunks function") var ErrAbortIteration error = fmt.Errorf("kisssql: abort iteration, should only be used inside QueryChunks function")
// ORMProvider describes the public behavior of this ORM // SQLProvider describes the public behavior of this ORM
type ORMProvider interface { type SQLProvider interface {
Insert(ctx context.Context, records ...interface{}) error Insert(ctx context.Context, records ...interface{}) error
Delete(ctx context.Context, ids ...interface{}) error Delete(ctx context.Context, ids ...interface{}) error
Update(ctx context.Context, records ...interface{}) error Update(ctx context.Context, records ...interface{}) error
@ -25,7 +25,7 @@ type ORMProvider interface {
QueryChunks(ctx context.Context, parser ChunkParser) error QueryChunks(ctx context.Context, parser ChunkParser) error
Exec(ctx context.Context, query string, params ...interface{}) error Exec(ctx context.Context, query string, params ...interface{}) error
Transaction(ctx context.Context, fn func(ORMProvider) error) error Transaction(ctx context.Context, fn func(SQLProvider) error) error
} }
// ChunkParser stores the arguments of the QueryChunks function // ChunkParser stores the arguments of the QueryChunks function

View File

@ -1,4 +1,4 @@
package kissorm package kisssql
import "strconv" import "strconv"

View File

@ -5,26 +5,26 @@ import (
"fmt" "fmt"
_ "github.com/mattn/go-sqlite3" _ "github.com/mattn/go-sqlite3"
"github.com/vingarcia/kissorm" "github.com/vingarcia/kisssql"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
) )
// User ... // User ...
type User struct { type User struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
// This field will be saved as JSON in the database // This field will be saved as JSON in the database
Address Address `kissorm:"address,json"` Address Address `kisssql:"address,json"`
} }
// PartialUpdateUser ... // PartialUpdateUser ...
type PartialUpdateUser struct { type PartialUpdateUser struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
Address *Address `kissorm:"address,json"` Address *Address `kisssql:"address,json"`
} }
// Address ... // Address ...
@ -35,7 +35,7 @@ type Address struct {
func main() { func main() {
ctx := context.Background() ctx := context.Background()
db, err := kissorm.New("sqlite3", "/tmp/hello.sqlite", kissorm.Config{ db, err := kisssql.New("sqlite3", "/tmp/hello.sqlite", kisssql.Config{
MaxOpenConns: 1, MaxOpenConns: 1,
TableName: "users", TableName: "users",
}) })
@ -102,8 +102,8 @@ func main() {
// Partial update technique 1: // Partial update technique 1:
err = db.Update(ctx, struct { err = db.Update(ctx, struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
}{ID: cris.ID, Age: 28}) }{ID: cris.ID, Age: 28})
if err != nil { if err != nil {
panic(err.Error()) panic(err.Error())

View File

@ -4,13 +4,13 @@ import (
"context" "context"
"time" "time"
"github.com/vingarcia/kissorm" "github.com/vingarcia/kisssql"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
) )
// Service ... // Service ...
type Service struct { type Service struct {
usersTable kissorm.ORMProvider usersTable kisssql.SQLProvider
streamChunkSize int streamChunkSize int
} }
@ -25,12 +25,12 @@ type Service struct {
// If this is not the case, it might be a good idea // If this is not the case, it might be a good idea
// to create a DTO struct to receive select queries. // to create a DTO struct to receive select queries.
type UserEntity struct { type UserEntity struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
Score *int `kissorm:"score"` Score *int `kisssql:"score"`
LastPayment time.Time `kissorm:"last_payment"` LastPayment time.Time `kisssql:"last_payment"`
Address *Address `kissorm:"address,json"` Address *Address `kisssql:"address,json"`
} }
// Address contains the user's address // Address contains the user's address
@ -42,7 +42,7 @@ type Address struct {
} }
// NewUserService ... // NewUserService ...
func NewUserService(usersTable kissorm.ORMProvider) Service { func NewUserService(usersTable kisssql.SQLProvider) Service {
return Service{ return Service{
usersTable: usersTable, usersTable: usersTable,
streamChunkSize: 100, streamChunkSize: 100,
@ -58,7 +58,7 @@ func (s Service) CreateUser(ctx context.Context, u UserEntity) error {
// user score. Defaults to 0 if not set. // user score. Defaults to 0 if not set.
func (s Service) UpdateUserScore(ctx context.Context, uID int, scoreChange int) error { func (s Service) UpdateUserScore(ctx context.Context, uID int, scoreChange int) error {
var scoreRow struct { var scoreRow struct {
Score int `kissorm:"score"` Score int `kisssql:"score"`
} }
err := s.usersTable.QueryOne(ctx, &scoreRow, "SELECT score FROM users WHERE id = ?", uID) err := s.usersTable.QueryOne(ctx, &scoreRow, "SELECT score FROM users WHERE id = ?", uID)
if err != nil { if err != nil {
@ -74,7 +74,7 @@ func (s Service) UpdateUserScore(ctx context.Context, uID int, scoreChange int)
// ListUsers returns a page of users // ListUsers returns a page of users
func (s Service) ListUsers(ctx context.Context, offset, limit int) (total int, users []UserEntity, err error) { func (s Service) ListUsers(ctx context.Context, offset, limit int) (total int, users []UserEntity, err error) {
var countRow struct { var countRow struct {
Count int `kissorm:"count"` Count int `kisssql:"count"`
} }
err = s.usersTable.QueryOne(ctx, &countRow, "SELECT count(*) as count FROM users") err = s.usersTable.QueryOne(ctx, &countRow, "SELECT count(*) as count FROM users")
if err != nil { if err != nil {
@ -91,7 +91,7 @@ func (s Service) ListUsers(ctx context.Context, offset, limit int) (total int, u
// function only when the ammount of data loaded might exceed the available memory and/or // function only when the ammount of data loaded might exceed the available memory and/or
// when you can't put an upper limit on the number of values returned. // when you can't put an upper limit on the number of values returned.
func (s Service) StreamAllUsers(ctx context.Context, sendUser func(u UserEntity) error) error { func (s Service) StreamAllUsers(ctx context.Context, sendUser func(u UserEntity) error) error {
return s.usersTable.QueryChunks(ctx, kissorm.ChunkParser{ return s.usersTable.QueryChunks(ctx, kisssql.ChunkParser{
Query: "SELECT * FROM users", Query: "SELECT * FROM users",
Params: []interface{}{}, Params: []interface{}{},
ChunkSize: s.streamChunkSize, ChunkSize: s.streamChunkSize,

View File

@ -7,17 +7,17 @@ import (
gomock "github.com/golang/mock/gomock" gomock "github.com/golang/mock/gomock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"github.com/tj/assert" "github.com/tj/assert"
"github.com/vingarcia/kissorm" "github.com/vingarcia/kisssql"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
"github.com/vingarcia/kissorm/structs" "github.com/vingarcia/kisssql/structs"
) )
func TestCreateUser(t *testing.T) { func TestCreateUser(t *testing.T) {
t.Run("should call kissorm.Insert correctly", func(t *testing.T) { t.Run("should call kisssql.Insert correctly", func(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,
@ -43,7 +43,7 @@ func TestCreateUser(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,
@ -54,8 +54,8 @@ func TestCreateUser(t *testing.T) {
usersTableMock.EXPECT().Insert(gomock.Any(), gomock.Any()). usersTableMock.EXPECT().Insert(gomock.Any(), gomock.Any()).
DoAndReturn(func(ctx context.Context, records ...interface{}) error { DoAndReturn(func(ctx context.Context, records ...interface{}) error {
for _, record := range records { for _, record := range records {
// The StructToMap function will convert a struct with `kissorm` tags // The StructToMap function will convert a struct with `kisssql` tags
// into a map using the kissorm attr names as keys. // into a map using the kisssql attr names as keys.
// //
// If you are inserting an anonymous struct (not usual) this function // If you are inserting an anonymous struct (not usual) this function
// can make your tests shorter: // can make your tests shorter:
@ -79,11 +79,11 @@ func TestCreateUser(t *testing.T) {
} }
func TestUpdateUserScore(t *testing.T) { func TestUpdateUserScore(t *testing.T) {
t.Run("should call kissorm.QueryOne() & Update() correctly", func(t *testing.T) { t.Run("should call kisssql.QueryOne() & Update() correctly", func(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,
@ -97,7 +97,7 @@ func TestUpdateUserScore(t *testing.T) {
// This function will use reflection to fill the // This function will use reflection to fill the
// struct fields with the values from the map // struct fields with the values from the map
return structs.FillStructWith(result, map[string]interface{}{ return structs.FillStructWith(result, map[string]interface{}{
// Use int this map the keys you set on the kissorm tags, e.g. `kissorm:"score"` // Use int this map the keys you set on the kisssql tags, e.g. `kisssql:"score"`
// Each of these fields represent the database rows returned // Each of these fields represent the database rows returned
// by the query. // by the query.
"score": 42, "score": 42,
@ -123,11 +123,11 @@ func TestUpdateUserScore(t *testing.T) {
} }
func TestListUsers(t *testing.T) { func TestListUsers(t *testing.T) {
t.Run("should call kissorm.QueryOne() & Query() correctly", func(t *testing.T) { t.Run("should call kisssql.QueryOne() & Query() correctly", func(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,
@ -140,7 +140,7 @@ func TestListUsers(t *testing.T) {
// This function will use reflection to fill the // This function will use reflection to fill the
// struct fields with the values from the map // struct fields with the values from the map
return structs.FillStructWith(result, map[string]interface{}{ return structs.FillStructWith(result, map[string]interface{}{
// Use int this map the keys you set on the kissorm tags, e.g. `kissorm:"score"` // Use int this map the keys you set on the kisssql tags, e.g. `kisssql:"score"`
// Each of these fields represent the database rows returned // Each of these fields represent the database rows returned
// by the query. // by the query.
"count": 420, "count": 420,
@ -185,11 +185,11 @@ func TestListUsers(t *testing.T) {
} }
func TestStreamAllUsers(t *testing.T) { func TestStreamAllUsers(t *testing.T) {
t.Run("should call kissorm.QueryChunks correctly", func(t *testing.T) { t.Run("should call kisssql.QueryChunks correctly", func(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,
@ -197,7 +197,7 @@ func TestStreamAllUsers(t *testing.T) {
} }
usersTableMock.EXPECT().QueryChunks(gomock.Any(), gomock.Any()). usersTableMock.EXPECT().QueryChunks(gomock.Any(), gomock.Any()).
DoAndReturn(func(ctx context.Context, parser kissorm.ChunkParser) error { DoAndReturn(func(ctx context.Context, parser kisssql.ChunkParser) error {
fn, ok := parser.ForEachChunk.(func(users []UserEntity) error) fn, ok := parser.ForEachChunk.(func(users []UserEntity) error)
require.True(t, ok) require.True(t, ok)
// Chunk 1: // Chunk 1:
@ -259,11 +259,11 @@ func TestStreamAllUsers(t *testing.T) {
} }
func TestDeleteUser(t *testing.T) { func TestDeleteUser(t *testing.T) {
t.Run("should call kissorm.Delete correctly", func(t *testing.T) { t.Run("should call kisssql.Delete correctly", func(t *testing.T) {
controller := gomock.NewController(t) controller := gomock.NewController(t)
defer controller.Finish() defer controller.Finish()
usersTableMock := NewMockORMProvider(controller) usersTableMock := NewMockSQLProvider(controller)
s := Service{ s := Service{
usersTable: usersTableMock, usersTable: usersTableMock,

View File

@ -9,34 +9,34 @@ import (
reflect "reflect" reflect "reflect"
gomock "github.com/golang/mock/gomock" gomock "github.com/golang/mock/gomock"
kissorm "github.com/vingarcia/kissorm" kisssql "github.com/vingarcia/kisssql"
) )
// MockORMProvider is a mock of ORMProvider interface. // MockSQLProvider is a mock of SQLProvider interface.
type MockORMProvider struct { type MockSQLProvider struct {
ctrl *gomock.Controller ctrl *gomock.Controller
recorder *MockORMProviderMockRecorder recorder *MockSQLProviderMockRecorder
} }
// MockORMProviderMockRecorder is the mock recorder for MockORMProvider. // MockSQLProviderMockRecorder is the mock recorder for MockSQLProvider.
type MockORMProviderMockRecorder struct { type MockSQLProviderMockRecorder struct {
mock *MockORMProvider mock *MockSQLProvider
} }
// NewMockORMProvider creates a new mock instance. // NewMockSQLProvider creates a new mock instance.
func NewMockORMProvider(ctrl *gomock.Controller) *MockORMProvider { func NewMockSQLProvider(ctrl *gomock.Controller) *MockSQLProvider {
mock := &MockORMProvider{ctrl: ctrl} mock := &MockSQLProvider{ctrl: ctrl}
mock.recorder = &MockORMProviderMockRecorder{mock} mock.recorder = &MockSQLProviderMockRecorder{mock}
return mock return mock
} }
// EXPECT returns an object that allows the caller to indicate expected use. // EXPECT returns an object that allows the caller to indicate expected use.
func (m *MockORMProvider) EXPECT() *MockORMProviderMockRecorder { func (m *MockSQLProvider) EXPECT() *MockSQLProviderMockRecorder {
return m.recorder return m.recorder
} }
// Delete mocks base method. // Delete mocks base method.
func (m *MockORMProvider) Delete(ctx context.Context, ids ...interface{}) error { func (m *MockSQLProvider) Delete(ctx context.Context, ids ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx} varargs := []interface{}{ctx}
for _, a := range ids { for _, a := range ids {
@ -48,14 +48,14 @@ func (m *MockORMProvider) Delete(ctx context.Context, ids ...interface{}) error
} }
// Delete indicates an expected call of Delete. // Delete indicates an expected call of Delete.
func (mr *MockORMProviderMockRecorder) Delete(ctx interface{}, ids ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Delete(ctx interface{}, ids ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx}, ids...) varargs := append([]interface{}{ctx}, ids...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Delete", reflect.TypeOf((*MockORMProvider)(nil).Delete), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Delete", reflect.TypeOf((*MockSQLProvider)(nil).Delete), varargs...)
} }
// Exec mocks base method. // Exec mocks base method.
func (m *MockORMProvider) Exec(ctx context.Context, query string, params ...interface{}) error { func (m *MockSQLProvider) Exec(ctx context.Context, query string, params ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx, query} varargs := []interface{}{ctx, query}
for _, a := range params { for _, a := range params {
@ -67,14 +67,14 @@ func (m *MockORMProvider) Exec(ctx context.Context, query string, params ...inte
} }
// Exec indicates an expected call of Exec. // Exec indicates an expected call of Exec.
func (mr *MockORMProviderMockRecorder) Exec(ctx, query interface{}, params ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Exec(ctx, query interface{}, params ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx, query}, params...) varargs := append([]interface{}{ctx, query}, params...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Exec", reflect.TypeOf((*MockORMProvider)(nil).Exec), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Exec", reflect.TypeOf((*MockSQLProvider)(nil).Exec), varargs...)
} }
// Insert mocks base method. // Insert mocks base method.
func (m *MockORMProvider) Insert(ctx context.Context, records ...interface{}) error { func (m *MockSQLProvider) Insert(ctx context.Context, records ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx} varargs := []interface{}{ctx}
for _, a := range records { for _, a := range records {
@ -86,14 +86,14 @@ func (m *MockORMProvider) Insert(ctx context.Context, records ...interface{}) er
} }
// Insert indicates an expected call of Insert. // Insert indicates an expected call of Insert.
func (mr *MockORMProviderMockRecorder) Insert(ctx interface{}, records ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Insert(ctx interface{}, records ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx}, records...) varargs := append([]interface{}{ctx}, records...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Insert", reflect.TypeOf((*MockORMProvider)(nil).Insert), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Insert", reflect.TypeOf((*MockSQLProvider)(nil).Insert), varargs...)
} }
// Query mocks base method. // Query mocks base method.
func (m *MockORMProvider) Query(ctx context.Context, records interface{}, query string, params ...interface{}) error { func (m *MockSQLProvider) Query(ctx context.Context, records interface{}, query string, params ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx, records, query} varargs := []interface{}{ctx, records, query}
for _, a := range params { for _, a := range params {
@ -105,14 +105,14 @@ func (m *MockORMProvider) Query(ctx context.Context, records interface{}, query
} }
// Query indicates an expected call of Query. // Query indicates an expected call of Query.
func (mr *MockORMProviderMockRecorder) Query(ctx, records, query interface{}, params ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Query(ctx, records, query interface{}, params ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx, records, query}, params...) varargs := append([]interface{}{ctx, records, query}, params...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Query", reflect.TypeOf((*MockORMProvider)(nil).Query), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Query", reflect.TypeOf((*MockSQLProvider)(nil).Query), varargs...)
} }
// QueryChunks mocks base method. // QueryChunks mocks base method.
func (m *MockORMProvider) QueryChunks(ctx context.Context, parser kissorm.ChunkParser) error { func (m *MockSQLProvider) QueryChunks(ctx context.Context, parser kisssql.ChunkParser) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
ret := m.ctrl.Call(m, "QueryChunks", ctx, parser) ret := m.ctrl.Call(m, "QueryChunks", ctx, parser)
ret0, _ := ret[0].(error) ret0, _ := ret[0].(error)
@ -120,13 +120,13 @@ func (m *MockORMProvider) QueryChunks(ctx context.Context, parser kissorm.ChunkP
} }
// QueryChunks indicates an expected call of QueryChunks. // QueryChunks indicates an expected call of QueryChunks.
func (mr *MockORMProviderMockRecorder) QueryChunks(ctx, parser interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) QueryChunks(ctx, parser interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "QueryChunks", reflect.TypeOf((*MockORMProvider)(nil).QueryChunks), ctx, parser) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "QueryChunks", reflect.TypeOf((*MockSQLProvider)(nil).QueryChunks), ctx, parser)
} }
// QueryOne mocks base method. // QueryOne mocks base method.
func (m *MockORMProvider) QueryOne(ctx context.Context, record interface{}, query string, params ...interface{}) error { func (m *MockSQLProvider) QueryOne(ctx context.Context, record interface{}, query string, params ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx, record, query} varargs := []interface{}{ctx, record, query}
for _, a := range params { for _, a := range params {
@ -138,14 +138,14 @@ func (m *MockORMProvider) QueryOne(ctx context.Context, record interface{}, quer
} }
// QueryOne indicates an expected call of QueryOne. // QueryOne indicates an expected call of QueryOne.
func (mr *MockORMProviderMockRecorder) QueryOne(ctx, record, query interface{}, params ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) QueryOne(ctx, record, query interface{}, params ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx, record, query}, params...) varargs := append([]interface{}{ctx, record, query}, params...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "QueryOne", reflect.TypeOf((*MockORMProvider)(nil).QueryOne), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "QueryOne", reflect.TypeOf((*MockSQLProvider)(nil).QueryOne), varargs...)
} }
// Transaction mocks base method. // Transaction mocks base method.
func (m *MockORMProvider) Transaction(ctx context.Context, fn func(kissorm.ORMProvider) error) error { func (m *MockSQLProvider) Transaction(ctx context.Context, fn func(kisssql.SQLProvider) error) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
ret := m.ctrl.Call(m, "Transaction", ctx, fn) ret := m.ctrl.Call(m, "Transaction", ctx, fn)
ret0, _ := ret[0].(error) ret0, _ := ret[0].(error)
@ -153,13 +153,13 @@ func (m *MockORMProvider) Transaction(ctx context.Context, fn func(kissorm.ORMPr
} }
// Transaction indicates an expected call of Transaction. // Transaction indicates an expected call of Transaction.
func (mr *MockORMProviderMockRecorder) Transaction(ctx, fn interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Transaction(ctx, fn interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Transaction", reflect.TypeOf((*MockORMProvider)(nil).Transaction), ctx, fn) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Transaction", reflect.TypeOf((*MockSQLProvider)(nil).Transaction), ctx, fn)
} }
// Update mocks base method. // Update mocks base method.
func (m *MockORMProvider) Update(ctx context.Context, records ...interface{}) error { func (m *MockSQLProvider) Update(ctx context.Context, records ...interface{}) error {
m.ctrl.T.Helper() m.ctrl.T.Helper()
varargs := []interface{}{ctx} varargs := []interface{}{ctx}
for _, a := range records { for _, a := range records {
@ -171,8 +171,8 @@ func (m *MockORMProvider) Update(ctx context.Context, records ...interface{}) er
} }
// Update indicates an expected call of Update. // Update indicates an expected call of Update.
func (mr *MockORMProviderMockRecorder) Update(ctx interface{}, records ...interface{}) *gomock.Call { func (mr *MockSQLProviderMockRecorder) Update(ctx interface{}, records ...interface{}) *gomock.Call {
mr.mock.ctrl.T.Helper() mr.mock.ctrl.T.Helper()
varargs := append([]interface{}{ctx}, records...) varargs := append([]interface{}{ctx}, records...)
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Update", reflect.TypeOf((*MockORMProvider)(nil).Update), varargs...) return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Update", reflect.TypeOf((*MockSQLProvider)(nil).Update), varargs...)
} }

2
go.mod
View File

@ -1,4 +1,4 @@
module github.com/vingarcia/kissorm module github.com/vingarcia/kisssql
go 1.14 go 1.14

View File

@ -1,4 +1,4 @@
package kissorm package kisssql
import ( import (
"database/sql/driver" "database/sql/driver"

View File

@ -1,4 +1,4 @@
package kissorm package kisssql
import ( import (
"context" "context"
@ -8,12 +8,12 @@ import (
"strings" "strings"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/vingarcia/kissorm/structs" "github.com/vingarcia/kisssql/structs"
) )
// DB represents the kissorm client responsible for // DB represents the kisssql client responsible for
// interfacing with the "database/sql" package implementing // interfacing with the "database/sql" package implementing
// the Kissorm interface `ORMProvider`. // the KissSQL interface `SQLProvider`.
type DB struct { type DB struct {
driver string driver string
dialect dialect dialect dialect
@ -21,7 +21,7 @@ type DB struct {
db sqlProvider db sqlProvider
// Most dbs have a single primary key, // Most dbs have a single primary key,
// But in future kissorm should work with compound keys as well // But in future kisssql should work with compound keys as well
idCols []string idCols []string
insertMethod insertMethod insertMethod insertMethod
@ -41,7 +41,7 @@ const (
) )
// Config describes the optional arguments accepted // Config describes the optional arguments accepted
// by the kissorm.New() function. // by the kisssql.New() function.
type Config struct { type Config struct {
// MaxOpenCons defaults to 1 if not set // MaxOpenCons defaults to 1 if not set
MaxOpenConns int MaxOpenConns int
@ -55,7 +55,7 @@ type Config struct {
IDColumns []string IDColumns []string
} }
// New instantiates a new Kissorm client // New instantiates a new KissSQL client
func New( func New(
dbDriver string, dbDriver string,
connectionString string, connectionString string,
@ -124,7 +124,7 @@ func (c DB) Query(
slicePtr := reflect.ValueOf(records) slicePtr := reflect.ValueOf(records)
slicePtrType := slicePtr.Type() slicePtrType := slicePtr.Type()
if slicePtrType.Kind() != reflect.Ptr { if slicePtrType.Kind() != reflect.Ptr {
return fmt.Errorf("kissorm: expected to receive a pointer to slice of structs, but got: %T", records) return fmt.Errorf("kisssql: expected to receive a pointer to slice of structs, but got: %T", records)
} }
sliceType := slicePtrType.Elem() sliceType := slicePtrType.Elem()
slice := slicePtr.Elem() slice := slicePtr.Elem()
@ -198,11 +198,11 @@ func (c DB) QueryOne(
) error { ) error {
t := reflect.TypeOf(record) t := reflect.TypeOf(record)
if t.Kind() != reflect.Ptr { if t.Kind() != reflect.Ptr {
return fmt.Errorf("kissorm: expected to receive a pointer to struct, but got: %T", record) return fmt.Errorf("kisssql: expected to receive a pointer to struct, but got: %T", record)
} }
t = t.Elem() t = t.Elem()
if t.Kind() != reflect.Struct { if t.Kind() != reflect.Struct {
return fmt.Errorf("kissorm: expected to receive a pointer to struct, but got: %T", record) return fmt.Errorf("kisssql: expected to receive a pointer to struct, but got: %T", record)
} }
rows, err := c.db.QueryContext(ctx, query, params...) rows, err := c.db.QueryContext(ctx, query, params...)
@ -509,7 +509,7 @@ func (c DB) Delete(
func normalizeIDsAsMaps(idNames []string, ids []interface{}) ([]map[string]interface{}, error) { func normalizeIDsAsMaps(idNames []string, ids []interface{}) ([]map[string]interface{}, error) {
if len(idNames) == 0 { if len(idNames) == 0 {
return nil, fmt.Errorf("internal kissorm error: missing idNames") return nil, fmt.Errorf("internal kisssql error: missing idNames")
} }
idMaps := []map[string]interface{}{} idMaps := []map[string]interface{}{}
@ -698,7 +698,7 @@ func (c DB) Exec(ctx context.Context, query string, params ...interface{}) error
} }
// Transaction just runs an SQL command on the database returning no rows. // Transaction just runs an SQL command on the database returning no rows.
func (c DB) Transaction(ctx context.Context, fn func(ORMProvider) error) error { func (c DB) Transaction(ctx context.Context, fn func(SQLProvider) error) error {
switch db := c.db.(type) { switch db := c.db.(type) {
case *sql.Tx: case *sql.Tx:
return fn(c) return fn(c)
@ -736,7 +736,7 @@ func (c DB) Transaction(ctx context.Context, fn func(ORMProvider) error) error {
return tx.Commit() return tx.Commit()
default: default:
return fmt.Errorf("unexpected error on kissorm: db attribute has an invalid type") return fmt.Errorf("unexpected error on kisssql: db attribute has an invalid type")
} }
} }
@ -789,14 +789,14 @@ func scanRows(rows *sql.Rows, record interface{}) error {
v := reflect.ValueOf(record) v := reflect.ValueOf(record)
t := v.Type() t := v.Type()
if t.Kind() != reflect.Ptr { if t.Kind() != reflect.Ptr {
return fmt.Errorf("kissorm: expected record to be a pointer to struct, but got: %T", record) return fmt.Errorf("kisssql: expected record to be a pointer to struct, but got: %T", record)
} }
v = v.Elem() v = v.Elem()
t = t.Elem() t = t.Elem()
if t.Kind() != reflect.Struct { if t.Kind() != reflect.Struct {
return fmt.Errorf("kissorm: expected record to be a pointer to struct, but got: %T", record) return fmt.Errorf("kisssql: expected record to be a pointer to struct, but got: %T", record)
} }
info := structs.GetTagInfo(t) info := structs.GetTagInfo(t)

View File

@ -1,4 +1,4 @@
package kissorm package kisssql
import ( import (
"context" "context"
@ -12,15 +12,15 @@ import (
"github.com/ditointernet/go-assert" "github.com/ditointernet/go-assert"
_ "github.com/lib/pq" _ "github.com/lib/pq"
_ "github.com/mattn/go-sqlite3" _ "github.com/mattn/go-sqlite3"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
) )
type User struct { type User struct {
ID uint `kissorm:"id"` ID uint `kisssql:"id"`
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
Address Address `kissorm:"address,json"` Address Address `kisssql:"address,json"`
} }
type Address struct { type Address struct {
@ -702,9 +702,9 @@ func TestUpdate(t *testing.T) {
c := newTestDB(db, driver, "users") c := newTestDB(db, driver, "users")
type partialUser struct { type partialUser struct {
ID uint `kissorm:"id"` ID uint `kisssql:"id"`
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
u := partialUser{ u := partialUser{
Name: "Letícia", Name: "Letícia",
@ -743,9 +743,9 @@ func TestUpdate(t *testing.T) {
c := newTestDB(db, driver, "users") c := newTestDB(db, driver, "users")
type partialUser struct { type partialUser struct {
ID uint `kissorm:"id"` ID uint `kisssql:"id"`
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
u := partialUser{ u := partialUser{
Name: "Letícia", Name: "Letícia",
@ -1210,7 +1210,7 @@ func TestTransaction(t *testing.T) {
_ = c.Insert(ctx, &User{Name: "User2"}) _ = c.Insert(ctx, &User{Name: "User2"})
var users []User var users []User
err = c.Transaction(ctx, func(db ORMProvider) error { err = c.Transaction(ctx, func(db SQLProvider) error {
db.Query(ctx, &users, "SELECT * FROM users ORDER BY id ASC") db.Query(ctx, &users, "SELECT * FROM users ORDER BY id ASC")
return nil return nil
}) })
@ -1238,7 +1238,7 @@ func TestTransaction(t *testing.T) {
_ = c.Insert(ctx, &u1) _ = c.Insert(ctx, &u1)
_ = c.Insert(ctx, &u2) _ = c.Insert(ctx, &u2)
err = c.Transaction(ctx, func(db ORMProvider) error { err = c.Transaction(ctx, func(db SQLProvider) error {
err = db.Insert(ctx, &User{Name: "User3"}) err = db.Insert(ctx, &User{Name: "User3"})
assert.Equal(t, nil, err) assert.Equal(t, nil, err)
err = db.Insert(ctx, &User{Name: "User4"}) err = db.Insert(ctx, &User{Name: "User4"})
@ -1309,11 +1309,11 @@ func TestScanRows(t *testing.T) {
assert.Equal(t, true, rows.Next()) assert.Equal(t, true, rows.Next())
var user struct { var user struct {
ID int `kissorm:"id"` ID int `kisssql:"id"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
// Omitted for testing purposes: // Omitted for testing purposes:
// Name string `kissorm:"name"` // Name string `kisssql:"name"`
} }
err = scanRows(rows, &user) err = scanRows(rows, &user)
assert.Equal(t, nil, err) assert.Equal(t, nil, err)
@ -1379,8 +1379,8 @@ func TestScanRows(t *testing.T) {
} }
var connectionString = map[string]string{ var connectionString = map[string]string{
"postgres": "host=localhost port=5432 user=postgres password=postgres dbname=kissorm sslmode=disable", "postgres": "host=localhost port=5432 user=postgres password=postgres dbname=kisssql sslmode=disable",
"sqlite3": "/tmp/kissorm.db", "sqlite3": "/tmp/kisssql.db",
} }
func createTable(driver string) error { func createTable(driver string) error {

View File

@ -1,11 +1,11 @@
package kissorm package kisssql
import "context" import "context"
var _ ORMProvider = MockORMProvider{} var _ SQLProvider = MockSQLProvider{}
// MockORMProvider ... // MockSQLProvider ...
type MockORMProvider struct { type MockSQLProvider struct {
InsertFn func(ctx context.Context, records ...interface{}) error InsertFn func(ctx context.Context, records ...interface{}) error
DeleteFn func(ctx context.Context, ids ...interface{}) error DeleteFn func(ctx context.Context, ids ...interface{}) error
UpdateFn func(ctx context.Context, records ...interface{}) error UpdateFn func(ctx context.Context, records ...interface{}) error
@ -15,45 +15,45 @@ type MockORMProvider struct {
QueryChunksFn func(ctx context.Context, parser ChunkParser) error QueryChunksFn func(ctx context.Context, parser ChunkParser) error
ExecFn func(ctx context.Context, query string, params ...interface{}) error ExecFn func(ctx context.Context, query string, params ...interface{}) error
TransactionFn func(ctx context.Context, fn func(db ORMProvider) error) error TransactionFn func(ctx context.Context, fn func(db SQLProvider) error) error
} }
// Insert ... // Insert ...
func (m MockORMProvider) Insert(ctx context.Context, records ...interface{}) error { func (m MockSQLProvider) Insert(ctx context.Context, records ...interface{}) error {
return m.InsertFn(ctx, records...) return m.InsertFn(ctx, records...)
} }
// Delete ... // Delete ...
func (m MockORMProvider) Delete(ctx context.Context, ids ...interface{}) error { func (m MockSQLProvider) Delete(ctx context.Context, ids ...interface{}) error {
return m.DeleteFn(ctx, ids...) return m.DeleteFn(ctx, ids...)
} }
// Update ... // Update ...
func (m MockORMProvider) Update(ctx context.Context, records ...interface{}) error { func (m MockSQLProvider) Update(ctx context.Context, records ...interface{}) error {
return m.UpdateFn(ctx, records...) return m.UpdateFn(ctx, records...)
} }
// Query ... // Query ...
func (m MockORMProvider) Query(ctx context.Context, records interface{}, query string, params ...interface{}) error { func (m MockSQLProvider) Query(ctx context.Context, records interface{}, query string, params ...interface{}) error {
return m.QueryFn(ctx, records, query, params...) return m.QueryFn(ctx, records, query, params...)
} }
// QueryOne ... // QueryOne ...
func (m MockORMProvider) QueryOne(ctx context.Context, record interface{}, query string, params ...interface{}) error { func (m MockSQLProvider) QueryOne(ctx context.Context, record interface{}, query string, params ...interface{}) error {
return m.QueryOneFn(ctx, record, query, params...) return m.QueryOneFn(ctx, record, query, params...)
} }
// QueryChunks ... // QueryChunks ...
func (m MockORMProvider) QueryChunks(ctx context.Context, parser ChunkParser) error { func (m MockSQLProvider) QueryChunks(ctx context.Context, parser ChunkParser) error {
return m.QueryChunksFn(ctx, parser) return m.QueryChunksFn(ctx, parser)
} }
// Exec ... // Exec ...
func (m MockORMProvider) Exec(ctx context.Context, query string, params ...interface{}) error { func (m MockSQLProvider) Exec(ctx context.Context, query string, params ...interface{}) error {
return m.ExecFn(ctx, query, params...) return m.ExecFn(ctx, query, params...)
} }
// Transaction ... // Transaction ...
func (m MockORMProvider) Transaction(ctx context.Context, fn func(db ORMProvider) error) error { func (m MockSQLProvider) Transaction(ctx context.Context, fn func(db SQLProvider) error) error {
return m.TransactionFn(ctx, fn) return m.TransactionFn(ctx, fn)
} }

View File

@ -68,7 +68,7 @@ func getCachedTagInfo(tagInfoCache map[reflect.Type]structInfo, key reflect.Type
} }
// StructToMap converts any struct type to a map based on // StructToMap converts any struct type to a map based on
// the tag named `kissorm`, i.e. `kissorm:"map_key_name"` // the tag named `kisssql`, i.e. `kisssql:"map_key_name"`
// //
// Valid pointers are dereferenced and copied to the map, // Valid pointers are dereferenced and copied to the map,
// null pointers are ignored. // null pointers are ignored.
@ -111,7 +111,7 @@ func StructToMap(obj interface{}) (map[string]interface{}, error) {
// FillStructWith is meant to be used on unit tests to mock // FillStructWith is meant to be used on unit tests to mock
// the response from the database. // the response from the database.
// //
// The first argument is any struct you are passing to a kissorm func, // The first argument is any struct you are passing to a kisssql func,
// and the second is a map representing a database row you want // and the second is a map representing a database row you want
// to use to update this struct. // to use to update this struct.
func FillStructWith(record interface{}, dbRow map[string]interface{}) error { func FillStructWith(record interface{}, dbRow map[string]interface{}) error {
@ -139,7 +139,7 @@ func FillStructWith(record interface{}, dbRow map[string]interface{}) error {
for colName, rawSrc := range dbRow { for colName, rawSrc := range dbRow {
fieldInfo := info.ByName(colName) fieldInfo := info.ByName(colName)
if !fieldInfo.Valid { if !fieldInfo.Valid {
// Ignore columns not tagged with `kissorm:"..."` // Ignore columns not tagged with `kisssql:"..."`
continue continue
} }
@ -242,7 +242,7 @@ func (p PtrConverter) Convert(destType reflect.Type) (reflect.Value, error) {
// FillSliceWith is meant to be used on unit tests to mock // FillSliceWith is meant to be used on unit tests to mock
// the response from the database. // the response from the database.
// //
// The first argument is any slice of structs you are passing to a kissorm func, // The first argument is any slice of structs you are passing to a kisssql func,
// and the second is a slice of maps representing the database rows you want // and the second is a slice of maps representing the database rows you want
// to use to update this struct. // to use to update this struct.
func FillSliceWith(entities interface{}, dbRows []map[string]interface{}) error { func FillSliceWith(entities interface{}, dbRows []map[string]interface{}) error {
@ -293,7 +293,7 @@ func getTagNames(t reflect.Type) structInfo {
byName: map[string]*fieldInfo{}, byName: map[string]*fieldInfo{},
} }
for i := 0; i < t.NumField(); i++ { for i := 0; i < t.NumField(); i++ {
name := t.Field(i).Tag.Get("kissorm") name := t.Field(i).Tag.Get("kisssql")
if name == "" { if name == "" {
continue continue
} }

View File

@ -4,13 +4,13 @@ import (
"testing" "testing"
"github.com/ditointernet/go-assert" "github.com/ditointernet/go-assert"
"github.com/vingarcia/kissorm/nullable" "github.com/vingarcia/kisssql/nullable"
) )
func TestStructToMap(t *testing.T) { func TestStructToMap(t *testing.T) {
type S1 struct { type S1 struct {
Name string `kissorm:"name_attr"` Name string `kisssql:"name_attr"`
Age int `kissorm:"age_attr"` Age int `kisssql:"age_attr"`
} }
t.Run("should convert plain structs to maps", func(t *testing.T) { t.Run("should convert plain structs to maps", func(t *testing.T) {
m, err := StructToMap(S1{ m, err := StructToMap(S1{
@ -39,8 +39,8 @@ func TestStructToMap(t *testing.T) {
}) })
type S2 struct { type S2 struct {
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
t.Run("should not ignore not nil pointers", func(t *testing.T) { t.Run("should not ignore not nil pointers", func(t *testing.T) {
@ -72,8 +72,8 @@ func TestStructToMap(t *testing.T) {
func TestFillStructWith(t *testing.T) { func TestFillStructWith(t *testing.T) {
t.Run("should fill a struct correctly", func(t *testing.T) { t.Run("should fill a struct correctly", func(t *testing.T) {
var user struct { var user struct {
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
} }
err := FillStructWith(&user, map[string]interface{}{ err := FillStructWith(&user, map[string]interface{}{
"name": "Breno", "name": "Breno",
@ -87,8 +87,8 @@ func TestFillStructWith(t *testing.T) {
t.Run("should fill ptr fields with ptr values", func(t *testing.T) { t.Run("should fill ptr fields with ptr values", func(t *testing.T) {
var user struct { var user struct {
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
err := FillStructWith(&user, map[string]interface{}{ err := FillStructWith(&user, map[string]interface{}{
"name": nullable.String("Breno"), "name": nullable.String("Breno"),
@ -102,8 +102,8 @@ func TestFillStructWith(t *testing.T) {
t.Run("should fill ptr fields with non-ptr values", func(t *testing.T) { t.Run("should fill ptr fields with non-ptr values", func(t *testing.T) {
var user struct { var user struct {
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
err := FillStructWith(&user, map[string]interface{}{ err := FillStructWith(&user, map[string]interface{}{
"name": "Breno", "name": "Breno",
@ -117,8 +117,8 @@ func TestFillStructWith(t *testing.T) {
t.Run("should fill non ptr fields with ptr values", func(t *testing.T) { t.Run("should fill non ptr fields with ptr values", func(t *testing.T) {
var user struct { var user struct {
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
} }
err := FillStructWith(&user, map[string]interface{}{ err := FillStructWith(&user, map[string]interface{}{
"name": nullable.String("Breno"), "name": nullable.String("Breno"),
@ -132,8 +132,8 @@ func TestFillStructWith(t *testing.T) {
t.Run("should fill ptr fields with nil when necessary", func(t *testing.T) { t.Run("should fill ptr fields with nil when necessary", func(t *testing.T) {
var user struct { var user struct {
Name *string `kissorm:"name"` Name *string `kisssql:"name"`
Age *int `kissorm:"age"` Age *int `kisssql:"age"`
} }
err := FillStructWith(&user, map[string]interface{}{ err := FillStructWith(&user, map[string]interface{}{
"name": nil, "name": nil,
@ -147,8 +147,8 @@ func TestFillStructWith(t *testing.T) {
t.Run("should interpret nil fields as zero values when necessary", func(t *testing.T) { t.Run("should interpret nil fields as zero values when necessary", func(t *testing.T) {
var user struct { var user struct {
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
} }
user.Name = "not empty" user.Name = "not empty"
user.Age = 42 user.Age = 42
@ -165,9 +165,9 @@ func TestFillStructWith(t *testing.T) {
t.Run("should ignore extra or missing fields", func(t *testing.T) { t.Run("should ignore extra or missing fields", func(t *testing.T) {
var user struct { var user struct {
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
Missing string `kissorm:"missing"` Missing string `kisssql:"missing"`
} }
user.Missing = "should be untouched" user.Missing = "should be untouched"
@ -187,8 +187,8 @@ func TestFillStructWith(t *testing.T) {
func TestFillSliceWith(t *testing.T) { func TestFillSliceWith(t *testing.T) {
t.Run("should fill a list correctly", func(t *testing.T) { t.Run("should fill a list correctly", func(t *testing.T) {
var users []struct { var users []struct {
Name string `kissorm:"name"` Name string `kisssql:"name"`
Age int `kissorm:"age"` Age int `kisssql:"age"`
} }
err := FillSliceWith(&users, []map[string]interface{}{ err := FillSliceWith(&users, []map[string]interface{}{
{ {