做者:韓鋒出處:DBAplus社羣分享 前端
Themis開源地址:https://github.com/CreditEaseDBAgit
拓展閱讀:宜信開源|數據庫審覈軟件Themis的規則解析與部署攻略github
我相信,這也是不少公司、不少DBA正在面臨或將來都會面臨的一些問題。正是存在問題,促使咱們考慮引入數據庫審覈平臺。sql
首先是運維規模與人力資源之間的矛盾。從咱們的狀況來看,運維了包括Oracle、MySQL、MongoDB、Redis四類數據庫,數據庫規模幾十套,支持公司千餘名開發人員及上百套業務系統。也許有的朋友會問,從運維規模上看,並非很大。mongodb
的確,與不少互聯網公司相比,數據庫數十套的估摸並非太大;但與互聯網類公司不一樣,相似宜信這類金融類公司對數據庫的依賴性更大,大量的應用是重數據庫類的,且其使用複雜程度也遠比互聯網類的複雜。DBA除了平常運維(這部分咱們也在經過自研平臺提高運維效率)外,還須要有大量精力應對數據庫設計、開發、優化類的工做。當面對大量的開發團隊須要服務時,這個矛盾就更加凸顯出來。數據庫
結構設計json
第二個挑戰,是數據庫設計、開發質量良莠不齊的問題。 上圖就展現了一個結構設計問題。某核心系統的核心表,在這個系統運行的SQL中,28%都是跟這個對象有關的。當咱們分析其結構時,發現了不少的問題:segmentfault
綜上所述,這個表設計的問題還有不少,並且這個表很是重要,大量語句訪問和其相關。緩存
SQL語句
上圖展現的是一個語句運行效率的問題。從字面可見,兩個表作關聯查詢,但在指定條件時沒有指定關聯條件。在下面的執行計劃中可見,數據庫採用了笛卡爾積的方式運行。從後面的成本、估算時間等可見,這是一個多麼「巨大」的SQL。其在線上運行的影響,可想而知。
也許有人會說,這是一我的爲失誤,通常不會發生。但我要說的是,第一,人爲失誤沒法避免,誰也不能保證寫出SQL的運行質量;第二,開發人員對數據庫的理解不一樣,很難保證寫出的SQL都是高效的;第三,開發人員面臨大量業務需求,常常處理趕工狀態,很難有更多的精力放在優化上面。這由於有這些問題,線上語句執行質量就成了DBA常常面臨的挑戰之一。
這是一張很經典的圖,它描述了和數據庫相關工做的職能劃分。做爲DBA,除了面臨以上挑戰外,從數據庫工做發展階段及自身發展需求來看,也面臨一個重心的轉移:原有傳統DBA的運維職能逐步被弱化,大量的工具、平臺的涌現及數據庫自我運維能力的提高,簡化DBA的工做;緊隨而來的數據庫架構、結構設計、SQL質量優化逐步成爲重點;再往上層的數據治理、建模等工做也愈來愈受到一些公司的重視。因而可知,DBA將來工做的中心也逐步上移。對中間數據邏輯結構部分,也須要一些工具、平臺更好地支撐DBA的工做。
除上述狀況外,我司還存在幾種的不平衡。
公司自有團隊人員上,仍是以初中級爲主,中高級人員相對較少。如何快速提高總體設計、優化能力,保證統一的優化效果成爲擺在面前的問題。
正是有了上述多種的不平衡,促使咱們考慮引入工具、平臺去解決數據庫質量問題。
我剛來到公司時,看到公司的這些問題,也曾考慮經過制度、規範的形式進行解決。一開始就着手製定了不少的規範,而後在各個部門去培訓、宣講。這種方式運行一段時間後,暴露出一些問題:
面臨上述這些挑戰、現存的各類問題,該如何解決?
通過討論,最後你們一致認爲,引入數據庫審覈平臺,能夠幫助解決上面所述問題。
在項目之初,我考察了業內其它企業是如何數據庫審覈的,大體可分爲三個思路:
第一類,是以BAT公司爲表明的互聯網類公司。它們經過自研的SQL引擎,可實現成本分析、自動審覈、訪問分流、限流等,可作到事前審覈、自動審覈。但技術難度較大,公司現有技術能力明顯不足。
第二類,是經過自研工具收集DB運行狀況,根據事前定義規則進行審覈,結合人工操做來完成整個審覈流程。這種方案只能作到過後審覈,但技術難度較小,靈活度很大。其核心就是規則集的制定,可根據狀況靈活擴展。
第三類,是一些商業產品,實現思路相似第二類,可是加上一些自主分析能力,功能更爲強大,但仍需人工介入處理且須要不小資金投入。並且考察幾款商業產品,沒有能徹底知足所需功能的。
綜合上面幾類作法,最終肯定咱們採用「工具+人工審覈」的方式,自研本身的審覈平臺。
在啓動研發這一平臺之初,咱們就在團隊內部達成了一些共識。
下面來看看,審覈平臺的基本功能及實現原理及方法,這部分是本次分享的重點。
在項目之初,咱們就平臺的定位作了描述:
做爲平臺的兩類主要使用方,研發人員和DBA均可以從平臺中受益。
整個平臺的基本實現原理很簡單,就是將咱們的審覈對象(目前支持四種),經過規則集進行篩選。符合規則的審覈對象,都是疑似有問題的。平臺會將這些問題及關聯信息提供出來,供人工甄別使用。因而可知,平臺的功能強大與否,主要取決於規則集的豐富程度。平臺也提供了部分擴展能力,方便擴展規則集。
審覈對象
在開始介紹平臺實現以前,再來熟悉下「審覈對象」這個概念。目前咱們支持的有四類對象,分別說明一下。
須要說明一下,這四類審覈對象中,後三種必須在系統上線運行後纔會抓取到,第一種能夠在只有數據結構的狀況下運行(個別規則還須要有數據)。
此外,上述規則中,除了第二類爲通用規則外,其餘都與具體數據庫相關。即每種的數據庫,都有本身不一樣的規則。
架構簡圖
這裏畫出是系統架構框架簡圖,我簡單說明一下。
圖中的方框部分,爲平臺的主要模塊。底色不一樣的模塊,表示當前的進度狀態不一樣。虛線表明數據流,實線表明控制流。其核心爲這幾個模塊:
流程圖
讓咱們從處理流程的角度,看看平臺的總體處理過程。
1) 「規則管理」部分,這部分主要完成如下一些功能。
2) 「任務管理」部分,這是後臺管理的一個部分,主要完成與任務相關的工做。系統中的大多數交互都是經過做業異步完成的。其後臺是經過celery+flower實現的。
3) 「數據採集」部分,這部分是經過任務調度定時出發採集做業完成,也有少許部分是實時查詢線上庫完成的。採集的結果保存在數據庫中,供後續分析部分調用。
4) 「規則解析」部分,這部分是由用戶經過界面觸發,任務調度模塊會啓動一個後臺異步任務完成解析工做。之因此設計爲異步完成,主要是審覈工做可能時間較長(特別是選擇審覈類別較多、審覈對象不少、開啓的審覈規則較多)的狀況。審覈結果會保存在數據庫中。
5) 「任務查看、導出」部分,在用戶發起審覈任務後,可在此部分查看進度(處於審覈中、仍是審覈完成)。當審覈完成後,可選擇審覈任務,瀏覽審覈結果或選擇導出都可。若是是選擇導出的話,會生成異步後臺做業生成文件,放置在下載服務器上。
以上就是整個審覈的大致流程。後續將看到各部分的詳細信息。
模塊劃分
總結一下,平臺主要是由上述四個模塊組成:數據採集、規則解析、系統管理、結果展現。後面將針對不一樣模塊的實現,進行詳細說明。
採集內容
先來看看數據採集模塊。從表格可見,兩種類型數據庫的採集內容不一樣。
Oracle提供了較爲豐富的信息,須要的基本均可採集到;MySQL功能相對能採集到的信息較少。
表格中的「對號+星號」,表示非定時做業完成,而是後面實時回庫抓取的。下面簡單說下,各部分的採集內容。
這些信息都將做爲後面審覈的依據。
採集原理
下面簡單介紹下采集的與原理:
概要說明
下面介紹整個系統最爲核心的部分—規則解析模塊,它所完成的功能是依據定義規則,審覈採集的數據,篩選出違反規則的數據。對篩選出的數據進行計分,並記錄下來供後續生成審覈報告使用。同時還會記錄附加信息,用於輔助進行一些判斷工做。
這裏有個核心的概念—「規則」。後面能夠看到一個內置規則的定義,你們就會比較清楚了。從分類來看,可大體分爲如下幾種。
規則定義
這是一個規則體的聲明對象,我說明一下各字段含義,你們也可對規則有個清晰的認識。
規則定義(對象級)
先來看第一類規則—對象規則。這是針對數據庫對象設置的一組規則。上面表格,顯示了一些示例。常見的對象,諸如表、分區、索引、字段、函數、存儲過程、觸發器、約束、序列等都是審覈的對象。以表爲例,內置了不少規則。
例如:第一個的「大表過多」。表示一個數據庫中的大表個數超過規則定義閥值。這裏的大表又是經過規則輸入參數來肯定,參數包括表記錄數、表物理尺寸。總體描述這個規則就是「數據庫中超過指定尺寸或指定記錄數的表的個數超過規定閥值,則觸發審覈規則」。其它對象的規則也相似。
規則實現(對象級)
對象規則的實現部分,比較簡單。除個別規則外,基本都是對數據字典信息進行查詢,而後依據規則定義進行判斷。上面示例就是對索引的一個規則實現中,查詢數據字典信息。
規則定義(執行計劃級)
第二類規則是執行計劃類的規則,它也劃分爲若干類別。例如訪問路徑類、表間關聯類、類型轉換類、綁定變量類等。
以最爲常見的的訪問路徑類爲例,進行說明下。如最爲常見的一個規則「大表掃描」。它表示的是SQL語句的執行中,執行了對大表的訪問,而且訪問的路徑是採用全表掃描的方式。這個規則的輸入參數,包含了對大表的定義(物理大小或記錄數);輸出部分則包括了表名、表大小及附加信息(包括整個執行計劃、指定大表的統計信息等內容)。
這類規則針對的數據源,是從線上數據庫中抓取的。Oracle部分是直接從AWR中按時間段提取的,MySQL部分是使用explain命令返查數據庫獲得的。
信息存儲格式
在這裏特別說明一下,在保存執行計劃的時候,使用了MongoDB這種文檔性數據庫。目的就是利用其schemaless特性,方便兼容不一樣數據庫、不一樣版本執行計劃的差別。均可以保存在一個集合中,後續的規則審覈也是利用的mongo中的查詢語句實現的。這也是最初引入mongo的初衷,後續也將其它類信息放入庫中。如今整個審覈平臺,除了pt工具接入的部分使用MySQL外,其他都在MongoDB中。此外,MySQL庫能夠直接輸出json格式的執行計劃,很方便就入庫了;Oracle部分也組成json格式入庫。
規則實現(執行計劃)
左邊就是一個Oracle的執行計劃保存在MongoDB中的樣子。其實就是將sqlplan字典數據插入到mongo中。右側就是一個規則實現的樣例,就是基於mongo的查詢語句。後面咱們會可看到一個詳細的示例。
規則實現
這裏以「大表全表掃描」規則爲例,進行說明。上面是在Oracle中的數據字典保存的執行計劃,下面是存在Mongo中的。可見,就是徹底複製下來的。
基於這樣的結構,如何實現規則過濾呢?其實就是經過mongo中的find語句實現的。下面具體解讀下這個語句的執行步驟。
規則實現(執行計劃)
這部分是MySQL中實現層次結果存儲的一個實例。
第一個圖展現的是原始的執行計劃。
第二個圖是代碼實現的摘要。
第三個圖是真正保存在庫中的樣子。核心部分就是對item_level的生成。
規則定義(文本級)
第三類規則是文本類的規則,這是一類與數據庫種類無關、描述SQL語句文本特徵的規則。在實現上是採用文本正則匹配或程序方式進行處理的。它的主要目的是規範開發人員的SQL寫法,避免複雜的、性能較差的、不規範的SQL寫法。
規則實現(文本級)
這部分描述的是文本規則的實現方式。第一個示例bad_join,是一種簡單規則,經過正則文本匹配實現。第二個示例sub_query,是經過程序判斷括號嵌套來完成對子查詢(或多級子查詢)的判斷。
規則定義(執行特徵級)
最後一類規則是執行特徵類的。這部分是與數據庫緊密關聯的,將符合必定執行特徵的語句篩選出來。這些語句不必定是低效的,可能只是將來考慮優化的重點,或者說優化效益最高的一些語句。這裏面主要都是一些對資源的消耗狀況等。
規則管理
後面經過一些界面展現,介紹下平臺的功能。
第一部分系統管理模塊中規則管理的部分。在這部分,可完成新增自有規則。其核心是規則實現部分,經過SQL語句、Mongo查詢語句、自定義Python文件的形式定義規則實現體。自定義規則的依據是現有抓取的數據源,定義者須要熟悉現有數據結構及含義。目前尚不支持自定義抓取數據源。
對定義好的規則,可在此處完成規則修改。主要是對規則狀態、閥值、扣分項等進行配置。
任務管理
在配置好規則後,可在此處完成任務發佈的工做。
上面是規則任務發佈的界面,在選擇數據源(ip、port、schema)後,選擇審覈類型及審覈日期。目前審覈數據源的定時策略仍是以天爲單位,所以日期不能選擇當天。
當任務發佈後,可在任務結果查看界面觀察執行狀況。根據審覈類型、數據源對象多少、語句多少等,審覈的時長不定,通常是在5分鐘之內。當審覈做業狀態爲「成功」時,表明審覈做業完成,能夠查看或導出審覈結果了。
對象審覈結果概覽
上圖是一個對象審覈報告的示例。在報告的開頭部分,是一個概覽頁面。它集中展現審覈報告中各種規則及扣分狀況;並經過一個餅圖展現其佔比狀況。這便於咱們集中精力先處理核心問題。
在最上面,還能夠觀察到有一個規則總分的顯示。這是咱們將規則扣分按照百分制,折算後獲得的一個分數。分值越高,表明違反的狀況越少,審覈對象的質量越高。引入「規則總分」這一項,在設計之初是有些爭議的,擔憂有了這個指標會比較打擊開發人員的積極性,不利於平臺的推廣使用。這裏有幾點,說明一下。
對象審覈結果明細
這部分是對象審覈的明細部分,對應每一個規則其詳細狀況,可在左側連接中進一步查看對象信息。篇幅所限,不作展現了。
執行計劃審覈結果概覽
這部分執行計劃的概覽展現,跟對象的狀況相似。也是每種規則的扣分狀況。
執行計劃審覈結果明細
這部分是執行計劃的明細部分。
展開以後,能夠看到違反每種規則的明細。上圖就是違反全表掃描的規則的明細部分。
在上面是一些通用的解決方案說明。這裏將可能觸發此類規則的狀況及解決方案進行了說明。至關於一個小知識庫,便於開發人員優化。後面在平臺二期,會作更爲精準的優化引擎部分,這部分還會展開。
下面是每條違反的語句狀況,咱們能夠看到語句文本、執行計劃、關聯信息(例如此規則的大表名稱)等。還能夠進一步點開語句,展開信息。
這部分是針對每條SQL的信息,包括語句文本、執行計劃、執行特徵、關聯對象統計信息等。DBA可從這些信息就能夠作一些初步的優化判斷工做。
此外,平臺也提供了導出功能。可導出爲excel文件,供用戶下載查看。這裏就展現了。
在實際開發過程當中,碰到了不少問題。咱們這裏簡單介紹兩個,例如:
MySQL在解析json格式執行計劃中暴露出的問題…
【會話進入sleep狀態,假死】
解決方法:執行會話以前設置wait_timtout=3,這個時間根據實際狀況進行調整。
【數據量過大,長時間沒有結果】
會話處於query狀態,可是數據量很大或由於數據庫對format=json支持不是很好,長時間解析不出來,會影響其餘會話。
解決方法:使用pt-kill工具殺掉會話。爲了防止誤殺,可打個標識「eXplAin format=json」,而後使用pt-kill識別eXplAin關鍵字。
此平臺在宜信公司運行以來,爲不少系統提供了審覈報告,大大加快了數據庫結構、SQL優化的速度,減輕了DBA的平常工做壓力。在工做實施過程當中,咱們也摸索了一套推行方法。該平臺已開源後,若有朋友使用,可參考實施。
收集信息階段
海量收集公司的數據庫系統的運行狀況,掌握第一手資料。快速瞭解各業務系統的質量,作好試點選擇工做。
人工分析階段
重點系統,人工介入分析。根據規則審覈中暴露出的核心問題,「以點帶面」,有針對性的給出分析及優化報告。
交流培訓階段
主動上門,跟開發團隊溝通交流報告狀況。借分析報告的機會,可對開發團隊進行必要的培訓工做,結合他們身邊的案例,更具備說服做用。
反饋改進階段
落實交流的成果,督促其改進。經過審覈平臺按期反饋改進質量。有必定基礎的團隊,可開發平臺,供開發人員本身使用。使SQL質量問題,再也不僅僅是DBA的問題,而和項目中的每一個人都有關係。
內容來源:宜信技術學院