在web開發(fā)中隨著版本的更新迭代,通常要在系統(tǒng)中維護(hù)多個(gè)版本的api,多個(gè)版本的api在數(shù)據(jù)結(jié)構(gòu)上往往也各不相同,今天就來一起學(xué)習(xí)下kubernetes中的Scheme機(jī)制是如何解決這個(gè)問題的,如何借助HTTP請求里面的數(shù)據(jù)進(jìn)行反序列化操作
創(chuàng)新互聯(lián)建站主要從事成都網(wǎng)站建設(shè)、網(wǎng)站制作、網(wǎng)頁設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)儋州,10余年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):18982081108
通常首先是webServer先進(jìn)行Http協(xié)議的處理,然后解析成基礎(chǔ)的webServer內(nèi)部的一個(gè)Http請求對象, 通常該對象持有對應(yīng)請求的請求頭和底層對應(yīng)的字節(jié)序列(從socket流中讀取)
接著首先會(huì)通常根據(jù)對應(yīng)的編碼格式來進(jìn)行反序列化,完成從字節(jié)序列到當(dāng)前接口的業(yè)務(wù)模型的映射, 然后在交給業(yè)務(wù)邏輯處理,從而最終進(jìn)行持久化存儲(chǔ), 本文的重點(diǎn)也就在反序列化部分
/api/{version}/{resource}/{action}
上面是一個(gè)基礎(chǔ)的web url通常我們都會(huì)為每個(gè)版本注冊一個(gè)對應(yīng)的URL, 其中會(huì)包含很關(guān)鍵的兩個(gè)信息即version與resource,通過這兩個(gè)信息,通常我們就可以知道這可能是某個(gè)資源的那個(gè)版本, 如果我們把后面的action也包裹進(jìn)來,我們通常就可以知道對應(yīng)的資源的那個(gè)具體操作
在微服務(wù)流行的今天我們通常會(huì)為按照業(yè)務(wù)功能來進(jìn)行微服務(wù)的切分,本質(zhì)上一個(gè)微服務(wù)可能就是實(shí)現(xiàn)某個(gè)具體業(yè)務(wù)場景的功能集合,比如用戶系統(tǒng)通常會(huì)包含用戶的所有相關(guān)操作,在kubernetes中也有類似的概念就是所謂的Group
POST /apis/batch/v1beta1/namespaces/{namespace}/cronjobs
POST /apis/apps/v1/namespaces/{namespace}/daemonsets
我們來看一個(gè)實(shí)例這是一個(gè)創(chuàng)建daemonsets和cronjobs的url, 如果按照Group、resource、version來進(jìn)行拆分可以拆成如下:batch、v1beta1、cronjobs和apps、v1、daemonsets,也就是大家嘗試的GroupVersionKind,其中kind對應(yīng)的就是resource
我們通過url里面獲取到資源的GroupVersionKind信息,如何將其映射為一個(gè)具體的類型呢? 這貌似就很簡單了結(jié)合反射和map來進(jìn)行就可以了,我們通過url獲取到對應(yīng)想的GVK信息,然后在通過我們的映射表,就知道對應(yīng)的模型是哪個(gè),接下來就只需要進(jìn)行轉(zhuǎn)換就行了
gvkToType map[schema.GroupVersionKind]reflect.Type
那如何將對應(yīng)的Http里面的數(shù)據(jù)流反序列化成內(nèi)部的一個(gè)對象呢,別忘記了是Http協(xié)議, 肯定就是header頭里面的信息了,我們通過header頭里面的序列化就可以知道對應(yīng)的編碼格式,只需要調(diào)用對應(yīng)格式的解碼就可以完成了
Content-Type: "application/json"
如果要將json格式的字節(jié)數(shù)組進(jìn)行解碼通常要進(jìn)行如下操作,我們需要傳入一個(gè)目標(biāo)對象的指針,然后由json將對應(yīng)的字節(jié)數(shù)據(jù)解析到目標(biāo)對象中,我們也需要這樣一個(gè)對象,用于存儲(chǔ)反序列化的結(jié)果
func Unmarshal(data []byte, v interface{}) error {}
那只要我再提供一個(gè)當(dāng)前版本對應(yīng)的對象構(gòu)造函數(shù)是不是就可以呢?答案是的
func() Object{ return 目標(biāo)對象 },
首先在進(jìn)行url注冊的時(shí)候,我們構(gòu)造出對應(yīng)url映射的資源的版本信息即GroupVersionKind,后續(xù)的很多操作我們可以通過對應(yīng)的版本映射獲取對應(yīng)的目標(biāo)操作或者對象,然后再通過Header里面的字段獲取對應(yīng)的解碼器,并將Body里面的字節(jié)序列進(jìn)行解碼到目標(biāo)對象,就可以實(shí)現(xiàn)多版本資源的映射和反序列化操作了