我能够在 Hyperledger(结构实现)中进行交易。我想通过传递用户的密钥来查看用户发起的所有事务及其有效负载详细信息。
例如:
A transfers 10 units to B
A transfers 5 units to C
D transfers 8 units to A
当我传递 A 的密钥时,fabric 必须向我提供 A 的所有交易。有什么办法吗?或者我应该使用哪个结构 API 函数调用?
我能够在 Hyperledger(结构实现)中进行交易。我想通过传递用户的密钥来查看用户发起的所有事务及其有效负载详细信息。
例如:
A transfers 10 units to B
A transfers 5 units to C
D transfers 8 units to A
当我传递 A 的密钥时,fabric 必须向我提供 A 的所有交易。有什么办法吗?或者我应该使用哪个结构 API 函数调用?
/chain/blocks/{Block}
端点携带指定块中的有序交易列表。
使用/chain
端点获取链的高度(块数),然后使用/chain/blocks/{Block}
REST 端点从每个块中检索交易。
您可以在链代码中开发适当的索引和查询功能。
对于每个事务,您将其详细信息存储在内部键/值存储 (stub.PutState) 中,用户作为键并返回与查询中的用户关联的所有事务 (stub.GetState)。
最好和最简单的方法是使用shim 包功能
GetHistoryForKey(密钥字符串)
正如文档所说:
链码可以调用 GetHistoryForKey 函数以返回跨时间的键值历史记录。GetHistoryForKey 旨在用于只读查询。
如果有人需要 Java SDK 和 go 链码组合。给你
在这里回答了类似的问题
Java 代码
public List<HistoryDao> getUFOHistory(String key) throws Exception {
String[] args = { key };
Logger.getLogger(QueryChaincode.class.getName()).log(Level.INFO, "UFO communication history - " + args[0]);
Collection<ProposalResponse> responses1Query = ucc.getChannelClient().queryByChainCode("skynetchaincode", "getHistoryForUFO", args);
String stringResponse = null;
ArrayList<HistoryDao> newArrayList = new ArrayList<>();
for (ProposalResponse pres : responses1Query) {
stringResponse = new String(pres.getChaincodeActionResponsePayload());
Logger.getLogger(QueryChaincode.class.getName()).log(Level.INFO, stringResponse);
newArrayList = gson.fromJson(stringResponse, new TypeToken<ArrayList<HistoryDao>>() {
}.getType());
}
if (null == stringResponse)
stringResponse = "Not able to find any ufo communication history";
return newArrayList;
}
你去 chancode 实现如下
去代码
func (t *SmartContract) getHistoryForUFO(APIstub shim.ChaincodeStubInterface, args []string) sc.Response {
if len(args) < 1 {
return shim.Error("Incorrect number of arguments. Expecting 1")
}
ufoId := args[0]
resultsIterator, err := APIstub.GetHistoryForKey(ufoId)
if err != nil {
return shim.Error(err.Error())
}
defer resultsIterator.Close()
var buffer bytes.Buffer
buffer.WriteString("[")
bArrayMemberAlreadyWritten := false
for resultsIterator.HasNext() {
response, err := resultsIterator.Next()
if err != nil {
return shim.Error(err.Error())
}
// Add a comma before array members, suppress it for the first array member
if bArrayMemberAlreadyWritten == true {
buffer.WriteString(",")
}
buffer.WriteString("{\"TxId\":")
buffer.WriteString("\"")
buffer.WriteString(response.TxId)
buffer.WriteString("\"")
buffer.WriteString(", \"Value\":")
// if it was a delete operation on given key, then we need to set the
//corresponding value null. Else, we will write the response.Value
//as-is (as the Value itself a JSON)
if response.IsDelete {
buffer.WriteString("null")
} else {
buffer.WriteString(string(response.Value))
}
buffer.WriteString(", \"Timestamp\":")
buffer.WriteString("\"")
buffer.WriteString(time.Unix(response.Timestamp.Seconds, int64(response.Timestamp.Nanos)).String())
buffer.WriteString("\"")
buffer.WriteString(", \"IsDelete\":")
buffer.WriteString("\"")
buffer.WriteString(strconv.FormatBool(response.IsDelete))
buffer.WriteString("\"")
buffer.WriteString("}")
bArrayMemberAlreadyWritten = true
}
buffer.WriteString("]")
fmt.Printf("- History returning:\n%s\n", buffer.String())
return shim.Success(buffer.Bytes())
}
如果您有疑问,请告诉我。
如果您使用的是 composer-client,您可以简单地使用 Historian 命令。
var historian = await businessNetworkConnection.getHistorian();
historian.getAll().then(historianRecords => console.log(historianRecords));