package proposer import ( "fmt" "math/big" "github.com/ethereum/go-ethereum/accounts/abi/bind" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/log" "github.com/ethereum/go-ethereum/sharding" "github.com/ethereum/go-ethereum/sharding/node" ) // createCollation creates collation base struct with header // and body. Header consists of shardID, ChunkRoot, period, // proposer addr and signatures. Body contains serialized blob // of a collations transactions. func createCollation(n node.Node, shardId *big.Int, period *big.Int, txs []*types.Transaction) (*sharding.Collation, error) { // shardId has to be within range if shardId.Cmp(big.NewInt(0)) < 0 || shardId.Cmp(big.NewInt(sharding.ShardCount)) > 0 { return nil, fmt.Errorf("can't create collation for shard %v. Must be between 0 and %v", shardId, sharding.ShardCount) } // check with SMC to see if we can add the header. if a, _ := checkHeaderSubmitted(n, shardId, period); !a { return nil, fmt.Errorf("can't create collation, collation with same period has already been added") } // serialized tx to blob for collation body. blobs, err := sharding.SerializeTxToBlob(txs) if err != nil { return nil, fmt.Errorf("can't create collation, serialization to blob failed: %v", err) } // construct the header, leave chunkRoot and signature fields empty, to be filled later. addr := n.Account().Address header := sharding.NewCollationHeader(shardId, nil, period, &addr, nil) // construct the body with header, blobs(serialized txs) and txs. collation := sharding.NewCollation(header, blobs, txs) collation.CalculateChunkRoot() sig, err := n.Sign(collation.Header().Hash()) if err != nil { return nil, fmt.Errorf("can't create collation, sign collationHeader failed: %v", err) } // add proposer signature to collation header. collation.Header().AddSig(sig) return collation, nil } // addHeader adds the collation header to the main chain by sending // an addHeader transaction to the sharding manager contract. // There can only exist one header per period per shard, it's proposer's // responsibility to check if a header has been added. func addHeader(n node.Node, collation sharding.Collation) error { log.Info("Adding header to SMC") txOps, err := n.CreateTXOpts(big.NewInt(0)) if err != nil { return fmt.Errorf("unable to initiate add header transaction: %v", err) } var chunkRoot [32]byte copy(chunkRoot[:], collation.Header().ChunkRoot().Bytes()) tx, err := n.SMCTransactor().AddHeader(txOps, collation.Header().ShardID(), collation.Header().Period(), chunkRoot) if err != nil { return fmt.Errorf("unable to add header to SMC: %v", err) } receipt, err := n.TransactionReceipt(tx.Hash()) if err != nil { return err } if receipt.Status != types.ReceiptStatusSuccessful { return fmt.Errorf("addHeader transaction failed") } return nil } // checkHeaderSubmitted checks if a collation header has already // submitted to the main chain. There can only be one header per shard // per period, proposer should check if a header's already submitted, // checkHeaderSubmitted returns true if it's available, false if it's unavailable. func checkHeaderSubmitted(n node.Node, shardId *big.Int, period *big.Int) (bool, error) { log.Info("Checking header in shard: %d, period: %d", shardId, period) // Get the period of the last header. lastPeriod, err := n.SMCCaller().LastSubmittedCollation(&bind.CallOpts{}, shardId) if err != nil { return false, fmt.Errorf("unable to get the period of last submitted collation: %v", err) } // True if current period is greater than last added period. return period.Cmp(lastPeriod) > 0, nil }