Skip to content
Snippets Groups Projects
Select Git revision
  • 070f4c32af8bbc0cfed827b04eab148b60f5c916
  • without_tipselection default
  • develop protected
  • fix/grafana-local-dashboard
  • wasp
  • fix/dashboard-explorer-freeze
  • master
  • feat/timerqueue
  • test/sync_debug_and_650
  • feat/sync_revamp_inv
  • wip/sync
  • tool/db-recovery
  • portcheck/fix
  • fix/synchronization
  • feat/new-dashboard-analysis
  • feat/refactored-analysis-dashboard
  • feat/new-analysis-dashboard
  • test/demo-prometheus-fpc
  • prometheus_metrics
  • wip/analysis-server
  • merge/fpc-test-value-transfer
  • v0.2.2
  • v0.2.1
  • v0.2.0
  • v0.1.3
  • v0.1.2
  • v0.1.1
  • v0.1.0
28 results

transfer_output.go

Blame
  • user avatar
    Hans Moog authored
    070f4c32
    History
    transfer_output.go 6.65 KiB
    package ledgerstate
    
    import (
    	"encoding/binary"
    	"sync"
    
    	"github.com/iotaledger/goshimmer/packages/objectstorage"
    	"github.com/iotaledger/goshimmer/packages/stringify"
    )
    
    type TransferOutput struct {
    	transferHash TransferHash
    	addressHash  AddressHash
    	balances     []*ColoredBalance
    	realityId    RealityId
    	consumers    map[TransferHash]empty
    
    	storageKey  []byte
    	ledgerState *LedgerState
    
    	realityIdMutex sync.RWMutex
    	consumersMutex sync.RWMutex
    }
    
    func NewTransferOutput(ledgerState *LedgerState, realityId RealityId, transferHash TransferHash, addressHash AddressHash, balances ...*ColoredBalance) *TransferOutput {
    	return &TransferOutput{
    		transferHash: transferHash,
    		addressHash:  addressHash,
    		balances:     balances,
    		realityId:    realityId,
    		consumers:    make(map[TransferHash]empty),
    
    		storageKey:  append(transferHash[:], addressHash[:]...),
    		ledgerState: ledgerState,
    	}
    }
    
    func (transferOutput *TransferOutput) GetRealityId() (realityId RealityId) {
    	transferOutput.realityIdMutex.RLock()
    	realityId = transferOutput.realityId
    	transferOutput.realityIdMutex.RUnlock()
    
    	return
    }
    
    func (transferOutput *TransferOutput) SetRealityId(realityId RealityId) {
    	transferOutput.realityIdMutex.Lock()
    	transferOutput.realityId = realityId
    	transferOutput.realityIdMutex.Unlock()
    }
    
    func (transferOutput *TransferOutput) GetBalances() []*ColoredBalance {
    	return transferOutput.balances
    }
    
    func (transferOutput *TransferOutput) addConsumer(consumer TransferHash) bool {
    	transferOutput.consumersMutex.RLock()
    	if _, exist := transferOutput.consumers[consumer]; exist {
    		transferOutput.consumersMutex.RUnlock()
    
    		return false
    	} else {
    		transferOutput.consumersMutex.RUnlock()
    
    		transferOutput.consumersMutex.Lock()
    		if len(transferOutput.consumers) == 0 {
    			transferOutput.markAsSpent()
    		} else {
    			panic("DOUBLE SPEND DETECTED")
    		}
    		transferOutput.consumers[consumer] = void
    		transferOutput.consumersMutex.Unlock()
    
    		return true
    	}
    }
    
    func (transferOutput *TransferOutput) getConsumers() (consumers map[TransferHash]empty) {
    	transferOutput.consumersMutex.RLock()
    	consumers = transferOutput.consumers
    	transferOutput.consumersMutex.RUnlock()
    
    	return
    }
    
    func (transferOutput *TransferOutput) markAsSpent() {
    	currentBookingKey := generateTransferOutputBookingStorageKey(transferOutput.realityId, transferOutput.addressHash, false, transferOutput.transferHash)
    
    	if cachedTransferOutputBooking, err := transferOutput.ledgerState.transferOutputBookings.Load(currentBookingKey); err != nil {
    		panic(err)
    	} else if !cachedTransferOutputBooking.Exists() {
    		panic("could not find TransferOutputBooking")
    	} else {
    		transferOutputBooking := cachedTransferOutputBooking.Get().(*TransferOutputBooking)
    		transferOutput.ledgerState.storeTransferOutputBooking(newTransferOutputBooking(transferOutputBooking.GetRealityId(), transferOutputBooking.GetAddressHash(), true, transferOutputBooking.GetTransferHash())).Release()
    
    		cachedTransferOutputBooking.Delete()
    		cachedTransferOutputBooking.Release()
    	}
    }
    
    func (transferOutput *TransferOutput) String() string {
    	return stringify.Struct("TransferOutput",
    		stringify.StructField("transferHash", transferOutput.transferHash.String()),
    		stringify.StructField("addressHash", transferOutput.addressHash.String()),
    		stringify.StructField("balances", transferOutput.balances),
    		stringify.StructField("realityId", transferOutput.realityId.String()),
    		stringify.StructField("spent", len(transferOutput.consumers) >= 1),
    	)
    }
    
    // region support object storage ///////////////////////////////////////////////////////////////////////////////////////
    
    func (transferOutput *TransferOutput) GetStorageKey() []byte {
    	return transferOutput.storageKey
    }
    
    func (transferOutput *TransferOutput) Update(other objectstorage.StorableObject) {}
    
    func (transferOutput *TransferOutput) MarshalBinary() ([]byte, error) {
    	balanceCount := len(transferOutput.balances)
    	consumerCount := len(transferOutput.consumers)
    
    	result := make([]byte, realityIdLength+4+balanceCount*coloredBalanceLength+4+consumerCount*transferHashLength)
    
    	copy(result[0:], transferOutput.realityId[:])
    
    	binary.LittleEndian.PutUint32(result[realityIdLength:], uint32(balanceCount))
    	for i := 0; i < balanceCount; i++ {
    		copy(result[realityIdLength+4+i*coloredBalanceLength:], transferOutput.balances[i].color[:colorLength])
    		binary.LittleEndian.PutUint64(result[realityIdLength+4+i*coloredBalanceLength+colorLength:], transferOutput.balances[i].balance)
    	}
    
    	offset := realityIdLength + 4 + balanceCount*coloredBalanceLength
    	binary.LittleEndian.PutUint32(result[offset:], uint32(consumerCount))
    	offset += 4
    	for consumer := range transferOutput.consumers {
    		copy(result[offset:], consumer[:transferHashLength])
    		offset += transferHashLength
    	}
    
    	return result, nil
    }
    
    func (transferOutput *TransferOutput) UnmarshalBinary(serializedObject []byte) error {
    	if err := transferOutput.transferHash.UnmarshalBinary(transferOutput.storageKey[:transferHashLength]); err != nil {
    		return err
    	}
    
    	if err := transferOutput.addressHash.UnmarshalBinary(transferOutput.storageKey[transferHashLength:]); err != nil {
    		return err
    	}
    
    	if err := transferOutput.realityId.UnmarshalBinary(serializedObject[:realityIdLength]); err != nil {
    		return err
    	}
    
    	if balances, err := transferOutput.unmarshalBalances(serializedObject[realityIdLength:]); err != nil {
    		return err
    	} else {
    		transferOutput.balances = balances
    	}
    
    	if consumers, err := transferOutput.unmarshalConsumers(serializedObject[realityIdLength+4+len(transferOutput.balances)*coloredBalanceLength:]); err != nil {
    		return err
    	} else {
    		transferOutput.consumers = consumers
    	}
    
    	return nil
    }
    
    func (transferOutput *TransferOutput) unmarshalBalances(serializedBalances []byte) ([]*ColoredBalance, error) {
    	balanceCount := int(binary.LittleEndian.Uint32(serializedBalances))
    
    	balances := make([]*ColoredBalance, balanceCount)
    	for i := 0; i < balanceCount; i++ {
    		coloredBalance := ColoredBalance{}
    		if err := coloredBalance.UnmarshalBinary(serializedBalances[4+i*coloredBalanceLength:]); err != nil {
    			return nil, err
    		}
    
    		balances[i] = &coloredBalance
    	}
    
    	return balances, nil
    }
    
    func (transferOutput *TransferOutput) unmarshalConsumers(serializedConsumers []byte) (map[TransferHash]empty, error) {
    	consumerCount := int(binary.LittleEndian.Uint32(serializedConsumers))
    
    	consumers := make(map[TransferHash]empty, consumerCount)
    	for i := 0; i < consumerCount; i++ {
    		transferHash := TransferHash{}
    		if err := transferHash.UnmarshalBinary(serializedConsumers[4+i*transferHashLength:]); err != nil {
    			return nil, err
    		}
    
    		consumers[transferHash] = void
    	}
    
    	return consumers, nil
    }
    
    // endregion ///////////////////////////////////////////////////////////////////////////////////////////////////////////