Bootstrap

【k8s源码篇之Informer篇1】理解 Informer 的缓存与索引数据结构的设计

参考

架构

Informer 和 Controller

  • 便于理解的架构图
  • 这里 Indexer「索引」 和 Local Store 「缓存」 是分开表示的
    • 在源码级别,基本上是一起实现的,一个结构体内涵盖

在这里插入图片描述

Informer 简要架构

  • 源码级简要理解

(三)Kubernetes 源码剖析之学习Informer机制_golang_03

Informer 详细架构

  • 源码级详细理解

img

代码解析

SharedInformer

// staging/src/k8s.io/client-go/tools/cache/shared_informer.go

type SharedIndexInformer interface {
   SharedInformer
   // AddIndexers add indexers to the informer before it starts.
   AddIndexers(indexers Indexers) error
   GetIndexer() Indexer
}
  
type sharedIndexInformer struct {
   // 索引和 缓存 store
   indexer    Indexer
   controller Controller
   // 处理函数,将是重点
   processor *sharedProcessor
   // 检测 cache 是否有变化,一把用作调试,默认是关闭的
   cacheMutationDetector MutationDetector
   // 构造 Reflector 需要
   listerWatcher ListerWatcher
   // 目标类型,给 Reflector 判断资源类型
   objectType runtime.Object
   // Reflector 进行重新同步周期
   resyncCheckPeriod time.Duration
   // 如果使用者没有添加 Resync 时间,则使用这个默认的重新同步周期
   defaultEventHandlerResyncPeriod time.Duration
   clock                           clock.Clock
   // 两个 bool 表达了三个状态:controller 启动前、已启动、已停止
   started, stopped bool
   startedLock      sync.Mutex
   // 当 Pop 正在消费队列,此时新增的 listener 需要加锁,防止消费混乱
   blockDeltas sync.Mutex
   // Watch 返回 err 的回调函数
   watchErrorHandler WatchErrorHandler
}
 
 
type sharedProcessor struct {
   listenersStarted bool
   listenersLock    sync.RWMutex
   listeners        []*processorListener
   syncingListeners []*processorListener // 需要 sync 的 listeners
   clock            clock.Clock
   wg               wait.Group
}

1. Indexer 结构讲解 —— 实现缓存和索引

img

Indexer 接口 —— 缓存和索引的高级抽象

可以看出 Indexer 封装了 Store 接口,那么就可以实现两部分行为

  1. 通过 Store 接口,实现对缓存(也就是图中 Local Store)的操作
  2. 通过本身的其他方法,构建便于查找数据的索引结构
// k8s.io/client-go/tools/cache/index.go

// Indexer 封装了 Store 的接口(相当于可以对缓存进行操作)
// 同时 Indexer 本身定义了多种方法,对索引进行操作(构建索引等)
// Store 的作用是:对实际的数据操作,存储、更新等,构建简单的 key —— object 之间的映射
// Indexer 的作用是:构建便于搜索的结构,如 获取某个 namespace 下所有 pod,那可能构建这样索引结构, namespace-key —— (在此namespace 下的  pod1 key,pod2 key ...)
// 通过  pod key 可以在 Store 获取对应的 存储对象(就是 Pod 的具体信息)
type Indexer interface {
  // 封装了 缓存 Local Store
	Store
  // 下面方法 为对 索引结构的操作
	// Index returns the stored objects whose set of indexed values
	// intersects the set of indexed values of the given object, for
	// the named index
	Index(indexName string, obj interface{}) ([]interface{}, error)
	// IndexKeys returns the storage keys of the stored objects whose
	// set of indexed values for the named index includes the given
	// indexed value
	IndexKeys(indexName, indexedValue string) ([]string, error)
	// ListIndexFuncValues returns all the indexed values of the given index
	ListIndexFuncValues(indexName string) []string
	// ByIndex returns the stored objects whose set of indexed values
	// for the named index includes the given indexed value
	ByIndex(indexName, indexedValue string) ([]interface{}, error)
	// GetIndexer return the indexers
	GetIndexers() Indexers

	// AddIndexers adds more indexers to this store.  If you call this after you already have data
	// in the store, the results are undefined.
	AddIndexers(newIndexers Indexers) error
}
缓存 Local Store 的抽象定义 —— Store 接口

