Evgenii Stratonikov
0e31c12e63
Drop duplicate entities. Format entities. Signed-off-by: Dmitrii Stepanov <d.stepanov@yadro.com> Signed-off-by: Evgenii Stratonikov <e.stratonikov@yadro.com>
336 lines
7.3 KiB
Go
336 lines
7.3 KiB
Go
package audit
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/ecdsa"
|
|
"crypto/elliptic"
|
|
"encoding/hex"
|
|
"math/big"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/internal/logs"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/innerring/processors/settlement/common"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/util/logger"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/audit"
|
|
cid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id"
|
|
oid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/user"
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
// CalculatePrm groups the required parameters of
|
|
// Calculator.CalculateForEpoch call.
|
|
type CalculatePrm struct {
|
|
// Number of epoch to perform the calculation.
|
|
Epoch uint64
|
|
}
|
|
|
|
type singleResultCtx struct {
|
|
eAudit uint64
|
|
|
|
auditResult *audit.Result
|
|
|
|
log *logger.Logger
|
|
|
|
txTable *common.TransferTable
|
|
|
|
cnrInfo common.ContainerInfo
|
|
|
|
cnrNodes []common.NodeInfo
|
|
|
|
passNodes map[string]common.NodeInfo
|
|
|
|
sumSGSize *big.Int
|
|
|
|
auditFee *big.Int
|
|
}
|
|
|
|
var (
|
|
bigGB = big.NewInt(1 << 30)
|
|
bigZero = big.NewInt(0)
|
|
bigOne = big.NewInt(1)
|
|
)
|
|
|
|
// Calculate calculates payments for audit results in a specific epoch of the network.
|
|
// Wraps the results in a money transfer transaction and sends it to the network.
|
|
func (c *Calculator) Calculate(p *CalculatePrm) {
|
|
log := &logger.Logger{Logger: c.opts.log.With(
|
|
zap.Uint64("current epoch", p.Epoch),
|
|
)}
|
|
|
|
if p.Epoch == 0 {
|
|
log.Info(logs.AuditSettlementsAreIgnoredForZeroEpoch)
|
|
return
|
|
}
|
|
|
|
log.Info(logs.AuditCalculateAuditSettlements)
|
|
|
|
log.Debug(logs.AuditGettingResultsForThePreviousEpoch)
|
|
prevEpoch := p.Epoch - 1
|
|
|
|
auditResults, err := c.prm.ResultStorage.AuditResultsForEpoch(prevEpoch)
|
|
if err != nil {
|
|
log.Error(logs.AuditCouldNotCollectAuditResults)
|
|
return
|
|
} else if len(auditResults) == 0 {
|
|
log.Debug(logs.AuditNoAuditResultsInPreviousEpoch)
|
|
return
|
|
}
|
|
|
|
auditFee, err := c.prm.AuditFeeFetcher.AuditFee()
|
|
if err != nil {
|
|
log.Warn(logs.AuditCantFetchAuditFeeFromNetworkConfig,
|
|
zap.String("error", err.Error()))
|
|
auditFee = 0
|
|
}
|
|
|
|
log.Debug(logs.AuditProcessingAuditResults,
|
|
zap.Int("number", len(auditResults)),
|
|
)
|
|
|
|
table := common.NewTransferTable()
|
|
|
|
for i := range auditResults {
|
|
c.processResult(&singleResultCtx{
|
|
log: log,
|
|
auditResult: auditResults[i],
|
|
txTable: table,
|
|
auditFee: big.NewInt(0).SetUint64(auditFee),
|
|
})
|
|
}
|
|
|
|
log.Debug(logs.AuditProcessingTransfers)
|
|
|
|
common.TransferAssets(c.prm.Exchanger, table, common.AuditSettlementDetails(prevEpoch))
|
|
}
|
|
|
|
func (c *Calculator) processResult(ctx *singleResultCtx) {
|
|
ctx.log = &logger.Logger{Logger: ctx.log.With(
|
|
zap.Stringer("cid", ctx.containerID()),
|
|
zap.Uint64("audit epoch", ctx.auditResult.Epoch()),
|
|
)}
|
|
|
|
ctx.log.Debug(logs.AuditReadingInformationAboutTheContainer)
|
|
|
|
ok := c.readContainerInfo(ctx)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
ctx.log.Debug(logs.AuditBuildingPlacement)
|
|
|
|
ok = c.buildPlacement(ctx)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
ctx.log.Debug(logs.AuditCollectingPassedNodes)
|
|
|
|
ok = c.collectPassNodes(ctx)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
ctx.log.Debug(logs.AuditCalculatingSumOfTheSizesOfAllStorageGroups)
|
|
|
|
ok = c.sumSGSizes(ctx)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
ctx.log.Debug(logs.AuditFillingTransferTable)
|
|
|
|
c.fillTransferTable(ctx)
|
|
}
|
|
|
|
func (c *Calculator) readContainerInfo(ctx *singleResultCtx) bool {
|
|
cnr, ok := ctx.auditResult.Container()
|
|
if !ok {
|
|
ctx.log.Error(logs.AuditMissingContainerInAuditResult)
|
|
return false
|
|
}
|
|
|
|
var err error
|
|
|
|
ctx.cnrInfo, err = c.prm.ContainerStorage.ContainerInfo(cnr)
|
|
if err != nil {
|
|
ctx.log.Error(logs.AuditCouldNotGetContainerInfo,
|
|
zap.String("error", err.Error()),
|
|
)
|
|
}
|
|
|
|
return err == nil
|
|
}
|
|
|
|
func (c *Calculator) buildPlacement(ctx *singleResultCtx) bool {
|
|
var err error
|
|
|
|
ctx.cnrNodes, err = c.prm.PlacementCalculator.ContainerNodes(ctx.auditEpoch(), ctx.containerID())
|
|
if err != nil {
|
|
ctx.log.Error(logs.AuditCouldNotGetContainerNodes,
|
|
zap.String("error", err.Error()),
|
|
)
|
|
}
|
|
|
|
empty := len(ctx.cnrNodes) == 0
|
|
if empty {
|
|
ctx.log.Debug(logs.AuditEmptyListOfContainerNodes)
|
|
}
|
|
|
|
return err == nil && !empty
|
|
}
|
|
|
|
func (c *Calculator) collectPassNodes(ctx *singleResultCtx) bool {
|
|
ctx.passNodes = make(map[string]common.NodeInfo)
|
|
|
|
for _, cnrNode := range ctx.cnrNodes {
|
|
// TODO(@cthulhu-rider): neofs-sdk-go#241 use dedicated method
|
|
ctx.auditResult.IteratePassedStorageNodes(func(passNode []byte) bool {
|
|
if !bytes.Equal(cnrNode.PublicKey(), passNode) {
|
|
return true
|
|
}
|
|
|
|
failed := false
|
|
|
|
ctx.auditResult.IterateFailedStorageNodes(func(failNode []byte) bool {
|
|
failed = bytes.Equal(cnrNode.PublicKey(), failNode)
|
|
return !failed
|
|
})
|
|
|
|
if !failed {
|
|
ctx.passNodes[hex.EncodeToString(passNode)] = cnrNode
|
|
}
|
|
|
|
return false
|
|
})
|
|
}
|
|
|
|
empty := len(ctx.passNodes) == 0
|
|
if empty {
|
|
ctx.log.Debug(logs.AuditNoneOfTheContainerNodesPassedTheAudit)
|
|
}
|
|
|
|
return !empty
|
|
}
|
|
|
|
func (c *Calculator) sumSGSizes(ctx *singleResultCtx) bool {
|
|
sumPassSGSize := uint64(0)
|
|
fail := false
|
|
|
|
var addr oid.Address
|
|
addr.SetContainer(ctx.containerID())
|
|
|
|
ctx.auditResult.IteratePassedStorageGroups(func(id oid.ID) bool {
|
|
addr.SetObject(id)
|
|
|
|
sgInfo, err := c.prm.SGStorage.SGInfo(addr)
|
|
if err != nil {
|
|
ctx.log.Error(logs.AuditCouldNotGetSGInfo,
|
|
zap.String("id", id.String()),
|
|
zap.String("error", err.Error()),
|
|
)
|
|
|
|
fail = true
|
|
|
|
return false // we also can continue and calculate at least some part
|
|
}
|
|
|
|
sumPassSGSize += sgInfo.Size()
|
|
|
|
return true
|
|
})
|
|
|
|
if fail {
|
|
return false
|
|
}
|
|
|
|
if sumPassSGSize == 0 {
|
|
ctx.log.Debug(logs.AuditZeroSumSGSize)
|
|
return false
|
|
}
|
|
|
|
ctx.sumSGSize = big.NewInt(int64(sumPassSGSize))
|
|
|
|
return true
|
|
}
|
|
|
|
func (c *Calculator) fillTransferTable(ctx *singleResultCtx) bool {
|
|
cnrOwner := ctx.cnrInfo.Owner()
|
|
|
|
// add txs to pay for storage node
|
|
for k, info := range ctx.passNodes {
|
|
ownerID, err := c.prm.AccountStorage.ResolveKey(info)
|
|
if err != nil {
|
|
ctx.log.Error(logs.AuditCouldNotResolvePublicKeyOfTheStorageNode,
|
|
zap.String("error", err.Error()),
|
|
zap.String("key", k),
|
|
)
|
|
|
|
return false // we also can continue and calculate at least some part
|
|
}
|
|
|
|
price := info.Price()
|
|
|
|
ctx.log.Debug(logs.AuditCalculatingStorageNodeSalaryForAudit,
|
|
zap.Stringer("sum SG size", ctx.sumSGSize),
|
|
zap.Stringer("price", price),
|
|
)
|
|
|
|
fee := big.NewInt(0).Mul(price, ctx.sumSGSize)
|
|
fee.Div(fee, bigGB)
|
|
|
|
if fee.Cmp(bigZero) == 0 {
|
|
fee.Add(fee, bigOne)
|
|
}
|
|
|
|
ctx.txTable.Transfer(&common.TransferTx{
|
|
From: cnrOwner,
|
|
To: *ownerID,
|
|
Amount: fee,
|
|
})
|
|
}
|
|
|
|
// add txs to pay inner ring node for audit result
|
|
auditIR, err := ownerFromKey(ctx.auditResult.AuditorKey())
|
|
if err != nil {
|
|
ctx.log.Error(logs.AuditCouldNotParsePublicKeyOfTheInnerRingNode,
|
|
zap.String("error", err.Error()),
|
|
zap.String("key", hex.EncodeToString(ctx.auditResult.AuditorKey())),
|
|
)
|
|
|
|
return false
|
|
}
|
|
|
|
ctx.txTable.Transfer(&common.TransferTx{
|
|
From: cnrOwner,
|
|
To: *auditIR,
|
|
Amount: ctx.auditFee,
|
|
})
|
|
|
|
return false
|
|
}
|
|
|
|
func (c *singleResultCtx) containerID() cid.ID {
|
|
cnr, _ := c.auditResult.Container()
|
|
return cnr
|
|
}
|
|
|
|
func (c *singleResultCtx) auditEpoch() uint64 {
|
|
if c.eAudit == 0 {
|
|
c.eAudit = c.auditResult.Epoch()
|
|
}
|
|
|
|
return c.eAudit
|
|
}
|
|
|
|
func ownerFromKey(key []byte) (*user.ID, error) {
|
|
pubKey, err := keys.NewPublicKeyFromBytes(key, elliptic.P256())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var id user.ID
|
|
user.IDFromKey(&id, (ecdsa.PublicKey)(*pubKey))
|
|
|
|
return &id, nil
|
|
}
|