Caddy源碼閱讀(二)啓動流程與 Event 事件通知

Caddy源碼閱讀(二)啓動流程與 Event 事件通知

Preface

Caddy 是 Go 語言構建的輕量配置化服務器。https://github.com/caddyserver/caddygit

Caddy 整個軟件能夠說是由不一樣的 插件 堆砌起來的。本身自己僅提供 Plugin 的註冊運行邏輯和 Server 的監聽服務功能。github

學習 caddy 的源碼,其實是學習 如何構建一個 鬆耦合的 抽象 Plugin 設計,即模塊化插拔的作法。安全

因此咱們的源碼閱讀,圍繞 Caddy 爲 Plugin 提供的基礎設施,和 Plugin 自身邏輯。服務器

caddy-overview (2).png
下面咱們從第一步,啓動流程開始閱讀。
以後的路徑應該是  Caddyfile 的解析,解析出的 配置文件如何消費,配置完成的服務器如何服務。併發

Overview

Package

這是 caddy 包的結構
image.pngimage.pngsvg

main.go

一切的開始  --- 
咱們查看 在 caddy 文件夾下的 main.go 函數。模塊化

這是 上圖 caddy 文件夾下的目錄結構。

其中 run.go 咱們在上一篇文章閱讀完成函數

main.go 的 Trick

在 caddy 文件夾中的 main 函數啓動 caddy 服務器。實際運行的是 run.go 中的文件,這是方便測試使用
main.go的代碼學習

image.png

經過改變 run 變量的值來方便測試,能夠學習一下。測試

啓動流程 

啓動 caddy 的流程

屏幕快照 2019-08-04 下午6.34.21.png

 caddyfileLoader 加載 caddyfile 配置  =》生成 Input 信息
Context =》 生成 Server

caddyfile 示例

caddyfile 簡單示例:
image.png

Instance 是運行操做的 Server 實例,能夠看到幾個主要的操做都是在他身上 

Server 兩種監聽模式 TCP UDP 

咱們首先關心的是 Start() 啓動服務器。

啓動服務器

發送 StartupEvent, 參照下文中 Event 理解

// Executes Startup events
caddy.EmitEvent(caddy.StartupEvent, nil)

讀取配置文件:參照個人接下來的文章 Caddy-解析Caddyfile

caddyfileinput, err := caddy.LoadCaddyfile(serverType)

啓動:

instance, err := caddy.Start(caddyfileinput)

發送 InstanceStartupEvent

caddy.EmitEvent(caddy.InstanceStartupEvent, instance)

Start()

// Start your engines
instance, err := caddy.Start(caddyfileinput)
if err != nil {
    mustLogFatalf("%v", err)
}

閱讀完代碼,畫一張圖幫助理解
屏幕快照 2019-08-04 下午6.35.03.png

這裏除了 Instance 以外還有兩個新名詞

 Controller:它是用來幫助 Directives 設置它自身的,經過讀取 Token,這裏的 Directives 實際上對應的就是上文所說的 caddyfile 中的配置文件選項。

這一點請參照 Caddy(三)Loader 下的 excuteDirective 理解。

 Token :是 caddy 本身的 詞法分析器 解析 caddyfile 配置文件出的選項的標記。

這一點請Caddy(三)中 Loader 中的 Parser 理解

caddy.Start.svg

咱們來看順序,第一遍從頂向下看。

第一個是 Input,這是 caddyfile 的變量結構,他能夠經過 Start()方法新建實例 Instance

Instance 經過從 caddyfile 讀取到信息的 Input 生成 Context

攜帶信息的 Context 承擔 新建 Server 的任務

Context 讀取 caddyfile 解析出的 ServerBlock 配置服務器

ServerBlock 包含 不一樣的 Tokens 他們會轉換爲 Directive

Directive 會被 Controller 消費,用於配置插件 安裝到服務器上

值得注意的是 Controller  更改的是 Instance 
對於 http 服務器來講還會增長 http 服務的中間件

若是不理解,首先記住 caddy 是 配置的 模塊化的服務器,

經過 caddyfile 配置 -> caddyfile
讀取它 -> Loader
解析配置目標-> token & directives
進行配置 -> controller & setup
啓動 -> instance & Start()

記住這個流程就能理解了。

Event 事件通知啓動插件

caddy-event.svg

引入

咱們看到,在 caddy 的 run.go 中有一行代碼是

caddy.EmitEvent(caddy.StartupEvent, nil)

這就是 caddy 中的 事件通知系統,通知的是全部的 plugin。

變量

caddy/plugin.go 包中

// eventHooks is a map of hook name to Hook. All hooks plugins
// must have a name.
eventHooks = &sync.Map{}

是一個保存全部 plugin hook 的 sync.Map{} 

這個標準包的 Map 是併發安全的, 一般咱們使用 Load() 或者 LoadOrStore() 方法存讀信息,Range() 方法遍歷,若是你須要,能夠引入你的 Go 程序中。

Logic

看內在實現

// EmitEvent executes the different hooks passing the EventType as an
// argument. This is a blocking function. Hook developers should
// use 'go' keyword if they don't want to block Caddy.
func EmitEvent(event EventName, info interface{}) {
    eventHooks.Range(func(k, v interface{}) bool {
        err := v.(EventHook)(event, info)
        if err != nil {
            log.Printf("error on '%s' hook: %v", k.(string), err)
        }
        return true
    })
}

很簡單,上文提過,eventHooks.Range 是遍歷信息,會遍歷全部保存的 EventHook 函數並運行。

那麼 Plugin 想使用接收某一個事件通知作相應操做的時候,只需把本身的 EventHook 函數註冊到這個 map 中

// eventHooks is a map of hook name to Hook. All hooks plugins
// must have a name.
eventHooks = &sync.Map{}

使用 RegisterEventHook 註冊
 type EventHook func(eventType EventName, eventInfo interface{}) error

// RegisterEventHook plugs in hook. All the hooks should register themselves
// and they must have a name.
func RegisterEventHook(name string, hook EventHook) {
    if name == "" {
        panic("event hook must have a name")
    }
    _, dup := eventHooks.LoadOrStore(name, hook)
    if dup {
        panic("hook named " + name + " already registered")
    }
}

那麼能夠監聽哪些事件呢?在 Plugin 中有定義常量

// Define names for the various events
const (
    StartupEvent         EventName = "startup"
    ShutdownEvent                  = "shutdown"
    CertRenewEvent                 = "certrenew"
    InstanceStartupEvent           = "instancestartup"
    InstanceRestartEvent           = "instancerestart"
)

啓動,關閉,刷新證書,這裏提到的 Instance 是 caddy 中的 Server 實例

結語

咱們概覽了 caddy 的 run 和 Start 啓動流程,接下來咱們會繼續深刻了解 Caddy 每一個部分流程。
能夠看以前的文章
Caddy源碼閱讀(一)Run詳解
和以後的
Caddy源碼閱讀(三)Caddyfile 解析 by Loader & Parser
Caddy源碼閱讀(四)Plugin & Controller 安裝插件
Caddy源碼閱讀(五) Instance & Server
caddy-plugins(一)自定義插件
caddy-plugins(二)caddy-grpc 一個插件實例

相關文章
相關標籤/搜索