可以看出 Store 接口主要定义 对缓存(Local Store)的操作

// k8s.io/client-go/tools/cache/store.go

// Store 接口定义了 对缓存的操作 (如删除、更新、获取等)
type Store interface {
	Add(obj interface{}) error
	Update(obj interface{}) error
	Delete(obj interface{}) error
	List() []interface{}
	ListKeys() []string
	Get(obj interface{}) (item interface{}, exists bool, err error)
	GetByKey(key string) (item interface{}, exists bool, err error)

	// Replace will delete the contents of the store, using instead the
	// given list. Store takes ownership of the list, you should not reference
	// it after calling this function.
	Replace([]interface{}, string) error
	Resync() error
}

Indexer 的实现 —— cache 结构体

查看代码可以发现,cache 结构体实现了 Indexer 中的所有方法(包含Store部分的方法)

因此可以说 cache 就是 Indexer 接口的实现

  • 其中 ThreadSafeStore 接口中的方法涵盖了 Indexer 接口的所有方法,因此若实现了 ThreadSafeStore 接口 就相当于实现了 Indexer 接口
  • KeyFunc 定义函数,用于计算 Object 的 key
// Store 的实际实现  cache
// k8s.io/client-go/tools/cache/store.go

// cache responsibilities are limited to:
//	1. Computing keys for objects via keyFunc
//  2. Invoking methods of a ThreadSafeStorage interface
type cache struct {
	// cacheStorage bears the burden of thread safety for the cache
  // 这是个接口其实 就涵盖了 Indexer 接口 的所有方法
  // 因此实现了 ThreadSafeStore 该接口,就相当于实现了 Indexer 接口
	cacheStorage ThreadSafeStore
	// keyFunc is used to make the key for objects stored in and retrieved from items, and
	// should be deterministic.
  // 用于计算 Object 的 key
	keyFunc KeyFunc
}
ThreadSafeStore 接口 —— 实现 Indexer 接口的准备

可以数一数

  • Indexer 接口(包含Store接口)一共定义了 15 个方法

  • ThreadSafeStore 也定义了 15 个方法

  • 同时他们的函数名称及函数定义一致

因此 实现了 ThreadSafeStore 接口 等同于 实现了 Indexer 接口

// k8s.io/client-go/tools/cache/thread_safe_store.go

// ThreadSafeStore 接口
// 表示可以对缓存的操作
type ThreadSafeStore interface {
	Add(key string, obj interface{})
	Update(key string, obj interface{})
	Delete(key string)
	Get(key string) (item interface{}, exists bool)
	List() []interface{}
	ListKeys() []string
	Replace(map[string]interface{}, string)
	Index(indexName string, obj interface{}) ([]interface{}, error)
	IndexKeys(indexName, indexKey string) ([]string, error)
	ListIndexFuncValues(name string) []string
	ByIndex(indexName, indexKey string) ([]interface{}, error)
	GetIndexers() Indexers

	// AddIndexers adds more indexers to this store.  If you call this after you already have data
	// in the store, the results are undefined.
	AddIndexers(newIndexers Indexers) error
	Resync() error
}
ThreadSafeStore 接口的实现 —— threadSafeMap 结构体

threadSafeMap 结构体 可以称之为 缓存(Local Store)的实际数据存储结构(通过map + Lock 进行存储)

  • 数据的实际存储 —— items map[string]interface{}
  • 索引的构建 —— indexers Indexers
  • 索引的存储 —— indices Indices
// k8s.io/client-go/tools/cache/thread_safe_store.go

// threadSafeMap implements ThreadSafeStore
type threadSafeMap struct {
  // 相当于缓存的本质,存储着 key-object
	lock  sync.RWMutex
	items map[string]interface{}

  // Indexers  Indices 也是map 结构
  // 相当于 索引的 结构、下面详细介绍
	// indexers maps a name to an IndexFunc
	indexers Indexers
	// indices maps a name to an Index
	indices Indices
}
真正索引的结构 —— Indexers、Indices、Index

