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

add support GetMultipleKeys #148

Merged
merged 1 commit into from
Feb 10, 2025
Merged
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
87 changes: 86 additions & 1 deletion shim/handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,8 @@ const (
established state = "established" // connection established
ready state = "ready" // ready for requests

defaultMaxSizeWriteBatch = 100
defaultMaxSizeWriteBatch = 100
defaultMaxSizeGetMultipleKeys = 100
)

// PeerChaincodeStream is the common stream interface for Peer - chaincode communication.
Expand Down Expand Up @@ -51,6 +52,9 @@ type Handler struct {
// if you can send the changes in batches.
usePeerWriteBatch bool
maxSizeWriteBatch uint32
// if you can get the multiple keys in batches.
usePeerGetMultipleKeys bool
maxSizeGetMultipleKeys uint32

// Multiple queries (and one transaction) with different txids can be executing in parallel for this chaincode
// responseChannels is the channel on which responses are communicated by the shim to the chaincodeStub.
Expand Down Expand Up @@ -267,6 +271,80 @@ func (h *Handler) handleGetState(collection string, key string, channelID string
return nil, fmt.Errorf("[%s] incorrect chaincode message %s received. Expecting %s or %s", shorttxid(responseMsg.Txid), responseMsg.Type, peer.ChaincodeMessage_RESPONSE, peer.ChaincodeMessage_ERROR)
}

// handleGetMultipleStates communicates with the peer to fetch the requested state information from the ledger.
func (h *Handler) handleGetMultipleStates(collection string, keys []string, channelID string, txID string) ([][]byte, error) {
if len(keys) == 0 {
return nil, nil
}

responses := make([][]byte, 0, len(keys))

if !h.usePeerGetMultipleKeys {
for _, key := range keys {
resp, err := h.handleGetState(collection, key, channelID, txID)
if err != nil {
return nil, err
}
responses = append(responses, resp)
}
return responses, nil
}

for ; len(keys) > int(h.maxSizeGetMultipleKeys); keys = keys[h.maxSizeGetMultipleKeys:] {
resp, err := h.handleOneSendGetMultipleStates(collection, keys[:h.maxSizeGetMultipleKeys], channelID, txID)
if err != nil {
return nil, err
}
responses = append(responses, resp...)
}

if len(keys) > 0 {
resp, err := h.handleOneSendGetMultipleStates(collection, keys, channelID, txID)
if err != nil {
return nil, err
}
responses = append(responses, resp...)
}

for i := range responses {
if len(responses[i]) == 0 {
responses[i] = nil
}
}

return responses, nil
}

// handleOneSendGetMultipleStates communicates with the peer to fetch one batch of keys from the ledger.
func (h *Handler) handleOneSendGetMultipleStates(collection string, keys []string, channelID string, txID string) ([][]byte, error) {
// Construct payload for GET_STATE_MULTIPLE
payloadBytes := marshalOrPanic(&peer.GetStateMultiple{Keys: keys, Collection: collection})

msg := &peer.ChaincodeMessage{Type: peer.ChaincodeMessage_GET_STATE_MULTIPLE, Payload: payloadBytes, Txid: txID, ChannelId: channelID}
responseMsg, err := h.callPeerWithChaincodeMsg(msg, channelID, txID)
if err != nil {
return nil, fmt.Errorf("[%s] error sending %s: %s", shorttxid(txID), peer.ChaincodeMessage_GET_STATE_MULTIPLE, err)
}

if responseMsg.Type == peer.ChaincodeMessage_RESPONSE {
// Success response
var gmkResult peer.GetStateMultipleResult
err = proto.Unmarshal(responseMsg.Payload, &gmkResult)
if err != nil {
return nil, errors.New("could not unmarshal get state multiple keys response")
}

return gmkResult.GetValues(), nil
}
if responseMsg.Type == peer.ChaincodeMessage_ERROR {
// Error response
return nil, fmt.Errorf("%s", responseMsg.Payload[:])
}

// Incorrect chaincode message received
return nil, fmt.Errorf("[%s] incorrect chaincode message %s received. Expecting %s or %s", shorttxid(responseMsg.Txid), responseMsg.Type, peer.ChaincodeMessage_RESPONSE, peer.ChaincodeMessage_ERROR)
}

func (h *Handler) handleGetPrivateDataHash(collection string, key string, channelID string, txid string) ([]byte, error) {
// Construct payload for GET_PRIVATE_DATA_HASH
payloadBytes := marshalOrPanic(&peer.GetState{Collection: collection, Key: key})
Expand Down Expand Up @@ -733,6 +811,13 @@ func (h *Handler) handleEstablished(msg *peer.ChaincodeMessage) error {
h.maxSizeWriteBatch = defaultMaxSizeWriteBatch
}

h.usePeerGetMultipleKeys = ccAdditionalParams.UseGetMultipleKeys
h.maxSizeGetMultipleKeys = ccAdditionalParams.MaxSizeGetMultipleKeys

if h.usePeerGetMultipleKeys && h.maxSizeGetMultipleKeys < defaultMaxSizeGetMultipleKeys {
h.maxSizeGetMultipleKeys = defaultMaxSizeGetMultipleKeys
}

return nil
}

Expand Down
18 changes: 18 additions & 0 deletions shim/interfaces.go
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,15 @@ type ChaincodeStubInterface interface {
// If the key does not exist in the state database, (nil, nil) is returned.
GetState(key string) ([]byte, error)

// GetMultipleStates retrieves the values of the specified keys from the ledger.
// It has similar semantics to the GetState function regarding read-your-own-writes behavior,
// meaning that updates made by a previous PutState operation within the same chaincode
// session are not visible to this function.
// If no key is passed, the function will return (nil, nil).
// GetMultipleStates returns values in the same order in which the keys are provided,
// and any keys missing from the ledger return a nil.
GetMultipleStates(keys ...string) ([][]byte, error)

// PutState puts the specified `key` and `value` into the transaction's
// writeset as a data-write proposal. PutState doesn't effect the ledger
// until the transaction is validated and successfully committed.
Expand Down Expand Up @@ -247,6 +256,15 @@ type ChaincodeStubInterface interface {
// that has not been committed.
GetPrivateData(collection, key string) ([]byte, error)

// GetMultiplePrivateData retrieves the values of the specified keys from the specified collection.
// It has similar semantics to the GetePrivateData function regarding read-your-own-writes behavior,
// meaning that updates made by a previous PutPrivateData operation within the same chaincode
// session are not visible to this function.
// If no key is passed, the function will return (nil, nil).
// GetMultiplePrivateData returns values in the same order in which the keys are provided,
// and any keys missing from the ledger return a nil.
GetMultiplePrivateData(collection string, keys ...string) ([][]byte, error)

// GetPrivateDataHash returns the hash of the value of the specified `key` from the specified
// `collection`
GetPrivateDataHash(collection, key string) ([]byte, error)
Expand Down
15 changes: 15 additions & 0 deletions shim/stub.go
Original file line number Diff line number Diff line change
Expand Up @@ -166,6 +166,13 @@ func (s *ChaincodeStub) GetState(key string) ([]byte, error) {
return s.handler.handleGetState(collection, key, s.ChannelID, s.TxID)
}

// GetMultipleStates documentation can be found in interfaces.go
func (s *ChaincodeStub) GetMultipleStates(keys ...string) ([][]byte, error) {
// Access public data by setting the collection to empty string
collection := ""
return s.handler.handleGetMultipleStates(collection, keys, s.ChannelID, s.TxID)
}

// SetStateValidationParameter documentation can be found in interfaces.go
func (s *ChaincodeStub) SetStateValidationParameter(key string, ep []byte) error {
return s.putStateMetadataEntry("", key, s.validationParameterMetakey, ep)
Expand Down Expand Up @@ -262,6 +269,14 @@ func (s *ChaincodeStub) GetPrivateData(collection string, key string) ([]byte, e
return s.handler.handleGetState(collection, key, s.ChannelID, s.TxID)
}

// GetMultiplePrivateData documentation can be found in interfaces.go
func (s *ChaincodeStub) GetMultiplePrivateData(collection string, keys ...string) ([][]byte, error) {
if collection == "" {
return nil, fmt.Errorf("collection must not be an empty string")
}
return s.handler.handleGetMultipleStates(collection, keys, s.ChannelID, s.TxID)
}

// GetPrivateDataHash documentation can be found in interfaces.go
func (s *ChaincodeStub) GetPrivateDataHash(collection string, key string) ([]byte, error) {
if collection == "" {
Expand Down
109 changes: 99 additions & 10 deletions shim/stub_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -252,12 +252,19 @@ func TestGetMSPID(t *testing.T) {
}

func TestChaincodeStubHandlers(t *testing.T) {
gmkResult := &peer.GetStateMultipleResult{
Values: [][]byte{[]byte("myvalue"), []byte("myvalue")},
}
getMultipleKeysBytes, err := proto.Marshal(gmkResult)
assert.NoError(t, err)

var tests = []struct {
name string
resType peer.ChaincodeMessage_Type
payload []byte
usePeerWriteBatch bool
testFunc func(*ChaincodeStub, *Handler, *testing.T, []byte)
name string
resType peer.ChaincodeMessage_Type
payload []byte
usePeerWriteBatch bool
usePeerGetMultipleKeys bool
testFunc func(*ChaincodeStub, *Handler, *testing.T, []byte)
}{
{
name: "Simple Response",
Expand Down Expand Up @@ -730,6 +737,86 @@ func TestChaincodeStubHandlers(t *testing.T) {
checkWriteBatch(t, s.handler.chatStream, 1, 2)
},
},
{
name: "Get Multiple Keys - peer new",
resType: peer.ChaincodeMessage_RESPONSE,
payload: getMultipleKeysBytes,
usePeerGetMultipleKeys: true,
testFunc: func(s *ChaincodeStub, h *Handler, t *testing.T, payload []byte) {
resp, err := s.GetMultipleStates("key", "key2", "key3", "key4")
assert.NoError(t, err)
assert.Len(t, resp, 4)
for _, r := range resp {
assert.Equal(t, []byte("myvalue"), r)
}

resp, err = s.GetMultiplePrivateData("col", "key", "key2", "key3", "key4")
assert.NoError(t, err)
assert.Len(t, resp, 4)
for _, r := range resp {
assert.Equal(t, []byte("myvalue"), r)
}
},
},
{
name: "Get Multiple Keys - peer old",
resType: peer.ChaincodeMessage_RESPONSE,
payload: []byte("myvalue"),
testFunc: func(s *ChaincodeStub, h *Handler, t *testing.T, payload []byte) {
resp, err := s.GetMultipleStates("key", "key2", "key3", "key4")
assert.NoError(t, err)
assert.Len(t, resp, 4)
for _, r := range resp {
assert.Equal(t, payload, r)
}

resp, err = s.GetMultiplePrivateData("col", "key", "key2", "key3", "key4")
assert.NoError(t, err)
assert.Len(t, resp, 4)
for _, r := range resp {
assert.Equal(t, payload, r)
}
},
},
{
name: "Get Multiple Keys error - peer new",
resType: peer.ChaincodeMessage_ERROR,
payload: []byte("error"),
usePeerGetMultipleKeys: true,
testFunc: func(s *ChaincodeStub, h *Handler, t *testing.T, payload []byte) {
_, err := s.GetMultipleStates("key", "key2")
assert.EqualError(t, err, string(payload))

_, err = s.GetMultiplePrivateData("col", "key", "key2")
assert.EqualError(t, err, string(payload))
},
},
{
name: "Get Multiple Keys error - peer old",
resType: peer.ChaincodeMessage_ERROR,
payload: []byte("error"),
testFunc: func(s *ChaincodeStub, h *Handler, t *testing.T, payload []byte) {
_, err := s.GetMultipleStates("key", "key2")
assert.EqualError(t, err, string(payload))

_, err = s.GetMultiplePrivateData("col", "key", "key2")
assert.EqualError(t, err, string(payload))
},
},
{
name: "Get Multiple Keys - without keys",
resType: peer.ChaincodeMessage_RESPONSE,
payload: []byte("myvalue"),
testFunc: func(s *ChaincodeStub, h *Handler, t *testing.T, payload []byte) {
resp, err := s.GetMultipleStates()
assert.NoError(t, err)
assert.Len(t, resp, 0)

resp, err = s.GetMultiplePrivateData("col")
assert.NoError(t, err)
assert.Len(t, resp, 0)
},
},
}

for _, test := range tests {
Expand All @@ -738,11 +825,13 @@ func TestChaincodeStubHandlers(t *testing.T) {
t.Parallel()

handler := &Handler{
cc: &mockChaincode{},
responseChannels: map[string]chan *peer.ChaincodeMessage{},
state: ready,
usePeerWriteBatch: test.usePeerWriteBatch,
maxSizeWriteBatch: 100,
cc: &mockChaincode{},
responseChannels: map[string]chan *peer.ChaincodeMessage{},
state: ready,
usePeerWriteBatch: test.usePeerWriteBatch,
maxSizeWriteBatch: 100,
usePeerGetMultipleKeys: test.usePeerGetMultipleKeys,
maxSizeGetMultipleKeys: 2,
}
stub := &ChaincodeStub{
ChannelID: "channel",
Expand Down
Loading