etcd watch 实现原理

news/2025/1/17 14:00:41/文章来源:https://www.cnblogs.com/daemon365/p/18240641

介绍

在 etcd 中,watch 是一个非常重要的特性,它可以让客户端监控 etcd 中的 key 或者一组 key,当 key 发生变化时,etcd 会通知客户端。本文将介绍 etcd watch 的实现原理。

etcdctl watch /test
# 当 /test 的值发生变化时,会输出如下信息
PUT
/test
a
PUT
/test
b
DELETE
/test

watch 的 api

etcd watch api 是由 grpc stream 实现的,客户端通过 grpc stream 发送 watch 请求,etcd 会将 key 的变化通过 stream 返回给客户端。

rpc Watch(stream WatchRequest) returns (stream WatchResponse) {option (google.api.http) = {post: "/v3/watch"body: "*"};
}

api 实现

func (ws *watchServer) Watch(stream pb.Watch_WatchServer) (err error) {sws := serverWatchStream{lg: ws.lg,clusterID: ws.clusterID,memberID:  ws.memberID,maxRequestBytes: ws.maxRequestBytes,sg:        ws.sg,watchable: ws.watchable,ag:        ws.ag,gRPCStream:  stream,watchStream: ws.watchable.NewWatchStream(),// chan for sending control response like watcher created and canceled.ctrlStream: make(chan *pb.WatchResponse, ctrlStreamBufLen),progress: make(map[mvcc.WatchID]bool),prevKV:   make(map[mvcc.WatchID]bool),fragment: make(map[mvcc.WatchID]bool),closec: make(chan struct{}),}sws.wg.Add(1)go func() {// 开启一个 goroutine 处理新的 event 然后发送给客户端sws.sendLoop()sws.wg.Done()}()errc := make(chan error, 1)go func() {// 开启一个 goroutine 处理客户端发送的 watch 请求if rerr := sws.recvLoop(); rerr != nil {if isClientCtxErr(stream.Context().Err(), rerr) {sws.lg.Debug("failed to receive watch request from gRPC stream", zap.Error(rerr))} else {sws.lg.Warn("failed to receive watch request from gRPC stream", zap.Error(rerr))streamFailures.WithLabelValues("receive", "watch").Inc()}errc <- rerr}}()// 处理结束select {case err = <-errc:if err == context.Canceled {err = rpctypes.ErrGRPCWatchCanceled}close(sws.ctrlStream)case <-stream.Context().Done():err = stream.Context().Err()if err == context.Canceled {err = rpctypes.ErrGRPCWatchCanceled}}sws.close()return err
}

这里 主要的逻辑是开启两个 goroutine,一个用于处理客户端发送的 watch 请求,另一个用于处理新的 event 然后发送给客户端。

sendLoop

func (sws *serverWatchStream) sendLoop() {// watch ids that are currently activeids := make(map[mvcc.WatchID]struct{})// watch responses pending on a watch id creation messagepending := make(map[mvcc.WatchID][]*pb.WatchResponse)interval := GetProgressReportInterval()progressTicker := time.NewTicker(interval)defer func() {progressTicker.Stop()// 清空chan ,清理待处理 eventfor ws := range sws.watchStream.Chan() {mvcc.ReportEventReceived(len(ws.Events))}for _, wrs := range pending {for _, ws := range wrs {mvcc.ReportEventReceived(len(ws.Events))}}}()for {select {case wresp, ok := <-sws.watchStream.Chan():// 从 watchStream.Chan() 中获取 event// 然后发送给客户端 if !ok {return}evs := wresp.Eventsevents := make([]*mvccpb.Event, len(evs))sws.mu.RLock()needPrevKV := sws.prevKV[wresp.WatchID]sws.mu.RUnlock()for i := range evs {events[i] = &evs[i]if needPrevKV && !IsCreateEvent(evs[i]) {opt := mvcc.RangeOptions{Rev: evs[i].Kv.ModRevision - 1}r, err := sws.watchable.Range(context.TODO(), evs[i].Kv.Key, nil, opt)if err == nil && len(r.KVs) != 0 {events[i].PrevKv = &(r.KVs[0])}}}canceled := wresp.CompactRevision != 0wr := &pb.WatchResponse{Header:          sws.newResponseHeader(wresp.Revision),WatchId:         int64(wresp.WatchID),Events:          events,CompactRevision: wresp.CompactRevision,Canceled:        canceled,}// Progress notifications can have WatchID -1// if they announce on behalf of multiple watchersif wresp.WatchID != clientv3.InvalidWatchID {if _, okID := ids[wresp.WatchID]; !okID {// buffer if id not yet announcedwrs := append(pending[wresp.WatchID], wr)pending[wresp.WatchID] = wrscontinue}}mvcc.ReportEventReceived(len(evs))sws.mu.RLock()fragmented, ok := sws.fragment[wresp.WatchID]sws.mu.RUnlock()var serr error// gofail: var beforeSendWatchResponse struct{}if !fragmented && !ok {serr = sws.gRPCStream.Send(wr)} else {serr = sendFragments(wr, sws.maxRequestBytes, sws.gRPCStream.Send)}if serr != nil {if isClientCtxErr(sws.gRPCStream.Context().Err(), serr) {sws.lg.Debug("failed to send watch response to gRPC stream", zap.Error(serr))} else {sws.lg.Warn("failed to send watch response to gRPC stream", zap.Error(serr))streamFailures.WithLabelValues("send", "watch").Inc()}return}sws.mu.Lock()if len(evs) > 0 && sws.progress[wresp.WatchID] {// elide next progress update if sent a key updatesws.progress[wresp.WatchID] = false}sws.mu.Unlock()case c, ok := <-sws.ctrlStream:// 处理客户端发送的 watch 请求if !ok {return}if err := sws.gRPCStream.Send(c); err != nil {if isClientCtxErr(sws.gRPCStream.Context().Err(), err) {sws.lg.Debug("failed to send watch control response to gRPC stream", zap.Error(err))} else {sws.lg.Warn("failed to send watch control response to gRPC stream", zap.Error(err))streamFailures.WithLabelValues("send", "watch").Inc()}return}// track id creationwid := mvcc.WatchID(c.WatchId)verify.Assert(!(c.Canceled && c.Created) || wid == clientv3.InvalidWatchID, "unexpected watchId: %d, wanted: %d, since both 'Canceled' and 'Created' are true", wid, clientv3.InvalidWatchID)if c.Canceled && wid != clientv3.InvalidWatchID {delete(ids, wid)continue}if c.Created {// flush buffered eventsids[wid] = struct{}{}for _, v := range pending[wid] {mvcc.ReportEventReceived(len(v.Events))if err := sws.gRPCStream.Send(v); err != nil {if isClientCtxErr(sws.gRPCStream.Context().Err(), err) {sws.lg.Debug("failed to send pending watch response to gRPC stream", zap.Error(err))} else {sws.lg.Warn("failed to send pending watch response to gRPC stream", zap.Error(err))streamFailures.WithLabelValues("send", "watch").Inc()}return}}delete(pending, wid)}case <-progressTicker.C:sws.mu.Lock()for id, ok := range sws.progress {if ok {sws.watchStream.RequestProgress(id)}sws.progress[id] = true}sws.mu.Unlock()case <-sws.closec:return}}
}

这里使用了 for select 循环:

  1. 从 watchStream.Chan() 中获取 event 然后发送给客户端。
  2. 处理客户端发送的 watch 请求。
  3. dispatch progress 事件。
  4. 处理结束。

recvLoop

func (sws *serverWatchStream) recvLoop() error {for {req, err := sws.gRPCStream.Recv()if err == io.EOF {return nil}if err != nil {return err}switch uv := req.RequestUnion.(type) {case *pb.WatchRequest_CreateRequest:if uv.CreateRequest == nil {break}creq := uv.CreateRequestif len(creq.Key) == 0 {// \x00 is the smallest keycreq.Key = []byte{0}}if len(creq.RangeEnd) == 0 {// force nil since watchstream.Watch distinguishes// between nil and []byte{} for single key / >=creq.RangeEnd = nil}if len(creq.RangeEnd) == 1 && creq.RangeEnd[0] == 0 {// support  >= key queriescreq.RangeEnd = []byte{}}err := sws.isWatchPermitted(creq)if err != nil {var cancelReason stringswitch err {case auth.ErrInvalidAuthToken:cancelReason = rpctypes.ErrGRPCInvalidAuthToken.Error()case auth.ErrAuthOldRevision:cancelReason = rpctypes.ErrGRPCAuthOldRevision.Error()case auth.ErrUserEmpty:cancelReason = rpctypes.ErrGRPCUserEmpty.Error()default:if err != auth.ErrPermissionDenied {sws.lg.Error("unexpected error code", zap.Error(err))}cancelReason = rpctypes.ErrGRPCPermissionDenied.Error()}wr := &pb.WatchResponse{Header:       sws.newResponseHeader(sws.watchStream.Rev()),WatchId:      clientv3.InvalidWatchID,Canceled:     true,Created:      true,CancelReason: cancelReason,}select {case sws.ctrlStream <- wr:continuecase <-sws.closec:return nil}}filters := FiltersFromRequest(creq)wsrev := sws.watchStream.Rev()rev := creq.StartRevisionif rev == 0 {rev = wsrev + 1}id, err := sws.watchStream.Watch(mvcc.WatchID(creq.WatchId), creq.Key, creq.RangeEnd, rev, filters...)if err == nil {sws.mu.Lock()if creq.ProgressNotify {sws.progress[id] = true}if creq.PrevKv {sws.prevKV[id] = true}if creq.Fragment {sws.fragment[id] = true}sws.mu.Unlock()} else {id = clientv3.InvalidWatchID}wr := &pb.WatchResponse{Header:   sws.newResponseHeader(wsrev),WatchId:  int64(id),Created:  true,Canceled: err != nil,}if err != nil {wr.CancelReason = err.Error()}select {case sws.ctrlStream <- wr:case <-sws.closec:return nil}case *pb.WatchRequest_CancelRequest:if uv.CancelRequest != nil {id := uv.CancelRequest.WatchIderr := sws.watchStream.Cancel(mvcc.WatchID(id))if err == nil {sws.ctrlStream <- &pb.WatchResponse{Header:   sws.newResponseHeader(sws.watchStream.Rev()),WatchId:  id,Canceled: true,}sws.mu.Lock()delete(sws.progress, mvcc.WatchID(id))delete(sws.prevKV, mvcc.WatchID(id))delete(sws.fragment, mvcc.WatchID(id))sws.mu.Unlock()}}case *pb.WatchRequest_ProgressRequest:if uv.ProgressRequest != nil {sws.mu.Lock()sws.watchStream.RequestProgressAll()sws.mu.Unlock()}default:// we probably should not shutdown the entire stream when// receive an invalid command.// so just do nothing instead.sws.lg.Sugar().Infof("invalid watch request type %T received in gRPC stream", uv)continue}}
}

这里主要处理客户端发送的 watch 请求,然后发送给 ctrlStream。sendLoop 会从 ctrlStream 中获取 event 然后发送给客户端。

WatchStream

这个 inferface 才是处理 watch 的主要逻辑

// WatchStream 是一个接口,定义了一个流式处理watch请求的机制
type WatchStream interface {// Watch 创建一个观察者。观察者会监听在给定的键或范围 [key, end) 上发生的事件或已发生的事件。//// 整个事件历史都可以被观察到,除非被压缩。// 如果 "startRev" <= 0,watch 将观察在当前修订版本之后的事件。//// 返回的 "id" 是这个观察者的ID。它作为 WatchID 出现在通过 stream 通道发送到创建的观察者的事件中。// 当 WatchID 不等于 AutoWatchID 时,使用指定的 WatchID,否则返回自动生成的 WatchID。Watch(id WatchID, key, end []byte, startRev int64, fcs ...FilterFunc) (WatchID, error)// Chan 返回一个通道。所有的watch响应将被发送到这个返回的通道。Chan() <-chan WatchResponse// RequestProgress 请求给定ID的观察者的进度。响应只有在观察者当前同步时才会被发送。// 响应将通过与此流关联的 WatchResponse 通道发送,以确保正确的顺序。// 响应不包含事件。响应中的修订版本是观察者自同步以来的进度。RequestProgress(id WatchID)// RequestProgressAll 请求所有共享此流的观察者的进度通知。// 如果所有观察者都已同步,将向此流的任意观察者发送带有watch ID -1的进度通知,并返回 true。RequestProgressAll() bool// Cancel 通过给定ID取消观察者。如果观察者不存在,将返回错误。Cancel(id WatchID) error// Close 关闭通道并释放所有相关资源。Close()// Rev 返回流上观察到的KV的当前修订版本。Rev() int64
}// WatchResponse 表示一个watch操作的响应。
type WatchResponse struct {// WatchID 是发送此响应的观察者的ID。WatchID WatchID// Events 包含所有需要发送的事件。Events []mvccpb.Event// Revision 是创建watch响应时KV的修订版本。// 对于正常响应,修订版本应该与Events中最后一个修改的修订版本相同。// 对于延迟响应的未同步观察者,修订版本大于Events中最后一个修改的修订版本。Revision int64// CompactRevision 在观察者由于压缩而被取消时设置。CompactRevision int64
}// 实现了 WatchStream
// watchStream 包含共享一个流通道发送被观察事件和其他控制事件的观察者集合。
type watchStream struct {// 可观察对象(例如KV存储)watchable watchable// 用于发送watch响应的通道ch        chan WatchResponse// 互斥锁,保护以下字段mu sync.Mutex // nextID 是为此流中下一个新观察者预分配的IDnextID   WatchID// 标志流是否已关闭closed   bool// 取消函数的映射,用于取消特定的观察者cancels  map[WatchID]cancelFunc// 观察者的映射,根据观察者ID索引watchers map[WatchID]*watcher
}// Watch 在流中创建一个新的观察者并返回其 WatchID。
func (ws *watchStream) Watch(id WatchID, key, end []byte, startRev int64, fcs ...FilterFunc) (WatchID, error) {// 防止键 >= 结束键(按字典顺序)的错误范围// 带有 'WithFromKey' 的watch请求具有空字节范围结束if len(end) != 0 && bytes.Compare(key, end) != -1 {return -1, ErrEmptyWatcherRange}// 获取互斥锁ws.mu.Lock()defer ws.mu.Unlock()// 如果流已关闭,返回错误if ws.closed {return -1, ErrEmptyWatcherRange}// 自动生成 WatchIDif id == clientv3.AutoWatchID {for ws.watchers[ws.nextID] != nil {ws.nextID++}id = ws.nextIDws.nextID++} else if _, ok := ws.watchers[id]; ok {return -1, ErrWatcherDuplicateID}// 创建新的观察者w, c := ws.watchable.watch(key, end, startRev, id, ws.ch, fcs...)// 保存取消函数和观察者ws.cancels[id] = cws.watchers[id] = wreturn id, nil
}
// Chan 返回用于接收watch响应的通道。
func (ws *watchStream) Chan() <-chan WatchResponse {return ws.ch
}
// Cancel 取消具有给定ID的观察者。
func (ws *watchStream) Cancel(id WatchID) error {// 获取互斥锁ws.mu.Lock()cancel, ok := ws.cancels[id]w := ws.watchers[id]ok = ok && !ws.closedws.mu.Unlock()// 如果观察者不存在或流已关闭,返回错误if !ok {return ErrWatcherNotExist}cancel()// 获取互斥锁ws.mu.Lock()// 在取消之前不删除观察者,以确保 Close() 调用时等待取消if ww := ws.watchers[id]; ww == w {delete(ws.cancels, id)delete(ws.watchers, id)}ws.mu.Unlock()return nil
}
// Close 关闭通道并释放所有相关资源。
func (ws *watchStream) Close() {// 获取互斥锁ws.mu.Lock()defer ws.mu.Unlock()// 取消所有观察者for _, cancel := range ws.cancels {cancel()}// 标记流已关闭并关闭通道ws.closed = trueclose(ws.ch)watchStreamGauge.Dec()
}
// Rev 返回流上观察到的KV的当前修订版本。
func (ws *watchStream) Rev() int64 {// 获取互斥锁ws.mu.Lock()defer ws.mu.Unlock()return ws.watchable.rev()
}
// RequestProgress 请求给定ID的观察者的进度。
func (ws *watchStream) RequestProgress(id WatchID) {// 获取互斥锁ws.mu.Lock()w, ok := ws.watchers[id]ws.mu.Unlock()// 如果观察者不存在,直接返回if !ok {return}// 请求进度ws.watchable.progress(w)
}
// RequestProgressAll 请求所有观察者的进度通知。
func (ws *watchStream) RequestProgressAll() bool {// 获取互斥锁ws.mu.Lock()defer ws.mu.Unlock()return ws.watchable.progressAll(ws.watchers)
}
  1. Watch 方法:创建一个新的观察者,如果指定的范围不正确或观察者ID重复,则返回错误。否则,创建观察者并保存取消函数和观察者实例。
  2. Chan 方法:返回用于接收watch响应的通道。
  3. Cancel 方法:取消给定ID的观察者,删除相关的取消函数和观察者实例。
  4. Close 方法:关闭所有观察者并释放资源。
  5. Rev 方法:返回当前观察到的KV修订版本。
  6. RequestProgress 方法:请求特定观察者的进度。
  7. RequestProgressAll 方法:请求所有观察者的进度通知。

可以可到 当调用 Watch 的时候 每个 watchId 都会调用 watchable.watch 并把自己 ch 放入进去

watchable

// watchable 接口定义了可观察对象的行为
type watchable interface {// watch 创建一个新的观察者,用于监听指定键或范围[startRev, end)上的事件。// 返回观察者指针和取消函数。watch(key, end []byte, startRev int64, id WatchID, ch chan<- WatchResponse, fcs ...FilterFunc) (*watcher, cancelFunc)// progress 通知特定观察者当前的进度。progress(w *watcher)// progressAll 通知所有观察者当前的进度。// 如果所有观察者都已同步,则返回 true。progressAll(watchers map[WatchID]*watcher) bool// rev 返回当前观察到的修订版本。rev() int64
}// watchableStore 是一个实现了 watchable 接口的结构体,代表一个可观察的存储
type watchableStore struct {// store 是一个指向基础存储的指针*store// mu 保护观察者组和批次。为了避免死锁,在锁定 store.mu 之前不应锁定 mu。mu sync.RWMutex// victims 是在 watch 通道上被阻塞的观察者批次victims []watcherBatchvictimc chan struct{}// unsynced 包含所有需要同步已经发生的事件的未同步观察者unsynced watcherGroup// synced 包含所有与存储进度同步的观察者// 映射的键是观察者监听的键synced watcherGroup// stopc 是一个用于停止操作的通道stopc chan struct{}// wg 用于等待所有 goroutine 完成wg sync.WaitGroup
}func (s *watchableStore) watch(key, end []byte, startRev int64, id WatchID, ch chan<- WatchResponse, fcs ...FilterFunc) (*watcher, cancelFunc) {// 创建一个新的观察者wa := &watcher{key:    key,end:    end,minRev: startRev,id:     id,ch:     ch,fcs:    fcs,}// 锁定 watchableStore 的互斥锁s.mu.Lock()// 锁定 store 的读写锁用于获取当前修订版本s.revMu.RLock()// 判断观察者是否与当前存储修订版本同步synced := startRev > s.store.currentRev || startRev == 0if synced {// 如果同步,设置最小修订版本为当前修订版本的下一个版本wa.minRev = s.store.currentRev + 1if startRev > wa.minRev {wa.minRev = startRev}// 将观察者添加到同步观察者组中s.synced.add(wa)} else {// 如果未同步,增加慢速观察者计数器slowWatcherGauge.Inc()// 将观察者添加到未同步观察者组中s.unsynced.add(wa)}// 解锁 store 的读写锁s.revMu.RUnlock()// 解锁 watchableStore 的互斥锁s.mu.Unlock()// 增加观察者计数器watcherGauge.Inc()// 返回观察者和取消函数return wa, func() { s.cancelWatcher(wa) }
}

newWatchableStore

func newWatchableStore(lg *zap.Logger, b backend.Backend, le lease.Lessor, cfg StoreConfig) *watchableStore {if lg == nil {lg = zap.NewNop()}s := &watchableStore{store:    NewStore(lg, b, le, cfg),victimc:  make(chan struct{}, 1),unsynced: newWatcherGroup(),synced:   newWatcherGroup(),stopc:    make(chan struct{}),}s.store.ReadView = &readView{s}s.store.WriteView = &writeView{s}if s.le != nil {// use this store as the deleter so revokes trigger watch eventss.le.SetRangeDeleter(func() lease.TxnDelete { return s.Write(traceutil.TODO()) })}s.wg.Add(2)go s.syncWatchersLoop()go s.syncVictimsLoop()return s
}

syncWatchersLoop

// syncWatchersLoop 每100毫秒同步一次unsynced集合中的观察者。
func (s *watchableStore) syncWatchersLoop() {defer s.wg.Done()// 设置等待时间为100毫秒waitDuration := 100 * time.MilliseconddelayTicker := time.NewTicker(waitDuration)defer delayTicker.Stop()for {// 锁定以获取未同步观察者的数量s.mu.RLock()st := time.Now()lastUnsyncedWatchers := s.unsynced.size()s.mu.RUnlock()unsyncedWatchers := 0// 如果有未同步观察者,同步这些观察者if lastUnsyncedWatchers > 0 {unsyncedWatchers = s.syncWatchers()}syncDuration := time.Since(st)// 重置定时器delayTicker.Reset(waitDuration)// 检查是否有更多待处理的工作if unsyncedWatchers != 0 && lastUnsyncedWatchers > unsyncedWatchers {// 公平对待其他存储操作,通过延长时间来避免占用太多资源delayTicker.Reset(syncDuration)}// 等待定时器或停止信号select {case <-delayTicker.C:case <-s.stopc:return}}
}// syncWatchers 通过以下步骤同步未同步的观察者:
//  1. 从未同步观察者组中选择一组观察者
//  2. 迭代该组以获取最小修订版本并移除压缩的观察者
//  3. 使用最小修订版本获取所有键值对,并将这些事件发送给观察者
//  4. 从未同步组中移除已同步的观察者,并移动到同步组中
func (s *watchableStore) syncWatchers() int {// 锁定s.mu.Lock()defer s.mu.Unlock()// 如果没有未同步观察者,返回0if s.unsynced.size() == 0 {return 0}// 锁定存储的读写锁s.store.revMu.RLock()defer s.store.revMu.RUnlock()// 为了从未同步观察者中找到键值对,我们需要找到最小修订版本curRev := s.store.currentRevcompactionRev := s.store.compactMainRev// 选择一组观察者wg, minRev := s.unsynced.choose(maxWatchersPerSync, curRev, compactionRev)minBytes, maxBytes := NewRevBytes(), NewRevBytes()minBytes = RevToBytes(Revision{Main: minRev}, minBytes)maxBytes = RevToBytes(Revision{Main: curRev + 1}, maxBytes)// UnsafeRange 返回键和值。在boltdb中,键是修订版本,值是实际的键值对。tx := s.store.b.ReadTx()tx.RLock()revs, vs := tx.UnsafeRange(schema.Key, minBytes, maxBytes, 0)evs := kvsToEvents(s.store.lg, wg, revs, vs)// 必须在kvsToEvents之后解锁,因为vs(来自boltdb内存)不是深拷贝。// 我们只能在Unmarshal之后解锁,这将进行深拷贝。// 否则我们将在boltdb重新mmap期间触发SIGSEGV。tx.RUnlock()// 创建一个新的观察者批次victims := make(watcherBatch)wb := newWatcherBatch(wg, evs)for w := range wg.watchers {if w.minRev < compactionRev {// 跳过因压缩而无法发送响应的观察者continue}w.minRev = curRev + 1eb, ok := wb[w]if !ok {// 将未通知的观察者移至同步s.synced.add(w)s.unsynced.delete(w)continue}if eb.moreRev != 0 {w.minRev = eb.moreRev}// 发送响应if w.send(WatchResponse{WatchID: w.id, Events: eb.evs, Revision: curRev}) {pendingEventsGauge.Add(float64(len(eb.evs)))} else {w.victim = true}// 处理受害者观察者if w.victim {victims[w] = eb} else {if eb.moreRev != 0 {// 保持未同步状态;还有更多要读取continue}s.synced.add(w)}s.unsynced.delete(w)}s.addVictim(victims)// 更新慢速观察者计数器vsz := 0for _, v := range s.victims {vsz += len(v)}slowWatcherGauge.Set(float64(s.unsynced.size() + vsz))return s.unsynced.size()
}

watcher & send

type watcher struct {// the watcher keykey []byte// end indicates the end of the range to watch.// If end is set, the watcher is on a range.end []byte// victim is set when ch is blocked and undergoing victim processingvictim bool// compacted is set when the watcher is removed because of compactioncompacted bool// restore is true when the watcher is being restored from leader snapshot// which means that this watcher has just been moved from "synced" to "unsynced"// watcher group, possibly with a future revision when it was first added// to the synced watcher// "unsynced" watcher revision must always be <= current revision,// except when the watcher were to be moved from "synced" watcher grouprestore bool// minRev is the minimum revision update the watcher will acceptminRev int64id     WatchIDfcs []FilterFunc// a chan to send out the watch response.// The chan might be shared with other watchers.ch chan<- WatchResponse
}func (w *watcher) send(wr WatchResponse) bool {progressEvent := len(wr.Events) == 0if len(w.fcs) != 0 {ne := make([]mvccpb.Event, 0, len(wr.Events))for i := range wr.Events {filtered := falsefor _, filter := range w.fcs {if filter(wr.Events[i]) {filtered = truebreak}}if !filtered {ne = append(ne, wr.Events[i])}}wr.Events = ne}// if all events are filtered out, we should send nothing.if !progressEvent && len(wr.Events) == 0 {return true}select {case w.ch <- wr:return truedefault:return false}
}

syncVictimsLoop

// syncVictimsLoop 尝试将预先计算的观察者响应写入被阻塞的观察者通道
func (s *watchableStore) syncVictimsLoop() {defer s.wg.Done()for {// 尝试更新所有受害者观察者for s.moveVictims() != 0 {// 持续更新,直到所有受害者观察者都处理完毕}// 检查是否有受害者观察者s.mu.RLock()isEmpty := len(s.victims) == 0s.mu.RUnlock()var tickc <-chan time.Timeif !isEmpty {tickc = time.After(10 * time.Millisecond)}// 等待10毫秒或收到新的受害者通知或停止信号select {case <-tickc:case <-s.victimc:case <-s.stopc:return}}
}// moveVictims 尝试使用已存在的事件数据更新观察者
func (s *watchableStore) moveVictims() (moved int) {s.mu.Lock()victims := s.victimss.victims = nils.mu.Unlock()var newVictim watcherBatchfor _, wb := range victims {// 再次尝试发送响应for w, eb := range wb {// 观察者已观察到存储,直到但不包括 w.minRevrev := w.minRev - 1if w.send(WatchResponse{WatchID: w.id, Events: eb.evs, Revision: rev}) {pendingEventsGauge.Add(float64(len(eb.evs)))} else {if newVictim == nil {newVictim = make(watcherBatch)}newVictim[w] = ebcontinue}moved++}// 将完成的受害者观察者分配到未同步/同步组s.mu.Lock()s.store.revMu.RLock()curRev := s.store.currentRevfor w, eb := range wb {if newVictim != nil && newVictim[w] != nil {// 无法发送watch响应,仍然是受害者continue}w.victim = falseif eb.moreRev != 0 {w.minRev = eb.moreRev}if w.minRev <= curRev {s.unsynced.add(w)} else {slowWatcherGauge.Dec()s.synced.add(w)}}s.store.revMu.RUnlock()s.mu.Unlock()}// 如果仍然有未处理的受害者,重新添加到受害者列表中if len(newVictim) > 0 {s.mu.Lock()s.victims = append(s.victims, newVictim)s.mu.Unlock()}return moved
}

Reference

  • https://blog.csdn.net/qq_24433609/article/details/120653747

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/722616.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue TypeScript 实战:掌握静态类型编程

这篇文章介绍了如何在TypeScript环境下为Vue.js应用搭建项目结构,包括初始化配置、创建Vue组件、实现状态管理利用Vuex、配置路由以及性能优化的方法,旨在提升开发效率与应用性能。title: Vue TypeScript 实战:掌握静态类型编程 date: 2024/6/10 updated: 2024/6/10 excerpt…

INFINI Labs 产品更新 | Easysearch 1.8.2 发布优化 CCR 性能

INFINI Labs 产品又更新啦~,包括 Easysearch v1.8.0、Gateway、Console、Agent、Loadgen v1.25.0。本次各产品更新了很多亮点功能,如 Easysearch 新增数据写入限流功能,可实现节点、分片级限流;Gateway 修复数据迁移过程中因消费不及时解压缩导致部分数据记录损坏而丢失记录…

Nginx Rewrite

目录1.常用的Nginx 正则表达式2.location3.rewrite 1.常用的Nginx 正则表达式 ^ :匹配输入字符串的起始位置 $ :匹配输入字符串的结束位置 * :匹配前面的字符零次或多次。如“ol*”能匹配“o”及“ol”、“oll” + :匹配前面的字符一次或多次。如“ol+”能匹配“ol”及“ol…

3_@Autowired注解失效分析

1. Aware 接口 Aware 接口提供了一种[内置]的注入手段,可以注入BeanFactory, ApplicationContext。内置的注入和初始化不受扩展功能的影响,总会被执行,因此Spring 框架的内部类常使用它们。 InitializingBean 接口提供了一种[内置]的初始化手段。 Aware的作用就是注入与容器…

【内存管理】内存布局

ARM32位系统的内存布局图 32位操作系统的内存布局很经典,很多书籍都是以32位系统为例子去讲解的。32位的系统可访问的地址空间为4GB,用户空间为1GB ~ 3GB,内核空间为3GB ~ 4GB。为什么要划分为用户空间和内核空间呢? 一般处理器会把运行模式分为好几个,比如x86分为rang0 ~…

【esp32 项目】中断读取按键

原理图:图 按键部分图 单片机部分 程序:KEY_USR 引脚配置成上拉输入 在Arduino中,配置一个IO为上拉输入可以使用pinMode()函数和digitalWrite()函数。pinMode()函数用于设置引脚模式,而digitalWrite()函数用于设置上拉电阻。 以下是一个示例代码,展示如何将Arduino的数字引…

【操作系统】页表映射

页表的一些术语 现在Linux内核中支持四级页表的映射,我们先看下内核中关于页表的一些术语:全局目录项,PGD(Page Global Directory)上级目录项,PUD(Page Upper Directory)中间目录项,PMD(Page Middle Directory)页表项,(Page Table)大家在看内核代码时会经常看的以…

算法金 | AI 基石,无处不在的朴素贝叶斯算法

大侠幸会,在下全网同名「算法金」 0 基础转 AI 上岸,多个算法赛 Top 「日更万日,让更多人享受智能乐趣」历史上,许多杰出人才在他们有生之年默默无闻, 却在逝世后被人们广泛追忆和崇拜。 18世纪的数学家托马斯贝叶斯(Thomas Bayes)便是这样一位人物贝叶斯的研究,初看似…

OO第五、六次大作业总结

一、前言在经过了第一阶段的迭代大作业之后,我们迎来了新的迭代大作业-智能电器模拟系统。 在这个阶段的大作业中,我们需要编写模拟家用电器与电路的程序。 二、设计与分析在经过了第一次迭代大作业的折磨之后,我总结之前出现的问题,在之前的大作业中,我在类的设计方面做的…

【转载】GDB高级技巧:边Debug边修复BUG,无需修改代码,无需重新编译

调试是每个程序员都逃不过的宿命! 程序调试是一件非常考验耐心的事情,因为调试过程中经常会需要反复的修改源码,重新编译、重新部署、重新运行,这个过程通常是非常枯燥和繁琐的。尤其对于大型项目,光是编译可能需要几十分钟,甚至几个小时,部署过程则可能更为复杂漫长! …

【转载】C 语言有什么奇技淫巧

快速范围判断 经常要批量判断某些值在不在范围内,如果 int 检测是 [0, N) 的话: if (x >= 0 && x < N) ...众所周知,现代 CPU 优化,减分支是重要手段,上述两次判断可以简写为: if (((unsigned int)x) < N) ...减少判断次数。如果 int 检测范围是 [minx,…

出现java.lang.NoSuchMethodError: xxx.xxx.xxx.xxxDto.getxxx()Ljava/lang/String

参考博客:https://blog.csdn.net/weixin_43530696/article/details/115732807 修改字段后JPA自动生成的代码未及时更新。删除JPA相关jar包后,重启项目。 直接rebuild项目 重启 ,如果没有用的话,先invalid Chches清缓存之后,再重启动项目一般即可解决您的资助是我最大的动力…