// map 索引类型 => 索引函数
type Indexers map[string]IndexFunc

Indexers 可以理解为:key —— 特征名称(如 namespace 特征),value —— 特征提取函数的名称(该函数可提取 object 的 namespace 名称)

// map 索引类型 => 索引值 map
type Indices map[string]Index

Indices 可以理解为: key —— 特征名称(如 namespace 特征),value —— 不同特征值所对应的 object 列表(如存在 2 个 ns,每个 ns 中有多个 pod,不同的 ns 会计算出不同的 特征值,然后指向的 object 列表为其对应 ns 下的 所有 pod 名称)

// 索引值 map: 由索引函数计算所得索引值(indexedValue) => [objKey1, objKey2…]
type Index map[string]sets.String

Index 可以理解为: key —— 特征值(每个 ns 的特征值都会不一致),value —— 特征值相同的所有 object(同一个ns 下的所有 pod 的特征值ns相同,因此该 object 列表记录着该 ns 下的所有 pod 名称)

// k8s.io/client-go/tools/cache/index.go

// Index maps the indexed value to a set of keys in the store that match on that value
// 索引值 map: 由索引函数计算所得索引值(indexedValue) => [objKey1, objKey2...]
type Index map[string]sets.String

// Indexers maps a name to a IndexFunc
// map 索引类型 => 索引函数
type Indexers map[string]IndexFunc

// Indices maps a name to an Index
// map 索引类型 => 索引值 map
type Indices map[string]Index

// k8s.io/apimachinery/pkg/util/sets/string.go
// sets.String 可以理解为 没有重复元素的数组,不过在 go 语言里面是采用 map 形式构建,因为 map 的 key 不能重复 
// sets.String is a set of strings, implemented via map[string]struct{} for minimal memory consumption.
type String map[string]Empty

为了便于理解我举个例子

// map 索引类型 => 索引函数
type Indexers map[string]IndexFunc

// --------- 场景
// 目前 Indexers 中有这样的数据{"namespace":NsHashFunc,"labels":LabelsHashFunc}
// 同时此时有 3 个 pod
// default namespace 下有俩个 pod : pod1  pod2
// test namespace 下有一个 pod: pod3
// 且这三个 pod 的 label 都不相同

// --------- 构建索引
// namespace 接下来将构建 Indices 和 Index
// 1. 计算所有 pod 的索引值,将 pod 逐个通过 Indexers 中的所有函数进行计算
//    pod1 —— NsHashFunc 计算结果为 100,LabelsHashFunc 计算结果为 2000
//    pod2 —— NsHashFunc 计算结果为 100,LabelsHashFunc 计算结果为 3000
//    pod3 —— NsHashFunc 计算结果为 200,LabelsHashFunc 计算结果为 4000
// 2. 构建索引 Indices map[string]Index
//    由 Indexers 有两种类型  namespace 和 labels,因此逐个构建 {"namespace":Index, "labels": Index}
//    Index map[string]sets.String 的 key 为索引值,值为 objectKey( 就是通过 cache 结构体中 KeyFunc 计算得到的,namspace/name 的组合)
//    遍历 namespace 类型的索引值,构建了如下 Index 结构: 
{
"100":{"default/pod1","default/pod2"},
"200":{"default/pod3"}
}
//    遍历 lables 类型的索引值,构建了如下 Index 结构: 
{
"2000":{"default/pod1"},
"3000":{"default/pod2"},
"4000":{"default/pod3"}
}
//    所以最后成的 Indices 为
{
"nameapces":{
"100":{"default/pod1","default/pod2"},
"200":{"default/pod3"}
},
"labels":{
"2000":{"default/pod1"},
"3000":{"default/pod2"},
"4000":{"default/pod3"}
}
}

悦读

道可道,非常道;名可名,非常名。 无名,天地之始,有名,万物之母。 故常无欲,以观其妙,常有欲,以观其徼。 此两者,同出而异名,同谓之玄,玄之又玄,众妙之门。

;