github.com/Ethersocial/go-esn@v0.3.7/swarm/network/stream/delivery.go (about) 1 // Copyright 2018 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 package stream 18 19 import ( 20 "context" 21 "errors" 22 23 "fmt" 24 25 "github.com/ethersocial/go-esn/metrics" 26 "github.com/ethersocial/go-esn/p2p/enode" 27 "github.com/ethersocial/go-esn/swarm/log" 28 "github.com/ethersocial/go-esn/swarm/network" 29 "github.com/ethersocial/go-esn/swarm/spancontext" 30 "github.com/ethersocial/go-esn/swarm/storage" 31 opentracing "github.com/opentracing/opentracing-go" 32 ) 33 34 const ( 35 swarmChunkServerStreamName = "RETRIEVE_REQUEST" 36 deliveryCap = 32 37 ) 38 39 var ( 40 processReceivedChunksCount = metrics.NewRegisteredCounter("network.stream.received_chunks.count", nil) 41 handleRetrieveRequestMsgCount = metrics.NewRegisteredCounter("network.stream.handle_retrieve_request_msg.count", nil) 42 43 requestFromPeersCount = metrics.NewRegisteredCounter("network.stream.request_from_peers.count", nil) 44 requestFromPeersEachCount = metrics.NewRegisteredCounter("network.stream.request_from_peers_each.count", nil) 45 ) 46 47 type Delivery struct { 48 chunkStore storage.SyncChunkStore 49 kad *network.Kademlia 50 getPeer func(enode.ID) *Peer 51 } 52 53 func NewDelivery(kad *network.Kademlia, chunkStore storage.SyncChunkStore) *Delivery { 54 return &Delivery{ 55 chunkStore: chunkStore, 56 kad: kad, 57 } 58 } 59 60 // SwarmChunkServer implements Server 61 type SwarmChunkServer struct { 62 deliveryC chan []byte 63 batchC chan []byte 64 chunkStore storage.ChunkStore 65 currentLen uint64 66 quit chan struct{} 67 } 68 69 // NewSwarmChunkServer is SwarmChunkServer constructor 70 func NewSwarmChunkServer(chunkStore storage.ChunkStore) *SwarmChunkServer { 71 s := &SwarmChunkServer{ 72 deliveryC: make(chan []byte, deliveryCap), 73 batchC: make(chan []byte), 74 chunkStore: chunkStore, 75 quit: make(chan struct{}), 76 } 77 go s.processDeliveries() 78 return s 79 } 80 81 // processDeliveries handles delivered chunk hashes 82 func (s *SwarmChunkServer) processDeliveries() { 83 var hashes []byte 84 var batchC chan []byte 85 for { 86 select { 87 case <-s.quit: 88 return 89 case hash := <-s.deliveryC: 90 hashes = append(hashes, hash...) 91 batchC = s.batchC 92 case batchC <- hashes: 93 hashes = nil 94 batchC = nil 95 } 96 } 97 } 98 99 // SetNextBatch 100 func (s *SwarmChunkServer) SetNextBatch(_, _ uint64) (hashes []byte, from uint64, to uint64, proof *HandoverProof, err error) { 101 select { 102 case hashes = <-s.batchC: 103 case <-s.quit: 104 return 105 } 106 107 from = s.currentLen 108 s.currentLen += uint64(len(hashes)) 109 to = s.currentLen 110 return 111 } 112 113 // Close needs to be called on a stream server 114 func (s *SwarmChunkServer) Close() { 115 close(s.quit) 116 } 117 118 // GetData retrives chunk data from db store 119 func (s *SwarmChunkServer) GetData(ctx context.Context, key []byte) ([]byte, error) { 120 chunk, err := s.chunkStore.Get(ctx, storage.Address(key)) 121 if err != nil { 122 return nil, err 123 } 124 return chunk.Data(), nil 125 } 126 127 // RetrieveRequestMsg is the protocol msg for chunk retrieve requests 128 type RetrieveRequestMsg struct { 129 Addr storage.Address 130 SkipCheck bool 131 HopCount uint8 132 } 133 134 func (d *Delivery) handleRetrieveRequestMsg(ctx context.Context, sp *Peer, req *RetrieveRequestMsg) error { 135 log.Trace("received request", "peer", sp.ID(), "hash", req.Addr) 136 handleRetrieveRequestMsgCount.Inc(1) 137 138 var osp opentracing.Span 139 ctx, osp = spancontext.StartSpan( 140 ctx, 141 "retrieve.request") 142 defer osp.Finish() 143 144 s, err := sp.getServer(NewStream(swarmChunkServerStreamName, "", false)) 145 if err != nil { 146 return err 147 } 148 streamer := s.Server.(*SwarmChunkServer) 149 150 var cancel func() 151 // TODO: do something with this hardcoded timeout, maybe use TTL in the future 152 ctx = context.WithValue(ctx, "peer", sp.ID().String()) 153 ctx = context.WithValue(ctx, "hopcount", req.HopCount) 154 ctx, cancel = context.WithTimeout(ctx, network.RequestTimeout) 155 156 go func() { 157 select { 158 case <-ctx.Done(): 159 case <-streamer.quit: 160 } 161 cancel() 162 }() 163 164 go func() { 165 chunk, err := d.chunkStore.Get(ctx, req.Addr) 166 if err != nil { 167 log.Warn("ChunkStore.Get can not retrieve chunk", "err", err) 168 return 169 } 170 if req.SkipCheck { 171 err = sp.Deliver(ctx, chunk, s.priority) 172 if err != nil { 173 log.Warn("ERROR in handleRetrieveRequestMsg", "err", err) 174 } 175 return 176 } 177 select { 178 case streamer.deliveryC <- chunk.Address()[:]: 179 case <-streamer.quit: 180 } 181 182 }() 183 184 return nil 185 } 186 187 type ChunkDeliveryMsg struct { 188 Addr storage.Address 189 SData []byte // the stored chunk Data (incl size) 190 peer *Peer // set in handleChunkDeliveryMsg 191 } 192 193 // TODO: Fix context SNAFU 194 func (d *Delivery) handleChunkDeliveryMsg(ctx context.Context, sp *Peer, req *ChunkDeliveryMsg) error { 195 var osp opentracing.Span 196 ctx, osp = spancontext.StartSpan( 197 ctx, 198 "chunk.delivery") 199 defer osp.Finish() 200 201 processReceivedChunksCount.Inc(1) 202 203 go func() { 204 req.peer = sp 205 err := d.chunkStore.Put(ctx, storage.NewChunk(req.Addr, req.SData)) 206 if err != nil { 207 if err == storage.ErrChunkInvalid { 208 // we removed this log because it spams the logs 209 // TODO: Enable this log line 210 // log.Warn("invalid chunk delivered", "peer", sp.ID(), "chunk", req.Addr, ) 211 req.peer.Drop(err) 212 } 213 } 214 }() 215 return nil 216 } 217 218 // RequestFromPeers sends a chunk retrieve request to 219 func (d *Delivery) RequestFromPeers(ctx context.Context, req *network.Request) (*enode.ID, chan struct{}, error) { 220 requestFromPeersCount.Inc(1) 221 var sp *Peer 222 spID := req.Source 223 224 if spID != nil { 225 sp = d.getPeer(*spID) 226 if sp == nil { 227 return nil, nil, fmt.Errorf("source peer %v not found", spID.String()) 228 } 229 } else { 230 d.kad.EachConn(req.Addr[:], 255, func(p *network.Peer, po int, nn bool) bool { 231 id := p.ID() 232 // TODO: skip light nodes that do not accept retrieve requests 233 if req.SkipPeer(id.String()) { 234 log.Trace("Delivery.RequestFromPeers: skip peer", "peer id", id) 235 return true 236 } 237 sp = d.getPeer(id) 238 if sp == nil { 239 log.Warn("Delivery.RequestFromPeers: peer not found", "id", id) 240 return true 241 } 242 spID = &id 243 return false 244 }) 245 if sp == nil { 246 return nil, nil, errors.New("no peer found") 247 } 248 } 249 250 err := sp.SendPriority(ctx, &RetrieveRequestMsg{ 251 Addr: req.Addr, 252 SkipCheck: req.SkipCheck, 253 HopCount: req.HopCount, 254 }, Top) 255 if err != nil { 256 return nil, nil, err 257 } 258 requestFromPeersEachCount.Inc(1) 259 260 return spID, sp.quit, nil 261 }