// Copyright 2017-2021 DERO Project. All rights reserved. // Use of this source code in any form is governed by RESEARCH license. // license can be found in the LICENSE file. // GPG: 0F39 E425 8C65 3947 702A 8234 08B2 0360 A03A 9DE8 // // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY // EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL // THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, // STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF // THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. package p2p //import "fmt" //import "net" import "sync/atomic" import "time" //import "container/list" //import log "github.com/sirupsen/logrus" import "github.com/romana/rlog" import "github.com/vmihailenco/msgpack" //import "github.com/deroproject/derosuite/crypto" //import "github.com/deroproject/derosuite/globals" import "github.com/deroproject/derohe/block" import "github.com/deroproject/derohe/errormsg" import "github.com/deroproject/derohe/transaction" // peer has responded with some objects, we must respond func (connection *Connection) Handle_ObjectResponse(buf []byte) { var response Object_Response_struct err := msgpack.Unmarshal(buf, &response) if err != nil { rlog.Warnf("Error while decoding incoming object response err %s %s", err, connection.logid) connection.Exit() } var expected Queued_Command select { case expected = <-connection.Objects: default: // if nothing is on queue the peer sent us bogus request, rlog.Warnf("Peer sent us a chain response, when we didnot request chain, Exiting, may be block the peer %s", connection.logid) connection.Exit() } if expected.Command != V2_COMMAND_OBJECTS_RESPONSE { rlog.Warnf("We were waiting for a different object, but peer sent something else, Exiting, may be block the peer %s", connection.logid) connection.Exit() } // we need to verify common and update common if len(response.CBlocks) != len(expected.BLID) { // we requested x block , peer sent us y blocks, time to ban peer rlog.Warnf("we got %d response for %d requests %s %s", len(response.CBlocks), len(expected.BLID), connection.logid) } if len(response.Txs) != len(expected.TXID) { // we requested x block , peer sent us y blocks, time to ban peer rlog.Warnf("we got %d response for %d requests %s %s", len(response.CBlocks), len(expected.BLID), connection.logid) } for i := 0; i < len(response.CBlocks); i++ { // process incoming full blocks var cbl block.Complete_Block // parse incoming block and deserialize it var bl block.Block // lets deserialize block first and see whether it is the requested object cbl.Bl = &bl err := bl.Deserialize(response.CBlocks[i].Block) if err != nil { // we have a block which could not be deserialized ban peer rlog.Warnf("Error Incoming block could not be deserilised err %s %s", err, connection.logid) connection.Exit() return } // check if deserialized hash is same as what we requested if bl.GetHash() != expected.BLID[i] { // user is trying to spoof block, ban hime connection.logger.Warnf("requested and response block mismatch") rlog.Warnf("Error block hash mismatch Actual %s Expected %s err %s %s", bl.GetHash(), expected.BLID[i], connection.logid) connection.Exit() } // give the chain some more time to respond atomic.StoreInt64(&connection.LastObjectRequestTime, time.Now().Unix()) // check whether the object was requested one // complete the txs for j := range response.CBlocks[i].Txs { var tx transaction.Transaction err = tx.DeserializeHeader(response.CBlocks[i].Txs[j]) if err != nil { // we have a tx which could not be deserialized ban peer rlog.Warnf("Error Incoming TX could not be deserialized err %s %s", err, connection.logid) connection.Exit() return } cbl.Txs = append(cbl.Txs, &tx) } // check if we can add ourselves to chain err, ok := chain.Add_Complete_Block(&cbl) if !ok && err == errormsg.ErrInvalidPoW { connection.logger.Warnf("This peer should be banned") connection.Exit() return } if !ok && err == errormsg.ErrPastMissing { rlog.Warnf("Error Incoming Block coould not be added due to missing past, so skipping future block err %s %s", err, connection.logid) return } // add the object to object pool from where it will be consume // queue_block_received(bl.GetHash(),&cbl) } for i := range response.Txs { // process incoming txs for mempool if !chain.Mempool.Mempool_TX_Exist(expected.TXID[i]) { // we still donot have it, so try to process it var tx transaction.Transaction err = tx.DeserializeHeader(response.Txs[i]) if err != nil { // we have a tx which could not be deserialized ban peer rlog.Warnf("Error Incoming TX could not be deserialized err %s %s", err, connection.logid) connection.Exit() return } chain.Add_TX_To_Pool(&tx) // currently we are ignoring error } } }