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

RFC: Fix a few pain points #6

Open
wants to merge 4 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all 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
56 changes: 0 additions & 56 deletions crazy_test.go

This file was deleted.

25 changes: 0 additions & 25 deletions error_helpers.go

This file was deleted.

29 changes: 23 additions & 6 deletions errors.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,9 @@ import (
"reflect"
)

// TypeError is an error type returned when param has difficulty deserializing a
// parameter value.
type TypeError struct {
// ValueError is an error type returned when param has difficulty deserializing
// a parameter value.
type ValueError struct {
// The key that was in error.
Key string
// The type that was expected.
Expand All @@ -17,9 +17,9 @@ type TypeError struct {
Err error
}

func (t TypeError) Error() string {
return fmt.Sprintf("param: error parsing key %q as %v: %v", t.Key, t.Type,
t.Err)
func (v ValueError) Error() string {
return fmt.Sprintf("param: error parsing key %q as %v: %v",
v.Key, v.Type, v.Err)
}

// SingletonError is an error type returned when a parameter is passed multiple
Expand Down Expand Up @@ -110,3 +110,20 @@ func (k KeyError) Error() string {
return fmt.Sprintf("param: error parsing key %q: unknown field %q on "+
"struct %q of type %v", k.FullKey, k.Field, k.Key, k.Type)
}

// InvalidParseError describes an invalid argument passed to Parse. It is always
// the result of programmer error.
type InvalidParseError struct {
Type reflect.Type
Hint string
}

func (err InvalidParseError) Error() string {
msg := fmt.Sprintf("param/parse: unsupported type %v", err.Type)

if err.Hint != "" {
msg += ": " + err.Hint
}

return msg
}
1 change: 1 addition & 0 deletions go.mod
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
module github.com/goji/param
32 changes: 18 additions & 14 deletions param.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,34 +26,38 @@ import (
)

// Parse the given arguments into the the given pointer to a struct object.
func Parse(params url.Values, target interface{}) (err error) {
func Parse(params url.Values, target interface{}) error {
v := reflect.ValueOf(target)

defer func() {
if r := recover(); r != nil {
var ok bool
err, ok = r.(error)
if !ok {
panic(err)
}
if v.Kind() != reflect.Ptr || v.Elem().Kind() != reflect.Struct {
hint := "target must be a pointer to a struct"
if v.Kind() == reflect.Ptr && v.IsNil() {
hint = "target may not be a nil pointer"
}
}()

if v.Kind() != reflect.Ptr || v.Elem().Kind() != reflect.Struct {
pebkac("Target of param.Parse must be a pointer to a struct. "+
"We instead were passed a %v", v.Type())
return InvalidParseError{
Type: v.Type(),
Hint: hint,
}
}

el := v.Elem()
t := el.Type()
cache := cacheStruct(t)
cache, err := cacheStruct(t)
if err != nil {
return err
}

for key, values := range params {
sk, keytail := key, ""
if i := strings.IndexRune(key, '['); i != -1 {
sk, keytail = sk[:i], sk[i:]
}
parseStructField(cache, key, sk, keytail, values, el)

err := parseStructField(cache, key, sk, keytail, values, el)
if err != nil {
return err
}
}

return nil
Expand Down
141 changes: 130 additions & 11 deletions param_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -71,25 +71,31 @@ func init() {
testTime, _ = time.Parse(time.RFC3339, testTimeString)
}

func singletonErrors(t *testing.T, field, valid, invalid string) {
func singletonErrorsWithInvalid(t *testing.T, field, valid, invalid string) {
e := Everything{}

err := Parse(url.Values{field: {invalid}}, &e)
if err == nil {
t.Errorf("Expected error parsing %q as %s", invalid, field)
}

err = Parse(url.Values{field + "[]": {valid}}, &e)
singletonErrors(t, field, valid)
}

func singletonErrors(t *testing.T, field, val string) {
e := Everything{}

err := Parse(url.Values{field + "[]": {val}}, &e)
if err == nil {
t.Errorf("Expected error parsing nested %s", field)
}

err = Parse(url.Values{field + "[nested]": {valid}}, &e)
err = Parse(url.Values{field + "[nested]": {val}}, &e)
if err == nil {
t.Errorf("Expected error parsing nested %s", field)
}

err = Parse(url.Values{field: {valid, valid}}, &e)
err = Parse(url.Values{field: {val, val}}, &e)
if err == nil {
t.Errorf("Expected error passing %s twice", field)
}
Expand Down Expand Up @@ -122,7 +128,7 @@ func TestBoolTyped(t *testing.T) {

func TestBoolErrors(t *testing.T) {
t.Parallel()
singletonErrors(t, "Bool", "true", "llama")
singletonErrorsWithInvalid(t, "Bool", "true", "llama")
}

var intAnswers = map[string]int{
Expand Down Expand Up @@ -158,7 +164,7 @@ func TestIntTyped(t *testing.T) {

func TestIntErrors(t *testing.T) {
t.Parallel()
singletonErrors(t, "Int", "1", "llama")
singletonErrorsWithInvalid(t, "Int", "1", "llama")

e := Everything{}
err := Parse(url.Values{"Int": {"4.2"}}, &e)
Expand Down Expand Up @@ -199,7 +205,7 @@ func TestUintTyped(t *testing.T) {

func TestUintErrors(t *testing.T) {
t.Parallel()
singletonErrors(t, "Uint", "1", "llama")
singletonErrorsWithInvalid(t, "Uint", "1", "llama")

e := Everything{}
err := Parse(url.Values{"Uint": {"4.2"}}, &e)
Expand Down Expand Up @@ -249,7 +255,7 @@ func TestFloatTyped(t *testing.T) {

func TestFloatErrors(t *testing.T) {
t.Parallel()
singletonErrors(t, "Float", "1.0", "llama")
singletonErrorsWithInvalid(t, "Float", "1.0", "llama")
}

func TestMap(t *testing.T) {
Expand Down Expand Up @@ -434,6 +440,17 @@ func TestStringTyped(t *testing.T) {
assertEqual(t, "e.AString", MyString("llama"), e.AString)
}

func TestStringErrors(t *testing.T) {
t.Parallel()
singletonErrors(t, "String", "str")

e := Everything{}
err := Parse(url.Values{"Int": {"4.2"}}, &e)
if err == nil {
t.Error("Expected error parsing float as int")
}
}

func TestStruct(t *testing.T) {
t.Parallel()
e := Everything{}
Expand Down Expand Up @@ -500,10 +517,112 @@ func TestTextUnmarshaler(t *testing.T) {

func TestTextUnmarshalerError(t *testing.T) {
t.Parallel()
e := Everything{}

err := Parse(url.Values{"Time": {"llama"}}, &e)
now := time.Now().Format(time.RFC3339)
singletonErrorsWithInvalid(t, "Time", now, "llama")
}

type Crazy struct {
A *Crazy
B *Crazy
Value int
Slice []int
Map map[string]Crazy
}

func TestCrazy(t *testing.T) {
t.Parallel()

c := Crazy{}
err := Parse(url.Values{
"A[B][B][A][Value]": {"1"},
"B[A][A][Slice][]": {"3", "1", "4"},
"B[Map][hello][A][Value]": {"8"},
"A[Value]": {"2"},
"A[Slice][]": {"9", "1", "1"},
"Value": {"42"},
}, &c)
if err != nil {
t.Error("Error parsing craziness: ", err)
}

// Exhaustively checking everything here is going to be a huge pain, so
// let's just hope for the best, pretend NPEs don't exist, and hope that
// this test covers enough stuff that it's actually useful.
assertEqual(t, "c.A.B.B.A.Value", 1, c.A.B.B.A.Value)
assertEqual(t, "c.A.Value", 2, c.A.Value)
assertEqual(t, "c.Value", 42, c.Value)
assertEqual(t, `c.B.Map["hello"].A.Value`, 8, c.B.Map["hello"].A.Value)

assertEqual(t, "c.A.B.B.B", (*Crazy)(nil), c.A.B.B.B)
assertEqual(t, "c.A.B.A", (*Crazy)(nil), c.A.B.A)
assertEqual(t, "c.A.A", (*Crazy)(nil), c.A.A)

if c.Slice != nil || c.Map != nil {
t.Error("Map and Slice should not be set")
}

sl := c.B.A.A.Slice
if len(sl) != 3 || sl[0] != 3 || sl[1] != 1 || sl[2] != 4 {
t.Error("Something is wrong with c.B.A.A.Slice")
}
sl = c.A.Slice
if len(sl) != 3 || sl[0] != 9 || sl[1] != 1 || sl[2] != 1 {
t.Error("Something is wrong with c.A.Slice")
}
}

func TestParseNilPtr(t *testing.T) {
var s *struct{}

err := Parse(url.Values{"A": {"123"}}, s)
if err == nil {
t.Errorf("Expected error parsing %T", s)
}
}

func TestUnsupportedStructKind(t *testing.T) {
var s struct {
Bad chan string
}

err := Parse(url.Values{"Bad": {"123"}}, &s)
if err == nil {
t.Errorf("Expected error parsing %T", s)
}
}

func TestUnsupportedNestedStruct(t *testing.T) {
var s struct {
Bad struct {
A chan string
}
}

err := Parse(url.Values{"Bad[A]": {"123"}}, &s)
if err == nil {
t.Errorf("Expected error parsing %T", s)
}
}

func TestUnsupportedParseKind(t *testing.T) {
var s struct {
Bad []chan string
}

err := Parse(url.Values{"Bad[]": {"123"}}, &s)
if err == nil {
t.Errorf("Expected error parsing %T", s)
}
}

func TestUnsupportedMapKey(t *testing.T) {
var s struct {
Bad map[int]int
}

err := Parse(url.Values{"Bad[123]": {"123"}}, &s)
if err == nil {
t.Error("expected error parsing llama as time")
t.Errorf("Expected error parsing %T", s)
}
}
Loading