Compare commits

..

No commits in common. "main" and "improve-coverage" have entirely different histories.

7 changed files with 180 additions and 16 deletions

View File

@ -15,12 +15,12 @@ jobs:
strategy:
fail-fast: false
matrix:
go: ['1.17', '1.18', '1.19']
go: ['1.13', '1.14', '1.15', '1.16']
steps:
- id: go
name: Set up Go
uses: actions/setup-go@v3
uses: actions/setup-go@v1
with:
go-version: ${{ matrix.go }}

2
.gitignore vendored
View File

@ -22,5 +22,3 @@ _testmain.go
*.exe
*.test
*.prof
go.*

View File

@ -1,10 +0,0 @@
all:
@echo
@echo
test:
redomod:
rm -f go.*
GO111MODULE= go mod init
GO111MODULE= go mod tidy

9
go.mod Normal file
View File

@ -0,0 +1,9 @@
module github.com/alexflint/go-scalar
go 1.15
require (
github.com/davecgh/go-spew v1.1.1 // indirect
github.com/pmezard/go-difflib v1.0.0 // indirect
github.com/stretchr/testify v1.2.2
)

6
go.sum Normal file
View File

@ -0,0 +1,6 @@
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w=
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=

View File

@ -108,13 +108,13 @@ func ParseValue(v reflect.Value, s string) error {
}
v.SetBool(x)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
x, err := strconv.ParseInt(s, 0, v.Type().Bits())
x, err := strconv.ParseInt(s, 10, v.Type().Bits())
if err != nil {
return err
}
v.SetInt(x)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
x, err := strconv.ParseUint(s, 0, v.Type().Bits())
x, err := strconv.ParseUint(s, 10, v.Type().Bits())
if err != nil {
return err
}

161
scalar_test.go Normal file
View File

@ -0,0 +1,161 @@
package scalar
import (
"net"
"net/mail"
"net/url"
"reflect"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
type textUnmarshaler struct {
val int
}
func (f *textUnmarshaler) UnmarshalText(b []byte) error {
f.val = len(b)
return nil
}
func assertParse(t *testing.T, expected interface{}, str string) {
v := reflect.New(reflect.TypeOf(expected)).Elem()
err := ParseValue(v, str)
if assert.NoError(t, err) {
assert.Equal(t, expected, v.Interface())
}
ptr := reflect.New(reflect.PtrTo(reflect.TypeOf(expected))).Elem()
err = ParseValue(ptr, str)
if assert.NoError(t, err) {
assert.Equal(t, expected, ptr.Elem().Interface())
}
assert.True(t, CanParse(v.Type()))
assert.True(t, CanParse(ptr.Type()))
}
func TestParseValue(t *testing.T) {
// strings
assertParse(t, "abc", "abc")
// booleans
assertParse(t, true, "true")
assertParse(t, false, "false")
// integers
assertParse(t, int(123), "123")
assertParse(t, int8(123), "123")
assertParse(t, int16(123), "123")
assertParse(t, int32(123), "123")
assertParse(t, int64(123), "123")
// unsigned integers
assertParse(t, uint(123), "123")
assertParse(t, byte(123), "123")
assertParse(t, uint8(123), "123")
assertParse(t, uint16(123), "123")
assertParse(t, uint32(123), "123")
assertParse(t, uint64(123), "123")
assertParse(t, uintptr(123), "123")
assertParse(t, rune(123), "123")
// floats
assertParse(t, float32(123), "123")
assertParse(t, float64(123), "123")
// durations
assertParse(t, 3*time.Hour+15*time.Minute, "3h15m")
// IP addresses
assertParse(t, net.IPv4(1, 2, 3, 4), "1.2.3.4")
// email addresses
assertParse(t, mail.Address{Address: "joe@example.com"}, "joe@example.com")
// MAC addresses
assertParse(t, net.HardwareAddr("\x01\x23\x45\x67\x89\xab"), "01:23:45:67:89:ab")
// MAC addresses
assertParse(t, net.HardwareAddr("\x01\x23\x45\x67\x89\xab"), "01:23:45:67:89:ab")
// URL
assertParse(t, url.URL{Scheme: "https", Host: "example.com", Path: "/a/b/c"}, "https://example.com/a/b/c")
// custom text unmarshaler
assertParse(t, textUnmarshaler{3}, "abc")
}
func TestParseErrors(t *testing.T) {
var err error
// this should fail because the pointer is nil and will not be settable
var p *int
err = ParseValue(reflect.ValueOf(p), "123")
assert.Equal(t, errPtrNotSettable, err)
// this should fail because the value will not be settable
var v int
err = ParseValue(reflect.ValueOf(v), "123")
assert.Equal(t, errNotSettable, err)
// this should fail due to a malformed boolean
var b bool
err = ParseValue(reflect.ValueOf(&b), "malformed")
assert.Error(t, err)
// this should fail due to a malformed boolean
var i int
err = ParseValue(reflect.ValueOf(&i), "malformed")
assert.Error(t, err)
// this should fail due to a malformed boolean
var u uint
err = ParseValue(reflect.ValueOf(&u), "malformed")
assert.Error(t, err)
// this should fail due to a malformed boolean
var f float64
err = ParseValue(reflect.ValueOf(&f), "malformed")
assert.Error(t, err)
// this should fail due to a malformed time duration
var d time.Duration
err = ParseValue(reflect.ValueOf(&d), "malfomed")
assert.Error(t, err)
// this should fail due to a malformed email address
var email mail.Address
err = ParseValue(reflect.ValueOf(&email), "malfomed")
assert.Error(t, err)
// this should fail due to a malformed time duration
var mac net.HardwareAddr
err = ParseValue(reflect.ValueOf(&mac), "malfomed")
assert.Error(t, err)
// this should fail due to a malformed time duration
var url url.URL
err = ParseValue(reflect.ValueOf(&url), "$:")
assert.Error(t, err)
// this should fail due to an unsupported type
var x struct{}
err = ParseValue(reflect.ValueOf(&x), "$")
assert.Error(t, err)
}
func TestParse(t *testing.T) {
var v int
err := Parse(&v, "123")
require.NoError(t, err)
assert.Equal(t, 123, v)
}
func TestCanParseReturnsFalse(t *testing.T) {
var x struct{}
assert.Equal(t, false, CanParse(reflect.TypeOf(x)))
}