首頁>技術>

k8s以及雲原生相關概念近年來一直比較火熱,阿丸最近搞了個相關專案,小結一下。

本文將重點分享阿里開源專案otter適配k8s部署的改造過程,其中的改造過程和技巧應該適用於將大多數開源專案改造到k8s進行部署。

1.背景

otter是阿里開源的分散式資料庫同步系統,基於資料庫增量日誌解析,並準實時同步到本機房或異地機房的mysql/oracle資料庫(相關內容可以參考https://github.com/alibaba/otter,本文不做過多贅述)。

為了充分利用物理資源、快速擴容同步節點、擁抱雲原生,決定使用k8s部署otter。

otter的專案整體上自成一體,出於改造成本考慮,儘量在專案已有基礎上,做一些適配,不改動原始碼。

本文將重點分享對於otter適配k8s部署的改造過程,有不當之處,還請多多指教。

涉及到幾個核心內容:

otter基本架構dockerfile編寫deployment編寫啟動指令碼改造K8s中固定ip/port訪問2.otter的基本架構

典型管理系統架構,manager(web管理)+node(工作節點)

manager執行時推送同步配置到node節點node節點將同步狀態反饋到manager上基於zookeeper,解決分散式狀態排程的,允許多node節點之間協同工作基於Canal開源產品,獲取資料庫增量日誌資料。當然,otter採用了canal的嵌入式模式,不是獨立的canal節點。

基於以上部署架構,我們只需要將otter-manager和otter-node部署到k8s上。

尤其是otter-node,需要利用k8s實現節點快速水平擴充套件、計算效能彈性擴縮容。

2.Dockerfile編寫2.1 otter-manager的Dockerfile

otter-manager比較簡單,包括幾個步驟:

使用一個centos的基礎映象設定時區建立目錄複製下載好的manager.deployer-4.2.18到指定目錄使用啟動startup-new.sh指令碼啟動(這裡對原本的啟動指令碼做了一些改造,後文進行詳述)

具體如下所示:

2.2 otter-node的Dockerfile

otter-node稍微有所不同,根據官方文件說明,需要安裝aria2來做檔案傳輸。

注意注意,由於aria2安裝非常慢,因此,我們需要先安裝aria2作為一個新的基礎映象,然後在新的基礎映象上構建otter-node映象,能大大提高後續映象構建速度。

新的基礎映象如下,命名為 registry.xxx.com/xxx/otter-node-base:1.0

然後在此基礎上構建新的otter-node映象。

注意,otter-node的配置方式比較特殊,需要先在otter-admin上獲取一個nid,然後才能執行一個otter-node。

所以,我們在dockerfile中以ARG 宣告一個nid,然後在後續構建映象的時候,透過 --docker-arg 傳入nid具體的值。

當然,如果把nid看作一個配置檔案的話,也可以用下文提到的configmap的形式在Deployment中掛載進去

3.Deployment編寫

什麼是Deployment?

k8s的一個Deployment控制器為 Pods 和 ReplicaSets 提供宣告式的更新能力。我們透過編寫Deployment描述期望的目標狀態,然後 Deployment 控制器更改Pods或者ReplicaSets的實際狀態, 使其變為期望狀態。

具體關於Deployment的知識不展開說明,可以參考k8s官方文件。

我們需要部署測試環境與生產環境兩套叢集,而無論是otter-manager還是otter-node,都依賴於讀取 conf/otter.properties 作為配置。

因此,我們需要根據環境,修改不同的otter.properties。

那麼,對於k8s部署來說,可以採用同一份映象,然後在不同環境(k8s的不同namespace)中將otter.properties作為configmap寫入,最後透過volume的形式掛載到pod的指定路徑上。

這裡對幾個名詞做簡單介紹,詳細內容可以參考k8s官方文件。

Configmap:一種 API 物件,用來將非機密性的資料儲存到健值對中。使用時,Pods可以將其用作環境變數、命令列引數或者儲存卷中的配置檔案。(如果想儲存的資料是機密的,可以使用 Secret,而不是configmap)Volume:卷的核心是包含一些資料的一個目錄,Pod 中的容器可以訪問該目錄。 所採用的特定的卷型別將決定該目錄如何形成的、使用何種介質儲存資料以及目錄中存放 的內容。使用卷時, 在 .spec.volumes 欄位中設定為 Pod 提供的卷,並在 .spec.containers[*].volumeMounts 欄位中宣告卷在容器中的掛載位置。

所以,首先在指定環境(namespace中)建立configmap,以otter.properties作為key,以檔案內容作為value。

具體命令如下

kubectl create configmap otter-manager-dev-config --from-file=otter.properties=conf/otter-dev.properties -n otter-system

在namespace為otter-system中建立一個名字為otter-manager-dev-config的configmap,其中,以otter.properties作為key,以檔案內容作為value。

產生的configmap如下圖所示

然後,將這個configmap在Deployment中用volume進行引用,然後透過volumeMounts掛載到指定目錄,Deployment具體如下所示。

這裡需要特別注意volumeMounts的路徑覆蓋問題,需要在volumeMounts中配置subPath為具體檔名。

4.啟動指令碼改造

Otter包括兩個部分,管理控制檯manager和工作執行節點node,正常情況下都是用各自的啟動指令碼startup.sh啟動的。

為了適配k8s,我們需要對啟動指令碼做改造,本文以otter-manager的啟動指令碼為例,otter-node也是類似。

將啟動指令碼startup.sh改造為 startup-moon.sh,重點解決兩個問題

前臺程序保持執行jvm引數自定義改造4.1 前臺程序保持執行

由於容器中用entrypoint啟動的程序為1號程序,一旦1號程序執行結束,容器就會退出了。

而原本的startup.sh中,用java啟動後,使用 “&” 將java程序轉換為後臺程序,所以startup.sh作為1號程序會很快執行結束,容器就會自動退出了。

所以我們需要將1號程序保持住,不要退出。

這裡考慮了兩個方案:

Startup.sh指令碼中增加一個前臺程序進行保持,比如 tail -f /dev/null 命令將“&”去掉,讓java啟動後就作為前臺程序一直保持

後來考慮了一下,還是選擇了方案二。主要原因是為了利用pod自動重啟的特性。

如果Java程序意外退出了,那麼方案二就能使得1號程序也結束,然後pod就能自動重啟了。而方案一的話,由於startup.sh指令碼仍然在執行tail,所以即使java程序退出,1號程序也不會結束。

具體修改如下:

最終pod中的程序如圖所示

1號程序是啟動指令碼1號程序的子程序是otter的java應用程序(前臺程序)4.2 虛擬機器大小自定義配置

由於otter專案中,將jvm的啟動引數配置在了start.sh中,不方便進行手動配置。

因此,將start.sh的配置jvm引數的邏輯註釋掉,採用自己配置的環境變數JAVA_OPTIONS進行注入。

這個環境變數的注入方式也比較簡單,就是在上面的Deployment中的env配置的(藍色框部分),方便以後手動修改jvm引數大小而不用修改映象。

5.k8s上固定IP/Port訪問

otter-node的部署中,有個比較特殊的地方。

不同於普通的微服務的無狀態擴充套件,otter-node的部署必須指定nid、ip、port,這種設計據說是為解決單機部署多例項而設計的,允許單機多node指定不同的埠(具體可以參考官方wiki,https://github.com/alibaba/otter/wiki/Node_Quickstart,這裡不展開說明)。

還是直接看看如何在k8s上進行適配吧。

這裡採用了k8s的NodePort進行處理。

在上面的配置中,可以使用IP1:3000 或者 IP2:3000 或者 IP3:3000 訪問service。

當然,為了保證不繫結特定KVM的IP,我們在前面掛一個SLB服務,透過訪問SLB的 虛擬IP:PORT 的形式訪問。

對於otter部署來說,otter-manager需要兩組 IP:PORT、每個node需要三組 IP:PORT。

注意,由於otter部署中,每個node需要暴露的port都是不同的,所以每次新增一個otter-node,都需要新增三組 IP:PORT。

我們以otter-node為例,來看下NodePort型別的Service的yml檔案吧。

kind為servicetype為NodePort配置了三組埠。port/targetport都是應用暴露埠,而nodePort是對外訪問埠。6.總結

經過這樣的改造,我們就能用k8s的部署otter-manager和otter-node了,並且能夠快速擴容節點、彈性使用機器資源。

我們回顧一下其中的關鍵問題和技巧:

Dockerfile編寫中,可以把環境相關依賴打成一個新的基礎映象,提高後續應用映象的構建速度。Dockerfile中,可以透過ARG定義一些構建過程中的變數,進行替換。對於不同環境的配置檔案,可以在不同環境(k8s的namespace)下配置不同的configmap,然後在Deployment檔案中透過volumeMounts的方式掛載進去。對於後臺程序,需要改造為前臺程序使得pod能夠保持對於一些特定的環境變數,可以在Deployment中透過env進行傳入。

其他開源專案如果有需要上k8s的,這些技巧應該都能用上。

阿丸把 Canal原始碼解析與實戰筆記HBase原理與實戰筆記MySQL實戰筆記Java實戰技巧筆記 等整理為合集,全是原創手打幹貨,免費分享給大家。

11
最新評論
  • BSA-TRITC(10mg/ml) TRITC-BSA 牛血清白蛋白改性標記羅丹明
  • 超詳細講解IJKPlayer的播放器實戰和原始碼分析(4)