mirror of
https://github.com/hyperledger/fabric-samples.git
synced 2026-06-17 15:35:09 +00:00
Created packages for the flat file store and the processor and moved functions, variables and constants from listener.go to those packages. Encapsulated everything not used outside the packages, introduced model.go files which later might be extracted into a model package and renamed parser/parsedBlock.go to parser/block.go. Signed-off-by: Stanislav Jakuschevskij <stas@two-giants.com>
114 lines
2.6 KiB
Go
114 lines
2.6 KiB
Go
package processor
|
|
|
|
import (
|
|
"fmt"
|
|
"offChainData/parser"
|
|
"offChainData/store"
|
|
|
|
"github.com/hyperledger/fabric-gateway/pkg/client"
|
|
)
|
|
|
|
type block struct {
|
|
block *parser.Block
|
|
checkpointer *client.FileCheckpointer
|
|
writeToStore store.Writer
|
|
channelName string
|
|
}
|
|
|
|
func NewBlock(
|
|
_block *parser.Block,
|
|
checkpointer *client.FileCheckpointer,
|
|
writeToStore store.Writer,
|
|
channelName string,
|
|
) *block {
|
|
return &block{
|
|
_block,
|
|
checkpointer,
|
|
writeToStore,
|
|
channelName,
|
|
}
|
|
}
|
|
|
|
func (b *block) Process() {
|
|
blockNumber := b.block.Number()
|
|
|
|
fmt.Println("\nReceived block", blockNumber)
|
|
|
|
for _, transaction := range b.validTransactions() {
|
|
aTransactionProcessor := transactionProcessor{
|
|
blockNumber,
|
|
transaction,
|
|
// TODO use pointer to parent and get blockNumber, store and channelName from parent
|
|
b.writeToStore,
|
|
b.channelName,
|
|
}
|
|
aTransactionProcessor.process()
|
|
|
|
transactionId := transaction.ChannelHeader().GetTxId()
|
|
b.checkpointer.CheckpointTransaction(blockNumber, transactionId)
|
|
}
|
|
|
|
b.checkpointer.CheckpointBlock(b.block.Number())
|
|
}
|
|
|
|
func (b *block) validTransactions() []*parser.Transaction {
|
|
result := []*parser.Transaction{}
|
|
for _, transaction := range b.getNewTransactions() {
|
|
if transaction.IsValid() {
|
|
result = append(result, transaction)
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (b *block) getNewTransactions() []*parser.Transaction {
|
|
transactions := b.block.Transactions()
|
|
|
|
lastTransactionId := b.checkpointer.TransactionID()
|
|
if lastTransactionId == "" {
|
|
// No previously processed transactions within this block so all are new
|
|
return transactions
|
|
}
|
|
|
|
// Ignore transactions up to the last processed transaction ID
|
|
lastProcessedIndex := b.findLastProcessedIndex()
|
|
return transactions[lastProcessedIndex+1:]
|
|
}
|
|
|
|
func (b *block) findLastProcessedIndex() int {
|
|
blockTransactionIds := []string{}
|
|
for _, transaction := range b.block.Transactions() {
|
|
blockTransactionIds = append(blockTransactionIds, transaction.ChannelHeader().GetTxId())
|
|
}
|
|
|
|
lastTransactionId := b.checkpointer.TransactionID()
|
|
lastProcessedIndex := -1
|
|
for index, id := range blockTransactionIds {
|
|
if id == lastTransactionId {
|
|
lastProcessedIndex = index
|
|
}
|
|
}
|
|
if lastProcessedIndex < 0 {
|
|
panic(
|
|
fmt.Errorf(
|
|
"checkpoint transaction ID %s not found in block %d containing transactions: %s",
|
|
lastTransactionId,
|
|
b.block.Number(),
|
|
b.joinByComma(blockTransactionIds),
|
|
),
|
|
)
|
|
}
|
|
return lastProcessedIndex
|
|
}
|
|
|
|
func (b *block) joinByComma(list []string) string {
|
|
result := ""
|
|
for index, item := range list {
|
|
if len(list)-1 == index {
|
|
result += item
|
|
} else {
|
|
result += item + ", "
|
|
}
|
|
}
|
|
return result
|
|
}
|