func send(packet common.Packet, pipe *common.PeerPipe) error { log.Current.Tracef("SyncProxy.send(): sending packet %s to peer (TCP %s)", packet.Name(), pipe.GetAddr()) if !pipe.Send(packet) { return common.NewError(common.SERVER_ERROR, fmt.Sprintf("SyncProxy.listen(): Fail to send packet %s to peer (TCP %s)", packet.Name(), pipe.GetAddr())) } return nil }
// // Add a follower and starts a listener for the follower. // If the leader is terminated, the pipe between leader // and follower will also be closed. // func (l *Leader) AddFollower(fid string, peer *common.PeerPipe, o *observer) { l.mutex.Lock() defer l.mutex.Unlock() // AddFollower requires holding the mutex such that the leader thread // will not be sending new proposal or commit (see sendProposal() and // sendCommit()) to followers. This allow this function to copy the // proposals and commits from the observer queue into the pipe, before // the leader has a chance to send new messages. for packet := o.getNext(); packet != nil; packet = o.getNext() { switch request := packet.(type) { case ProposalMsg: txid := common.Txnid(request.GetTxnid()) log.Current.Debugf("Leader.AddFollower() : send observer's packet %s, txid %d", packet.Name(), txid) case CommitMsg: txid := common.Txnid(request.GetTxnid()) log.Current.Debugf("Leader.AddFollower() : send observer's packet %s, txid %d", packet.Name(), txid) } peer.Send(packet) } // Rememeber the old message listener and start a new one. oldListener, ok := l.followers[fid] listener := newListener(fid, peer, l) l.followers[fid] = listener go listener.start() // kill the old message listener if ok && oldListener != nil { log.Current.Debugf("Leader.AddFollower() : old Listener found for follower %s. Terminating old listener", fid) oldListener.terminate() } else { // notify a brand new follower (not just replacing an existing one) l.changech <- true } }