package blockchain import ( "bytes" "crypto/sha256" "encoding/gob" "encoding/hex" "fmt" "log" ) // Transaction represents a Bitcoin transaction type Transaction struct { ID []byte // 32 byte hash TxInput []TXInput TxOutput []TXOutput } // TXOutput is the struct of transaction output in a transaction. type TXOutput struct { Value int Address string } // TXInput is the struct of transaction input in a transaction. type TXInput struct { TxID []byte TxOutputIndex int Address string } // DefaultCoinbaseValue is the default value of coinbase transaction. const DefaultCoinbaseValue = 1000 // SetID sets ID of a transaction (32 byte hash of the whole transaction) func (tx *Transaction) SetID() { var encoded bytes.Buffer var hash [32]byte enc := gob.NewEncoder(&encoded) err := enc.Encode(tx) if err != nil { log.Panic(err) } hash = sha256.Sum256(encoded.Bytes()) tx.ID = hash[:] } // NewCoinbaseTX creates a new coinbase transaction func NewCoinbaseTX(to, data string) *Transaction { if data == "" { data = fmt.Sprintf("Reward to '%s'", to) } txin := TXInput{[]byte{}, -1, data} txout := TXOutput{DefaultCoinbaseValue, to} tx := Transaction{nil, []TXInput{txin}, []TXOutput{txout}} tx.SetID() return &tx } // Used for debuging. func (txInput *TXInput) String() string { res := fmt.Sprintf("TxID: %v, ", hex.EncodeToString(txInput.TxID)) res += fmt.Sprintf("TxOutputIndex: %v, ", txInput.TxOutputIndex) res += fmt.Sprintf("Address: %v", txInput.Address) return res } // Used for debuging. func (txOutput *TXOutput) String() string { res := fmt.Sprintf("Value: %v, ", txOutput.Value) res += fmt.Sprintf("Address: %v", txOutput.Address) return res } // Used for debuging. func (tx *Transaction) String() string { res := fmt.Sprintf("ID: %v\n", hex.EncodeToString(tx.ID)) res += fmt.Sprintf("TxInput:\n") for id, value := range tx.TxInput { res += fmt.Sprintf("%v: %v\n", id, value.String()) } res += fmt.Sprintf("TxOutput:\n") for id, value := range tx.TxOutput { res += fmt.Sprintf("%v: %v\n", id, value.String()) } return res }