当前位置:网站首页>ETCD数据库源码分析——处理Entry记录简要流程
ETCD数据库源码分析——处理Entry记录简要流程
2022-07-04 22:33:00 【肥叔菌】
(1)当客户端向etcd集群发送了一次请求之后,请求中的封装Entry记录会先被提交给etcd-raft模块进行处理,其中,etcd-raft模块会先将Entry记录保存到raftLog.unstable中。
我们以ETCD数据库源码分析——服务端PUT流程为例,kvServer结构体Put函数处理流程如下:首先会对请求消息进行各方面的检查,检查完之后会将所有的请求交给其内封装的RaftKV接口进行处理,待处理完成得到响应消息之后,会通过header.fill()方法填充响应的头信息,最后将完整的响应消息返回给客户端。因此,往下追踪 srv.(KVServer).Put(ctx, in)其实就是调用(s *EtcdServer) Put()函数。
从raftRequestOnce流程看出最终调用的是 (s *EtcdServer) processInternalRaftRequestOnce(…)函数,在该函数里面有一句关键调用 s.r.Propose(cctx, data)。s是EtcdServer, r是其里面的成员变量raftNode, 这就是进入raft协议相关的节奏了。通过对Propose函数的追踪,我们可以看到最终函数走到了stepWithWaitOption
// raft/node.go
func (n *node) Propose(ctx context.Context, data []byte) error {
return n.stepWait(ctx, pb.Message{
Type: pb.MsgProp, Entries: []pb.Entry{
{
Data: data}}}) } // 这里给Message加上了Type为pb.MsgProp,Entry为初入的PutRequest
func (n *node) stepWait(ctx context.Context, m pb.Message) error {
return n.stepWithWaitOption(ctx, m, true) }
// Step advances the state machine using msgs. The ctx.Err() will be returned, if any.
func (n *node) stepWithWaitOption(ctx context.Context, m pb.Message, wait bool) error {
if m.Type != pb.MsgProp {
// 这里不会走
select {
case n.recvc <- m: return nil
case <-ctx.Done(): return ctx.Err()
case <-n.done: return ErrStopped
}
}
ch := n.propc // 取出node结构体提供的propc通道
pm := msgWithResult{
m: m}
if wait {
pm.result = make(chan error, 1) }
select {
case ch <- pm: if !wait {
return nil } // 将带有msg的pm结构体放入propc通道
case <-ctx.Done(): return ctx.Err()
case <-n.done: return ErrStopped
}
select {
case err := <-pm.result:
if err != nil {
return err }
case <-ctx.Done(): return ctx.Err()
case <-n.done: return ErrStopped
}
return nil
}
从raft/node.go的run函数可以看出,当从propc通道中取出msgWithResult,在从其中取出消息msg,设置msg的来源节点为本节点id。然后调用raft模块的Step函数驱动状态机。
(2)etcd-raft模块将该Entry记录封装到前面介绍的Ready实例中,返回给上层模块进行持久化。
(3)当上层模块收到待持久化的Entry记录之后,会先将其记录到WAL日志文件中,然后进行持久化操作,最后通知etcd-raft模块进行处理。
(4)此时etcd-raft模块就会将该Entry记录从unstable移动到storage中保存。
(5)待该Entry记录被复制到集群中的半数以上节点时,该Entry记录会被Leader节点确认为已提交(committed),并封装进Ready实例返回给上层模块。
(6)此时上层模块即可将该Ready实例中携带的待应用Entry记录应用到状态机中。
边栏推荐
- MYSQL架构——用户权限与管理
- [ODX studio edit PDX] - 0.2-how to compare two pdx/odx files of compare
- MYSQL架构——逻辑架构
- 【剑指Offer】6-10题
- Li Kou 98: verify binary search tree
- A complete tutorial for getting started with redis: Pipeline
- 特征缩放 标准化 归一化
- Serial port data frame
- Redis入门完整教程:客户端通信协议
- Attack and defense world misc advanced zone 2017_ Dating_ in_ Singapore
猜你喜欢
Google Earth engine (GEE) - tasks upgrade enables run all to download all images in task types with one click
安装人大金仓数据库
A complete tutorial for getting started with redis: getting to know redis for the first time
[graph theory] topological sorting
Redis入门完整教程:HyperLogLog
共创软硬件协同生态:Graphcore IPU与百度飞桨的“联合提交”亮相MLPerf
Unity修仙手游 | lua动态滑动功能(3种源码具体实现)
Erik baleog and Olaf, advanced area of misc in the attack and defense world
Analysis of the self increasing and self decreasing of C language function parameters
【机器学习】手写数字识别
随机推荐
攻防世界 MISC 进阶区 hong
NFT insider 64: e-commerce giant eBay submitted an NFT related trademark application, and KPMG will invest $30million in Web3 and metauniverse
Redis入门完整教程:集合详解
通过Go语言创建CA与签发证书
Set up a website with a sense of ceremony, and post it to 1/2 of the public network through the intranet
P2181 对角线和P1030 [NOIP2001 普及组] 求先序排列
Summary of wechat applet display style knowledge points
Redis入门完整教程:发布订阅
华泰证券是国家认可的券商吗?开户安不安全?
Redis: redis configuration file related configuration and redis persistence
Google Earth engine (GEE) - tasks upgrade enables run all to download all images in task types with one click
Attack and defense world misc advanced area can_ has_ stdio?
Lost in the lock world of MySQL
攻防世界 MISC 进阶区 3-11
浅聊一下中间件
页面关闭前,如何发送一个可靠请求
【机器学习】手写数字识别
PMO: compare the sample efficiency of 25 molecular optimization methods
攻防世界 MISC 进阶区 can_has_stdio?
Advanced area a of attack and defense world misc Masters_ good_ idea