Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(string): added nullable String/Null for sql/json #47

Merged
merged 5 commits into from
Jun 12, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## [1.5.1]
- !fix(orderby): use BuildOption instead of allowedColumns (#46)
- feat(string): added nullable String/Null for sql/json (#47)

## [1.5.0] - 2024-04-30
### Changed
Expand Down
2 changes: 1 addition & 1 deletion bool_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ func TestBool(t *testing.T) {
d, err := sql.Open("sqlite3", "file::memory:")
require.NoError(t, err)

_, err = d.Exec("CREATE TABLE `users` (`id` id NOT NULL,`status` BIT(1), PRIMARY KEY (`id`))")
_, err = d.Exec("CREATE TABLE `users` (`id` int NOT NULL,`status` BIT(1), PRIMARY KEY (`id`))")
require.NoError(t, err)

result, err := d.Exec("INSERT INTO `users`(`id`, `status`) VALUES(?, ?)", 10, Bool(true))
Expand Down
61 changes: 61 additions & 0 deletions null.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
package sqle

import (
"database/sql"
"database/sql/driver"
"encoding/json"
)

var nullJsonBytes = []byte("null")

const nullJson = "null"

type Null[T any] struct {
sql.Null[T]
}

func NewNull[T any](v T, valid bool) Null[T] {
return Null[T]{Null: sql.Null[T]{V: v, Valid: valid}}
}

// Scan implements the [sql.Scanner] interface.
func (t *Null[T]) Scan(value any) error { // skipcq: GO-W1029
return t.Null.Scan(value)
}

// Value implements the [driver.Valuer] interface.
func (t Null[T]) Value() (driver.Value, error) { // skipcq: GO-W1029
return t.Null.Value()
}

// TValue returns the underlying value of the Null struct.
func (t *Null[T]) TValue() T { // skipcq: GO-W1029
return t.Null.V
}

// MarshalJSON implements the json.Marshaler interface
func (t Null[T]) MarshalJSON() ([]byte, error) { // skipcq: GO-W1029
if t.Valid {
return json.Marshal(t.Null.V)

Check warning on line 39 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L37-L39

Added lines #L37 - L39 were not covered by tests
}
return nullJsonBytes, nil

Check warning on line 41 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L41

Added line #L41 was not covered by tests
}

// UnmarshalJSON implements the json.Unmarshaler interface
func (t *Null[T]) UnmarshalJSON(data []byte) error { // skipcq: GO-W1029
if len(data) == 0 || string(data) == nullJson {
t.Null.Valid = false
return nil

Check warning on line 48 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L45-L48

Added lines #L45 - L48 were not covered by tests
}

var v T
err := json.Unmarshal(data, &v)
if err != nil {
return err

Check warning on line 54 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L51-L54

Added lines #L51 - L54 were not covered by tests
}

t.Null.V = v
t.Null.Valid = true

Check warning on line 58 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L57-L58

Added lines #L57 - L58 were not covered by tests

return nil

Check warning on line 60 in null.go

View check run for this annotation

Codecov / codecov/patch

null.go#L60

Added line #L60 was not covered by tests
}
56 changes: 56 additions & 0 deletions string.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
package sqle

import (
"database/sql/driver"
"encoding/json"
)

type String struct {
Null[string]
}

func NewString(s string) String {
return String{Null: NewNull(s, true)}
}

// Scan implements the [sql.Scanner] interface.
func (t *String) Scan(value any) error { // skipcq: GO-W1029
return t.Null.Scan(value)
}

// Value implements the [driver.Valuer] interface.
func (t String) Value() (driver.Value, error) { // skipcq: GO-W1029
return t.Null.Value()
}

// Time returns the underlying time.Time value of the Time struct.
func (t *String) String() string { // skipcq: GO-W1029
return t.TValue()
}

// MarshalJSON implements the json.Marshaler interface
func (t String) MarshalJSON() ([]byte, error) { // skipcq: GO-W1029
if t.Valid {
return json.Marshal(t.TValue())
}
return nullJsonBytes, nil
}

// UnmarshalJSON implements the json.Unmarshaler interface
func (t *String) UnmarshalJSON(data []byte) error { // skipcq: GO-W1029
if len(data) == 0 || string(data) == nullJson {
t.Null.Valid = false
return nil
}

var v string
err := json.Unmarshal(data, &v)
if err != nil {
return err

Check warning on line 49 in string.go

View check run for this annotation

Codecov / codecov/patch

string.go#L49

Added line #L49 was not covered by tests
}

t.Null.V = v
t.Null.Valid = true

return nil
}
112 changes: 112 additions & 0 deletions string_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
package sqle

import (
"database/sql"
"encoding/json"
"testing"

"github.com/stretchr/testify/require"
)

func TestStringInSQL(t *testing.T) {

v := "has value"
d, err := sql.Open("sqlite3", "file::memory:")
require.NoError(t, err)

_, err = d.Exec("CREATE TABLE `strings` (`id` int NOT NULL,`name` varchar(125), PRIMARY KEY (`id`))")
require.NoError(t, err)

result, err := d.Exec("INSERT INTO `strings`(`id`) VALUES(?)", 10)
require.NoError(t, err)

rows, err := result.RowsAffected()
require.NoError(t, err)
require.Equal(t, int64(1), rows)

result, err = d.Exec("INSERT INTO `strings`(`id`, `name`) VALUES(?, ?)", 20, v)
require.NoError(t, err)

rows, err = result.RowsAffected()
require.NoError(t, err)
require.Equal(t, int64(1), rows)

var v10 String
err = d.QueryRow("SELECT `name` FROM `strings` WHERE id=?", 10).Scan(&v10)
require.NoError(t, err)

require.EqualValues(t, false, v10.Valid)

var v20 String
err = d.QueryRow("SELECT `name` FROM `strings` WHERE id=?", 20).Scan(&v20)
require.NoError(t, err)

require.EqualValues(t, true, v20.Valid)
require.EqualValues(t, v, v20.String())

result, err = d.Exec("INSERT INTO `strings`(`id`,`name`) VALUES(?, ?)", 11, v10)
require.NoError(t, err)

rows, err = result.RowsAffected()
require.NoError(t, err)
require.Equal(t, int64(1), rows)

result, err = d.Exec("INSERT INTO `strings`(`id`, `name`) VALUES(?, ?)", 21, v20)
require.NoError(t, err)

rows, err = result.RowsAffected()
require.NoError(t, err)
require.Equal(t, int64(1), rows)

var v11 String
err = d.QueryRow("SELECT `name` FROM `strings` WHERE id=?", 11).Scan(&v11)
require.NoError(t, err)

require.EqualValues(t, false, v11.Valid)

var v21 String
err = d.QueryRow("SELECT `name` FROM `strings` WHERE id=?", 21).Scan(&v21)
require.NoError(t, err)

require.EqualValues(t, true, v21.Valid)
require.EqualValues(t, v, v21.String())

}

func TestStringInJSON(t *testing.T) {

sysString := "has value"

bufSysString, err := json.Marshal(sysString)
require.NoError(t, err)

sqleString := NewString(sysString)

bufSqleString, err := json.Marshal(sqleString)
require.NoError(t, err)

require.Equal(t, bufSysString, bufSqleString)

var jsSqleString String
// Unmarshal sqle.Time from time.Time json bytes
err = json.Unmarshal(bufSysString, &jsSqleString)
require.NoError(t, err)

require.Equal(t, sysString, jsSqleString.String())
require.Equal(t, true, jsSqleString.Valid)

var jsSysString string
// Unmarshal time.Time from sqle.Time json bytes
err = json.Unmarshal(bufSqleString, &jsSysString)
require.NoError(t, err)
require.Equal(t, sysString, jsSysString)

var nullString String
err = json.Unmarshal([]byte("null"), &nullString)
require.NoError(t, err)
require.Equal(t, false, nullString.Valid)

bufNull, err := json.Marshal(nullString)
require.NoError(t, err)
require.Equal(t, []byte("null"), bufNull)
}
8 changes: 2 additions & 6 deletions time.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,6 @@ import (
"time"
)

var nullTimeJsonBytes = []byte("null")

const nullTimeJson = "null"

// Time represents a nullable time value.
type Time struct {
sql.NullTime
Expand Down Expand Up @@ -41,12 +37,12 @@ func (t Time) MarshalJSON() ([]byte, error) { // skipcq: GO-W1029
if t.Valid {
return json.Marshal(t.NullTime.Time)
}
return nullTimeJsonBytes, nil
return nullJsonBytes, nil
}

// UnmarshalJSON implements the json.Unmarshaler interface
func (t *Time) UnmarshalJSON(data []byte) error { // skipcq: GO-W1029
if len(data) == 0 || string(data) == nullTimeJson {
if len(data) == 0 || string(data) == nullJson {
t.NullTime.Time = time.Time{}
t.NullTime.Valid = false
return nil
Expand Down
Loading