上一篇博文我們閱讀了postgresql中查詢分析模塊的源碼。查詢分析模塊對(duì)前臺(tái)送來(lái)的命令進(jìn)行詞法分析、語(yǔ)法分析和語(yǔ)義分析后獲得對(duì)應(yīng)的查詢樹(shù)(Query)。在獲得查詢樹(shù)之后,程序開(kāi)始對(duì)查詢樹(shù)進(jìn)行查詢重寫(xiě)處理。
這一篇文章我們進(jìn)入查詢重寫(xiě)模塊源碼的閱讀。還記得上一篇文章的那張函數(shù)調(diào)用關(guān)系圖么?不記得沒(méi)關(guān)系,我再放一遍。
上次的查詢分析模塊走了1~7這些步驟。而查詢重寫(xiě)模塊即如上圖的標(biāo)記所示,函數(shù)pg_rewrite_query是進(jìn)行查詢重寫(xiě)處理的入口函數(shù)。該函數(shù)定義在src/backend/tcop/postgres.c中。
函數(shù)pg_rewrite_query的參數(shù)就是查詢分析模塊的返回值(Query)。查詢重寫(xiě)模塊使用規(guī)則系統(tǒng)判斷是否要對(duì)查詢樹(shù)進(jìn)行重寫(xiě)。如果查詢樹(shù)中每個(gè)對(duì)象(或者說(shuō)目標(biāo))被定義了轉(zhuǎn)換規(guī)則,那么程序就會(huì)使用該規(guī)則重寫(xiě)查詢樹(shù)。這部分的源碼大部分都在src/backend/rewrite文件夾下。
下面將首先介紹查詢重寫(xiě)模塊中使用的規(guī)則系統(tǒng),在此基礎(chǔ)上介紹查詢重寫(xiě)源碼的調(diào)用路徑。
1.規(guī)則系統(tǒng)
毫不夸張地說(shuō),規(guī)則系統(tǒng)是查詢重寫(xiě)模塊的核心內(nèi)容,該模塊正是根據(jù)規(guī)則系統(tǒng)中描述的規(guī)則來(lái)決定如何對(duì)一個(gè)查詢樹(shù)進(jìn)行重寫(xiě)。規(guī)則系統(tǒng)由一些列的規(guī)則組成。這些規(guī)則都存儲(chǔ)在系統(tǒng)表pg_rewrite中。該表的結(jié)構(gòu)如下圖所示。每一條記錄即為一條規(guī)則。
其中每個(gè)屬性的意義如下:
屬性名 類型 描述 oid oid 行標(biāo)識(shí)符(隱藏屬性; 必須明確選擇) rulename name 規(guī)則名稱 ev_class oid 使用這條規(guī)則的表名稱 ev_type char 規(guī)則適用的事件類型:1 = SELECT, 2 = UPDATE, 3 = INSERT, 4 = DELETEev_enabled char 控制規(guī)則在哪個(gè)session_replication_role模塊觸發(fā)。 O = 規(guī)則在 "origin" 和 "local" 模塊觸發(fā), D = 規(guī)則被禁用, R = 規(guī)則在 "replica" 模塊觸發(fā), A = 規(guī)則總是觸發(fā)。 is_instead&nbs