/* * 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 }