您好,登錄后才能下訂單哦!
Kubernetes 中日志重要性
通常日志最基礎的作用是記錄程序的運行軌跡,在此之上會衍生出非常多的功能,例如線上監控、告警、運營分析、安全分析等等(詳情可以參見第一篇文章《6 個 K8s 日志系統建設中的典型問題,你遇到過幾個?》,這些功能反過來也對日志具備一定的要求,我們需要盡可能的將日志規范化,以減少收集、解析、分析的代價。
在 Kubernetes 中,環境的動態性很強,日志基本上都是易失的,因此需要實時將日志采集到中心的存儲中,為了配合日志采集,對于日志的輸出、采集會有更多的要求。
下述我們列舉了 Kubernetes 中,日志輸出的常見注意事項(其中標記 (*)的是 Kubernetes 中特有的項目):
日志等級是用來區分日志對應事件嚴重程度的說明,這是所有日志中必須具備的一個選項。通常日志會分為 6 個不同的等級:
作為程序員,一定要合理設置日志等級,個人在開發過程中總結了以下幾點經驗:
通常在沒有約束的情況下,程序員的發揮天馬行空,各種日志內容都會出現,這些只有開發自己才能看懂的日志很難進行分析和告警。因此我們需要一個日志頂向下的規范來約束項目中的開發人員,讓所有的日志看起來是一個人打印的而且是易于分析的。
日志中通常必備的字段有:Time、Level、Location。對于特定模塊/流程/業務,還需要有一些 Common 的字段,例如:
日志的字段規約最好由運維平臺/中間件平臺自頂向下推動,約束每個模塊/流程的程序員按照規定打印日志。
通常我們建議使用 KeyValue 對形式的日志格式,比如我們阿里的飛天日志庫采用的就是這種形式:
[2019-12-30 21:45:30.611992] [WARNING] [958] [block_writer.cpp:671] path:pangu://localcluster/index/3/prom/7/1577711464522767696_0_1577711517 min_time:1577712000000000 max_time:1577715600000000 normal_count:27595 config:prom start_line:57315569 end_line:57343195 latency(ms):42 type:AddBlock
KeyValue 對的日志可以完全自解析且易于理解,同時便于日志采集時自動解析。
另外推薦的是 JSON 日志格式,支持以 JSON 格式輸出的日志庫很多,而且大部分的日志采集 Agent 都支持 JSON 格式的日志收集。
{"addr":"tcp://0.0.0.0:10010","caller":"main.go:98","err":"listen tcp: address tcp://0.0.0.0:10010: too many colons in address","level":"error","msg":"Failed to listen","ts":"2019-03-08T10:02:47.469421Z"}
注意:絕大部分場景不建議使用非可讀的日志格式(例如 ProtoBuf、Binlog 等)。
非必要情況下,盡量不要一條日志輸出成多行,這種對于采集、解析和索引的代價都比較高。
日志的輸出量直接影響到磁盤使用以及對于應用的性能消耗,日志太多不利于查看、采集、分析;日志太少不利于監控,同時在出現問題的時候沒辦法調查。
一般線上應用需合理控制日志的數據量:
建議一個應用不同類型的日志輸出到不同的目標(文件),這樣便于分類采集、查看和監控。例如:
日志作為業務系統的輔助模塊,一定不能影響到業務正常的工作,因此日志模塊的性能消耗需要單獨額外注意,一般在選擇/開發日志庫時,需要對日志庫進行性能測試,確保正常情況下日志的性能消耗不超過整體 CPU 占用的 5%。
注意:一定要確保日志打印是異步的,不能阻塞業務系統運行。
開源的日志庫非常多,基本每個語言都有數十種,選擇一個符合公司/業務需求的日志庫需要精挑細選,有一個簡單的指導原則是盡可能使用比較流行的日志庫的穩定版本,入坑的幾率要小一點。例如:
在虛擬機/物理機的場景中,絕大部分應用都以文件的形式輸出日志(只有一些系統應用輸出到 syslog/journal);而在容器場景中,多了一個標準輸出的方式,應用把日志打到 stdout 或 stderr 上,日志會自動進入到 docker 的日志模塊,可以通過 docker logs 或 kubectl logs 直接查看。
容器的標準輸出只適應于比較單一的應用,例如 K8s 中的一些系統組件,線上的服務類應用通常都會涉及到多個層級(中間件)、和各種服務交互,一般日志都會分為好幾類,如果全部打印到容器的標準輸出,很難區分處理。<br />同時容器標準輸出對于 DockerEngine 的性能消耗特別大,實測 10W/s 的日志量會額外占用 DockerEngine 1 個核心的 CPU(單核 100%)。
在 Kubernetes 中,還可以將日志庫直接對接日志系統,日志打印的時候不落盤而直接傳輸到日志系統后端。這種使用方式免去了日志落盤、Agent 采集的過程,整體性能會高很多。
這種方式我們一般只建議日志量極大的場景使用,普通情況下還是直接落盤,相比直接發送到后端的方式,落盤增加了一層文件緩存,在網絡失敗的情況下還能緩存一定的數據,在日志系統不可用的情況下我們的研發運維同學可以直接查看文件的日志,提高整體的可靠性。
Kubernetes 提供了多種存儲方式,一般在云上,都會提供本地存儲、遠程文件存儲、對象存儲等方式。由于日志寫入的 QPS 很高,和應用也直接相關,如果使用遠程類型的存儲,會額外多 2-3 次網絡通信開銷。我們一般建議使用本地存儲的方式,可以使用 HostVolume 或者 EmptyDir 的方式,這樣對于寫入和采集的性能影響會盡可能的小。
相比傳統虛擬機/物理機的場景,Kubernetes 對于節點、應用層提供了強大的調度、容錯、縮/擴容能力,我們通過 Kubernetes 很容易就能讓應用獲得高可靠運行、極致彈性。這些優勢帶來的一個現象是:節點動態創建/刪除、容器動態創建/刪除,這樣日志也會隨時銷毀,沒辦法保證日志的存儲周期能夠滿足 DevOps、審計等相關的需求。
在動態的環境下實現日志的長期存儲只能通過中心化的日志存儲來實現,通過實時的日志采集方式,將各個節點、各個容器的日志在秒級內采集到日志中心系統上,即使節點/容器掛掉也能夠通過日志還原當時的現場。
日志輸出是日志系統建設中非常重要的環節,公司/產品線一定要遵循一個統一的日志規范,這樣才能保證后續日志采集、分析、監控、可視化能夠順利進行。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。