Skip to content

Commit

Permalink
fix(mongo): serialization value in the transaction method
Browse files Browse the repository at this point in the history
Signed-off-by: Luigi Rende <[email protected]>
  • Loading branch information
luigirende committed Oct 28, 2024
1 parent ab9422d commit af8bd78
Show file tree
Hide file tree
Showing 2 changed files with 79 additions and 1 deletion.
15 changes: 14 additions & 1 deletion state/mongodb/mongodb.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ import (
"strings"
"time"

"github.com/dapr/components-contrib/contenttype"

"github.com/google/uuid"
"go.mongodb.org/mongo-driver/bson"
"go.mongodb.org/mongo-driver/bson/bsonrw"
Expand Down Expand Up @@ -528,7 +530,18 @@ func (m *MongoDB) doTransaction(sessCtx mongo.SessionContext, operations []state
var err error
switch req := o.(type) {
case state.SetRequest:
err = m.setInternal(sessCtx, &req)
{
isJSON := (len(req.Metadata) > 0 && req.Metadata[metadata.ContentType] == contenttype.JSONContentType)
if isJSON {
if bytes, ok := req.Value.([]byte); ok {
err = json.Unmarshal(bytes, &req.Value)
if err != nil {
break
}
}
}
err = m.setInternal(sessCtx, &req)
}
case state.DeleteRequest:
err = m.deleteInternal(sessCtx, &req)
}
Expand Down
65 changes: 65 additions & 0 deletions tests/conformance/state/state.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ package state

import (
"context"
"encoding/base64"
"encoding/json"
"fmt"
"slices"
Expand Down Expand Up @@ -784,6 +785,70 @@ func ConformanceTests(t *testing.T, props map[string]string, statestore state.St
assert.Equal(t, v, res.Data)
}
})

t.Run("transaction-serialization-grpc-json", func(t *testing.T) {
features := statestore.Features()
// this check for exclude redis 7
if state.FeatureQueryAPI.IsPresent(features) {
json := "{\"id\":1223,\"name\":\"test\"}"
keyTest1 := key + "-key-grpc"
valueTest := []byte(json)
keyTest2 := key + "-key-grpc-no-json"

metadataTest1 := map[string]string{
"contentType": "application/json",
}

operations := []state.TransactionalStateOperation{
state.SetRequest{
Key: keyTest1,
Value: valueTest,
Metadata: metadataTest1,
},
state.SetRequest{
Key: keyTest2,
Value: valueTest,
},
}

expected := map[string][]byte{
keyTest1: []byte(json),
keyTest2: []byte(json),
}

expectedMetadata := map[string]map[string]string{
keyTest1: metadataTest1,
}

// Act
transactionStore, ok := statestore.(state.TransactionalStore)
assert.True(t, ok)
err := transactionStore.Multi(context.Background(), &state.TransactionalStateRequest{
Operations: operations,
})
require.NoError(t, err)

// Assert
for k, v := range expected {
res, err := statestore.Get(context.Background(), &state.GetRequest{
Key: k,
Metadata: expectedMetadata[k],
})
expectedValue := res.Data

// In redisjson when set the value with contentType = application/Json store the value in base64
if strings.HasPrefix(string(expectedValue), "\"ey") {
valueBase64 := strings.Trim(string(expectedValue), "\"")
expectedValueDecoded, _ := base64.StdEncoding.DecodeString(valueBase64)
require.NoError(t, err)
assert.Equal(t, expectedValueDecoded, v)
} else {
require.NoError(t, err)
assert.Equal(t, expectedValue, v)
}
}
}
})
} else {
t.Run("component does not implement TransactionalStore interface", func(t *testing.T) {
_, ok := statestore.(state.TransactionalStore)
Expand Down

0 comments on commit af8bd78

Please sign in to comment.