Skip to content

Commit

Permalink
Add merge and split fields utilities to be used by encoders (#212)
Browse files Browse the repository at this point in the history
  • Loading branch information
nolag authored and reductionista committed Dec 4, 2023
1 parent a16abbe commit 1472a9b
Show file tree
Hide file tree
Showing 3 changed files with 175 additions and 2 deletions.
6 changes: 4 additions & 2 deletions pkg/loop/internal/chain_reader_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,19 @@ package internal

import (
"errors"
"github.com/stretchr/testify/assert"
"google.golang.org/grpc/test/bufconn"
"net"
"sync"
"testing"

"github.com/stretchr/testify/assert"
"google.golang.org/grpc/test/bufconn"

"github.com/stretchr/testify/require"

"github.com/smartcontractkit/chainlink-relay/pkg/loop/internal/pb"

"context"

"github.com/fxamacker/cbor/v2"
"github.com/mitchellh/mapstructure"
ocrtypes "github.com/smartcontractkit/libocr/offchainreporting2plus/types"
Expand Down
84 changes: 84 additions & 0 deletions pkg/utils/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,16 @@ package utils

import (
"context"
"encoding/base64"
"math"
"math/big"
mrand "math/rand"
"reflect"
"strings"
"time"

"github.com/smartcontractkit/chainlink-common/pkg/services"
"github.com/smartcontractkit/chainlink-common/pkg/types"
)

// WithJitter adds +/- 10% to a duration
Expand Down Expand Up @@ -67,3 +70,84 @@ func FitsInNBitsSigned(n int, bi *big.Int) bool {
}
return bi.BitLen() <= n-1
}

func MergeValueFields(valueFields []map[string]any) (map[string]any, error) {
numItems := len(valueFields)

switch numItems {
case 0:
return map[string]any{}, nil
default:
mergedReflect := map[string]reflect.Value{}
for k, v := range valueFields[0] {
rv := reflect.ValueOf(v)
slice := reflect.MakeSlice(reflect.SliceOf(rv.Type()), numItems, numItems)
slice.Index(0).Set(rv)
mergedReflect[k] = slice
}

for i, valueField := range valueFields[1:] {
if len(valueField) != len(mergedReflect) {
return nil, types.InvalidTypeError{}
}

for k, slice := range mergedReflect {
if value, ok := valueField[k]; ok {
sliceElm := slice.Index(i + 1)
rv := reflect.ValueOf(value)
if !rv.Type().AssignableTo(sliceElm.Type()) {
return nil, types.InvalidTypeError{}
}
sliceElm.Set(rv)
} else {
return nil, types.InvalidTypeError{}
}
}
}

merged := map[string]any{}

for k, v := range mergedReflect {
merged[k] = v.Interface()
}

return merged, nil
}
}

func SplitValueFields(decoded map[string]any) ([]map[string]any, error) {
var result []map[string]any

for k, v := range decoded {
iv := reflect.ValueOf(v)
kind := iv.Kind()
if kind != reflect.Slice && kind != reflect.Array {
if kind != reflect.String {
return nil, types.NotASliceError{}
}
rawBytes, err := base64.StdEncoding.DecodeString(v.(string))
if err != nil {
return nil, types.InvalidTypeError{}
}
iv = reflect.ValueOf(rawBytes)
}

length := iv.Len()
if result == nil {
result = make([]map[string]any, length)
for i := 0; i < length; i++ {
result[i] = map[string]any{}
}
}

if len(result) != length {
return nil, types.InvalidTypeError{}
}

for i := 0; i < length; i++ {
result[i][k] = iv.Index(i).Interface()
}
}

return result, nil
}
87 changes: 87 additions & 0 deletions pkg/utils/utils_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,11 @@ import (
"math/big"
"testing"

"github.com/stretchr/testify/require"

"github.com/stretchr/testify/assert"

"github.com/smartcontractkit/chainlink-relay/pkg/types"
"github.com/smartcontractkit/chainlink-relay/pkg/utils"
)

Expand All @@ -27,3 +30,87 @@ func TestFitsInNBitsSigned(t *testing.T) {
assert.False(t, utils.FitsInNBitsSigned(16, bi))
})
}

func TestMergeValueFields(t *testing.T) {
t.Parallel()
t.Run("Merges fields", func(t *testing.T) {
input := []map[string]any{
{"Foo": int32(1), "Bar": "Hi"},
{"Foo": int32(2), "Bar": "How"},
{"Foo": int32(3), "Bar": "Are"},
{"Foo": int32(4), "Bar": "You?"},
}

output, err := utils.MergeValueFields(input)
require.NoError(t, err)

expected := map[string]any{
"Foo": []int32{1, 2, 3, 4},
"Bar": []string{"Hi", "How", "Are", "You?"},
}
assert.Equal(t, expected, output)
})

t.Run("Returns error if keys are not the same", func(t *testing.T) {
input := []map[string]any{
{"Foo": int32(1), "Bar": "Hi"},
{"Zap": 2, "Foo": int32(2), "Bar": "How"},
}

_, err := utils.MergeValueFields(input)

assert.IsType(t, types.InvalidTypeError{}, err)
})

t.Run("Returns error if values are not compatible types", func(t *testing.T) {
input := []map[string]any{
{"Foo": int32(1), "Bar": "Hi"},
{"Foo": int32(2), "Bar": int32(3)},
}

_, err := utils.MergeValueFields(input)

assert.IsType(t, types.InvalidTypeError{}, err)
})
}

func TestSplitValueField(t *testing.T) {
t.Parallel()
t.Run("Returns slit field values", func(t *testing.T) {
input := map[string]any{
"Foo": []int32{1, 2, 3, 4},
"Bar": [4]string{"Hi", "How", "Are", "You?"},
}

output, err := utils.SplitValueFields(input)
require.NoError(t, err)

expected := []map[string]any{
{"Foo": int32(1), "Bar": "Hi"},
{"Foo": int32(2), "Bar": "How"},
{"Foo": int32(3), "Bar": "Are"},
{"Foo": int32(4), "Bar": "You?"},
}
assert.Equal(t, expected, output)
})

t.Run("Returns error if lengths do not match", func(t *testing.T) {
input := map[string]any{
"Foo": []int32{1, 2, 3},
"Bar": []string{"Hi", "How", "Are", "You?"},
}

_, err := utils.SplitValueFields(input)
assert.IsType(t, types.InvalidTypeError{}, err)
})

t.Run("Returns error if item is not an array or slice", func(t *testing.T) {
input := map[string]any{
"Foo": int32(3),
"Bar": []string{"Hi", "How", "Are", "You?"},
}

_, err := utils.SplitValueFields(input)
assert.IsType(t, types.NotASliceError{}, err)
})
}

0 comments on commit 1472a9b

Please sign in to comment.