fabric-samples/chaincode/marbles_transfer/marbles_transfer_queries.go
Tiffany Harris df292e0e8d Implement sale, bid, and history queries
- Add marbles proposed sale and bid prices queries
- Add marbles history query
- Add verification of ownership when selling marble

Signed-off-by: Tiffany Harris <tiffany.harris@ibm.com>
Signed-off-by: Dereck Luo <dereckluo@gmail.com>
2020-05-25 21:22:01 -04:00

195 lines
6.5 KiB
Go

/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package main
import (
"encoding/json"
"fmt"
"time"
"github.com/hyperledger/fabric-contract-api-go/contractapi"
)
// QueryResult structure used for handling result of query
type QueryResult struct {
Record *Marble
TxId string `json:"txId"`
Timestamp time.Time `json:"timestamp"`
}
type Agreement struct {
ID string `json:"marble_id"`
Price int `json:"price"`
TradeID string `json:"trade_id"`
}
// GetAsset returns the public marble data
func (s *SmartContract) GetAsset(ctx contractapi.TransactionContextInterface, marbleID string) (*Marble, error) {
// since only public data is accessed in this function, no access control is required
marbleJSON, err := ctx.GetStub().GetState(marbleID)
if err != nil {
return nil, fmt.Errorf("failed to read from world state: %s", err.Error())
}
if marbleJSON == nil {
return nil, fmt.Errorf("%s does not exist", marbleID)
}
marble := new(Marble)
_ = json.Unmarshal(marbleJSON, marble)
return marble, nil
}
// GetAssetPrivateProperties returns the immutable marble properties from owner's private data collection
func (s *SmartContract) GetAssetPrivateProperties(ctx contractapi.TransactionContextInterface, marbleID string) (string, error) {
// Get client org id and verify it matches peer org id.
// In this scenario, client is only authorized to read/write private data from its own peer.
collection, err := getClientImplicitCollectionName(ctx)
if err != nil {
return "", err
}
immutableProperties, err := ctx.GetStub().GetPrivateData(collection, marbleID)
if err != nil {
return "", fmt.Errorf("failed to read marble private properties from client org's collection: %s", err.Error())
}
if immutableProperties == nil {
return "", fmt.Errorf("marble private details does not exist in client org's collection: %s", marbleID)
}
return string(immutableProperties), nil
}
// GetAssetSalesPrice returns the sales price as an integer
func (s *SmartContract) GetAssetSalesPrice(ctx contractapi.TransactionContextInterface, marbleID string) (string, error) {
return getAssetPrice(ctx, marbleID, typeMarbleForSale)
}
// GetAssetBidPrice returns the bid price as an integer
func (s *SmartContract) GetAssetBidPrice(ctx contractapi.TransactionContextInterface, marbleID string) (string, error) {
return getAssetPrice(ctx, marbleID, typeMarbleBid)
}
// getAssetPrice gets the bid or ask price from caller's implicit private data collection
func getAssetPrice(ctx contractapi.TransactionContextInterface, marbleID string, priceType string) (string, error) {
collection, err := getClientImplicitCollectionName(ctx)
if err != nil {
return "", err
}
marblePriceKey, err := ctx.GetStub().CreateCompositeKey(priceType, []string{marbleID})
if err != nil {
return "", fmt.Errorf("failed to create composite key: %s", err.Error())
}
marblePriceJSON, err := ctx.GetStub().GetPrivateData(collection, marblePriceKey)
if err != nil {
return "", fmt.Errorf("failed to read marble price from implicit private data collection: %s", err.Error())
}
if marblePriceJSON == nil {
return "", fmt.Errorf("marble price does not exist: %s", marbleID)
}
return string(marblePriceJSON), nil
}
// QueryAssetSaleAgreements returns all of an organization's proposed sales
func (s *SmartContract) QueryAssetSaleAgreements(ctx contractapi.TransactionContextInterface) ([]Agreement, error) {
return queryAgreementsByType(ctx, typeMarbleForSale)
}
// QueryAssetBuyAgreements returns all of an organization's proposed buys
func (s *SmartContract) QueryAssetBuyAgreements(ctx contractapi.TransactionContextInterface) ([]Agreement, error) {
return queryAgreementsByType(ctx, typeMarbleBid)
}
func queryAgreementsByType(ctx contractapi.TransactionContextInterface, agreeType string) ([]Agreement, error) {
collection, err := getClientImplicitCollectionName(ctx)
if err != nil {
return nil, err
}
// Query for any object type starting with `agreeType`
agreementsIterator, err := ctx.GetStub().GetPrivateDataByPartialCompositeKey(collection, agreeType, []string{})
if err != nil {
return nil, fmt.Errorf("failed to read from private data collection: %s", err.Error())
}
defer agreementsIterator.Close()
agreements := []Agreement{}
for agreementsIterator.HasNext() {
resp, err := agreementsIterator.Next()
if err != nil {
return nil, err
}
newAgree := new(Agreement)
err = json.Unmarshal(resp.Value, newAgree)
if err != nil {
return nil, err
}
agreements = append(agreements, *newAgree)
}
return agreements, nil
}
// TODO add a JSON index and query to return all of an organization's marbles larger than a certain size (only works when using CouchDB state database)
// hint: see sample index at https://github.com/hyperledger/fabric-samples/blob/master/chaincode/marbles02/go/META-INF/statedb/couchdb/indexes/indexOwner.json
// hint: see sample query at https://github.com/hyperledger/fabric-samples/blob/master/chaincode/marbles02/go/marbles_chaincode.go#L515
// QueryAssetHistory returns the chain of custody for a marble since issuance
func (s *SmartContract) QueryAssetHistory(ctx contractapi.TransactionContextInterface, marbleID string) ([]QueryResult, error) {
resultsIterator, err := ctx.GetStub().GetHistoryForKey(marbleID)
if err != nil {
return nil, err
}
defer resultsIterator.Close()
records := []QueryResult{}
for resultsIterator.HasNext() {
response, err := resultsIterator.Next()
if err != nil {
return nil, err
}
marble := new(Marble)
err = json.Unmarshal(response.Value, marble)
if err != nil {
return nil, err
}
record := QueryResult{
TxId: response.TxId,
Timestamp: time.Unix(response.Timestamp.Seconds, int64(response.Timestamp.Nanos)),
Record: marble,
}
records = append(records, record)
}
return records, nil
}