Skip to content
This repository has been archived by the owner on Jul 24, 2024. It is now read-only.

Support encoding.BinaryMarshaler types #75

Open
wants to merge 2 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
14 changes: 10 additions & 4 deletions inmem.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package libchan

import (
"encoding"
"encoding/binary"
"errors"
"io"
Expand Down Expand Up @@ -312,6 +313,13 @@ func (w *pipeSender) copyValue(v interface{}) (interface{}, error) {
return w.copyChannelMessage(val)
case map[interface{}]interface{}:
return w.copyChannelInterfaceMessage(val)
case encoding.BinaryMarshaler:
p, err := val.MarshalBinary()
if err != nil {
return nil, err
}

return w.copyValue(p)
default:
if rv := reflect.ValueOf(v); rv.Kind() == reflect.Ptr {
if rv.Elem().Kind() == reflect.Struct {
Expand Down Expand Up @@ -419,10 +427,8 @@ func (w *pipeSender) copyChannelInterfaceMessage(m map[interface{}]interface{})
return mCopy, nil
}
func (w *pipeSender) copyStructure(m interface{}) (interface{}, error) {
v := reflect.ValueOf(m)
if v.Kind() == reflect.Ptr {
v = v.Elem()
}
v := reflect.Indirect(reflect.ValueOf(m))

if v.Kind() != reflect.Struct {
return nil, errors.New("invalid non struct type")
}
Expand Down
17 changes: 13 additions & 4 deletions spdy/copy.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package spdy

import (
"encoding"
"errors"
"io"
"net"
Expand Down Expand Up @@ -55,6 +56,13 @@ func (c *channel) copyValue(v interface{}) (interface{}, error) {
return c.copyChannelMessage(val)
case map[interface{}]interface{}:
return c.copyChannelInterfaceMessage(val)
case encoding.BinaryMarshaler:
p, err := val.MarshalBinary()
if err != nil {
return nil, err
}

return c.copyValue(p)
default:
if rv := reflect.ValueOf(v); rv.Kind() == reflect.Ptr {
if rv.Elem().Kind() == reflect.Struct {
Expand Down Expand Up @@ -163,10 +171,8 @@ func (c *channel) copyChannelInterfaceMessage(m map[interface{}]interface{}) (in
}

func (c *channel) copyStructure(m interface{}) (interface{}, error) {
v := reflect.ValueOf(m)
if v.Kind() == reflect.Ptr {
v = v.Elem()
}
v := reflect.Indirect(reflect.ValueOf(m))

if v.Kind() != reflect.Struct {
return nil, errors.New("invalid non struct type")
}
Expand All @@ -177,6 +183,9 @@ func (c *channel) copyStructValue(v reflect.Value) (interface{}, error) {
mCopy := make(map[string]interface{})
t := v.Type()
for i := 0; i < v.NumField(); i++ {
// TODO(stevvooe): Calling Interface without checking if a type can be
// interfaced may lead to panics. This value copier may need to be
// refactored to handle arbitrary types.
vCopy, vErr := c.copyValue(v.Field(i).Interface())
if vErr != nil {
return nil, vErr
Expand Down
16 changes: 16 additions & 0 deletions spdy/copy_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package spdy

import (
"testing"

"github.com/docker/libchan/testutil"
)

func TestTypeTransmission(t *testing.T) {
sender, receiver, err := Pipe()
if err != nil {
t.Fatalf("error creating pipe: %v", err)
}

testutil.CheckTypeTransmission(t, receiver, sender)
}
62 changes: 62 additions & 0 deletions testutil/check.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
// Package testutil contains checks that implementations of libchan transports
// can use to check compliance. For now, this will only work with Go, but
// cross-language tests could be added here, as well.
package testutil

import (
"io"
"io/ioutil"
"reflect"
"strings"
"testing"
"time"

"github.com/docker/libchan"
)

func CheckTypeTransmission(t *testing.T, receiver libchan.Receiver, sender libchan.Sender) {
// Add types that should be transmitted by value to this struct. Their
// equality will be tested with reflect.DeepEquals.
type ValueTypes struct {
I int
T time.Time
}

// Add other types, that may include readers or stateful items.
type A struct {
// TODO(stevvooe): Ideally, this would be embedded but libchan doesn't
// seem to transmit embedded structs correctly.
V ValueTypes
Reader io.ReadCloser // TODO(stevvooe): Only io.ReadCloser is support for now.
}

readerContent := "asdf"
expected := A{
V: ValueTypes{
I: 1234,
T: time.Now(),
},
Reader: ioutil.NopCloser(strings.NewReader(readerContent)),
}

go func() {
if err := sender.Send(expected); err != nil {
t.Fatalf("unexpected error sending: %v", err)
}
}()

var received A
if err := receiver.Receive(&received); err != nil {
t.Fatalf("unexpected error receiving: %v", err)
}

if !reflect.DeepEqual(received.V, expected.V) {
t.Fatalf("expected structs to be equal: %#v != %#v", received.V, expected.V)
}

receivedContent, _ := ioutil.ReadAll(received.Reader)
if string(receivedContent) != readerContent {
t.Fatalf("reader transmitted different content %q != %q", receivedContent, readerContent)
}

}
14 changes: 14 additions & 0 deletions testutil/check_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
package testutil

import (
"github.com/docker/libchan"

"testing"
)

// TestTypeTransmission tests the main package (to avoid import cycles) and
// provides and example of how this test should be used in other packages.
func TestTypeTransmission(t *testing.T) {
receiver, sender := libchan.Pipe()
CheckTypeTransmission(t, receiver, sender)
}