移動端的跨平臺技術不是一個新話題,早在幾年前,WebView容器、React Native、Weex、Flutter、小程序等移動端跨平臺框架就風起雲涌。爲何跨平臺這麼有吸引力呢?咱們設想一下若是能夠作到一次開發,多端複用,那麼對於公司來講,就能夠下降用人成本。對於開發來講,只須要學習一個框架,就能夠在Android和iOS雙平臺上開發。節約下來的成本,能夠投入到產品快速驗證、快速上線。這對全部人來講都有着極大的吸引力。本節先針對部分移動端跨平臺技術進行一些簡要的介紹,以便讀者可以更好地理解後面的內容。html
1.1.1 WebView容器前端
WebView容器的工做原理是基於Web技術來實現界面和功能,經過將原生的接口封裝、暴露給JavaScript調用,JavaScript編寫的頁面能夠運行在系統自帶的WebView中。這樣作的優點是,對於前端開發者比較友好,能夠很快地實現頁面跨端,同時保留調用原生的能力,經過搭建橋接層和原生能力打通。但這種設計,跨端的能力受限於橋接層,當調用以前沒有的原生能力時,就須要增長橋。另外,瀏覽器內核的渲染獨立於系統組件,沒法保證原生體驗,渲染的效果會差很多。ios
1.1.2 React Native小程序
2015年,Facebook推出了React Native,一經推出就備受關注。它的思路是最大化地複用前端的生態和Native的生態,和WebView容器的最大區別在於View的渲染體系。React Native拋棄了低效的瀏覽器內核渲染,轉而使用本身的DSL生成中間格式,而後映射到對應的平臺,渲染成平臺的組件。相對WebView容器,體驗會有必定的提高。不過,渲染時須要JavaScript和原生之間通訊,在有些場景可能會致使卡頓。另外就是,渲染仍是在Native層,要求開發人員對Native有必定的熟悉度。後端
1.1.3 Flutter瀏覽器
2018年Google推出Flutter,經過Dart語言構建一套跨平臺的開發組件,全部組件基於Skia引擎自繪,在性能上能夠和Native平臺的View相媲美。Flutter站在前人的肩膀上,參考了React的狀態管理、Web的自繪製UI、React Native的HotReload等特色,同時考慮了與Native通訊的Channel機制、自渲染、完備的開發工具鏈。Flutter與上述Recat Native、WebView容器本質上都是不一樣的,它沒有使用WebView、JavaScript解釋器或者系統平臺自帶的原生控件,而是有一套本身專屬的Widget,底層渲染使用自身的高性能C/C++ 引擎自繪。但大部分移動端發展到今天,都已經造成了本身的架構,在現有基礎上加上Flutter,會造成原有架構和Flutter雙平臺共存的問題。目前,對新的App來講,是最被看好的跨端方案。微信
做爲中國領先的生活服務電子商務平臺,美團致力於用科技鏈接消費者和商家,提供服務以知足人們平常「吃」的需求,並進一步擴展至多種生活和旅遊服務。而做爲公司最爲重要的業務之一,美團外賣從2013年建立以來,已經從單一的品類擴展到附近美食、水果、蔬菜、超市、鮮花、蛋糕等多品類,從早午晚餐,發展到下午茶、宵夜,中餐、西餐、家常菜、小吃、快餐、海鮮、火鍋、川菜、蛋糕、烤肉、水果、飲料、甜點等多種類餐飲。美團外賣能夠說是當前電商領域,最爲複雜的業務之一。網絡
業務的複雜,給系統架構也帶來了不小的挑戰。美團外賣業務之因此說是當前電商領域最爲複雜的業務,主要源於如下幾點特徵:數據結構
綜上所述,能夠發現美團外賣不只僅自身業務比較複雜,並且對外的角色也很複雜。在美團內部,外賣不只僅是美團平臺的一個頻道業務,並且本身自己也是一個平臺業務,同時美團外賣還承擔着新業務發展的平臺角色。這意味着想要支持好美團外賣業務的發展是一件很是有挑戰的事情。架構
好的架構源於不停地衍變而非設計。美團外賣的架構,歷史上也是經歷了不少次迭代。因爲外賣業務形態不斷地發生變化,原有的設計也須要不斷地跟隨業務形態進行演進。在不斷探索和實踐過程當中,咱們經歷了若干個大的架構變遷。從考慮如何高效地複用代碼支持外賣App,逐漸地衍變成如何去解決多端代碼複用問題,再從多端的代碼複用到支持其餘頻道業務的平臺架構上。在平臺化架構建設完成後,咱們又開始嘗試利用動態化技術去支持業務快速上線的訴求。現在,咱們面臨着多端複用、平臺能力、平臺支撐、單頁面多業務團隊、業務動態訴求強等多個業務場景問題。下文咱們針對美團外賣移動端架構的變遷史,作一些簡單的概述,以便讀者閱讀本文時能有更好的延續性。
1.3.1 組件化架構
早期階段,美團外賣做爲公司的一個孵化業務,在2013年末完成了美團外賣App的1.0版本。隨着外賣業務的驗證成功和跑通,訂單量也快速增加,在2014年末突破了日訂單量100萬。隨後在2015年2月,外賣以Native的形式接入美團App,成爲美團App的一個業務頻道。在接入過程當中,咱們從美團外賣App拷貝了大量的代碼到美團App的外賣頻道,兩個App上的外賣業務代碼也分別由兩個獨立的團隊維護。早期外賣業務變化快,App迭代頻繁,寫代碼的方式也比較粗放,同時美團App也處在一個平臺化轉變的時期,代碼的穩定性和質量都在變化和提高當中。這些因素致使了外賣代碼內各子系統之間耦合嚴重,邊界模糊,「你中有我,我中有你」的狀況隨處可見。這對代碼質量、功能擴展以及開發效率都形成很大的影響。此時,咱們架構重構的目的,就是但願將各個子系統劃分爲相對獨立的組件,建設組件能夠直接複用,架構以下圖所示:
1.3.2 平臺化架構
如上文所述,你們能夠知道美團外賣和美團外賣頻道是由不一樣的團隊在維護髮展。2015年,公司考慮到業務發展的一致性,將美團外賣頻道團隊正式歸於美團外賣。從組織架構上來講,美團外賣和美團外賣頻道,逐漸融合成一個團隊,可是兩端的差別性,致使咱們不得不仍然階段性地維持原有的兩班人馬,各自去維護獨立外賣App和美團外賣頻道。如何解決這個問題?兩端代碼複用看起來是惟一的途徑。另外,隨着業務的快速發展,外賣App所承載的業務模塊愈來愈多,產品功能愈來愈複雜,團隊規模也愈來愈大,如閃購、跑腿等業務想以獨立的Native包的形態接入外賣App,還有外賣的異地研發團隊的創建,都帶來了挑戰。這使得咱們在2017年開始了第二次架構重構——平臺化架構,目標是但願可以支持多端複用和支持不一樣團隊的業務發展。經過抽象出平臺能力層、業務解耦、創建殼容器,最終實現了平臺化架構,架構以下圖所示:
1.3.3 RN混合架構
在平臺化架構以後,美團外賣功能持續增長,美團外賣客戶端安裝包的體積也在持續增長。回顧2017年和2018年,每一年幾乎都增加100%。若是沒有一個有效的手段,安裝包將變得愈加臃腫。另外,因爲原生應用須要依託於應用市場進行更新,每次產品的更新,必須依賴用戶的主動更新,使得版本的迭代週期很長。業務上的這些痛點,不斷地督促咱們去反思到底有沒有一種框架能夠解決這些問題。
在2015年的時候,Facebook發佈了很是具備顛覆性的React Native框架,簡稱RN。從名字上看,就能夠清楚的明白,這是混合式開發模式,RN使用Native來渲染,JS來編碼,從而實現了跨平臺開發、快速編譯、快速發佈、高效渲染和佈局。RN做爲一種跨平臺的移動應用開發框架,它的特性很是符合咱們的訴求。美團也積極的探索RN技術。在RN的基礎上,美團在腳手架、組件庫、預加載、分包構建、發佈運維等方面進行了全面的定製及優化,大幅提高RN的開發及發佈運維效率,造成了MRN(Meituan React Native)技術體系。
從2018年開始,美團外賣客戶端團隊開始嘗試使用MRN框架來解決業務上的問題。使用RN的另外一方面的好處是,能逐漸的抹平Android和iOS開發技術棧帶來的問題,使用一套代碼,兩個平臺上線,理論上人效能夠提高一倍,支持的業務需求也能夠提高一倍,架構以下圖所示:
上文說到,外賣業務已經發展到多App複用、單頁面多業務團隊開發的業務階段。要知足這樣的業務場景下,尋求一個可持續發展的業務架構是件不容易的事情。通過咱們以前架構演進,咱們得到了寶貴的經驗:在平臺化架構的時候,咱們將App和業務進行解耦,將App作成殼容器,業務造成獨立的業務庫,集成到殼容器裏面,從而屏蔽了多App的問題,提升了業務的複用度。在RN混合式架構裏面,咱們引入了RN容器,經過這個容器,使得業務屏蔽了Android和iOS的平臺差別。藉助這些成功的經驗,咱們進一步思考,若是咱們嘗試進一步的細分外賣的業務場景,將不一樣場景下的基礎能力建設成殼容器,業務集成到容器內,是否能夠更好的支撐咱們多App複用、單頁面多業務團隊的當前現狀呢?
容器化架構的願景是:
當咱們把承載外賣業務的環境進行了抽象和標準化後,就能夠得到如下若干點好處。首先動態化屬性提高,咱們能夠把原有必須在客戶端上寫的業務放到了遠端,業務的動態性獲得很大的提高,具有隨時上線業務的可能。對於開發過程而言,編譯部署的速度也獲得了極大提高。若是涉及到客戶端的代碼改動,那客戶端的編譯打包,即便是增量的編譯,也至少是秒級的編譯速度。而容器化後,咱們只打包必要的業務,把業務動態下發到容器呈現,客戶端代碼自己不會有變化,這樣就能夠從秒級的編譯減小到毫秒級的編譯。一樣,業務動態下發,對減小客戶端的包大小也有很大的幫助。
而後,容器位於應用以內,咱們嚮應用中引入相同的容器SDK,容器屏蔽了應用之間的差別,對於Android和iOS平臺,在設計上,經過容器這一層去儘量屏蔽平臺之間的差別,使業務開發人員只須要認識容器,不須要花費大量的精力去關注應用和平臺之間的差別,從而使得開發效率獲得了極大的提高。
其次,容器化後,容器對承載的內容是有接口協議要求的,承載的內容只有知足容器定義的協議才能獲得容器帶來的好處,這促使業務獲得了更細粒度的細分,業務開發時候,對模塊化的意識獲得了保障。另外,容器這一層提供的接口在Android和iOS上是標準化的,業務的開發也由於依賴的標準化,而趨向標準化,雙端的業務一致性獲得了提高。這些潛在的架構好處,對將來的業務維護和擴展都打下了比較好的地基。
整個外賣容器化架構能夠按照從下到上,從左到右的視角進行解讀:
最底層是系統服務,由於咱們採用了H5和RN這樣跨端的技術棧,使得iOS系統和Android系統成爲了最底層。
系統服務之上是集團基於Native建設的基建,全公司通用,覆蓋了研發工程中方方面面的基礎服務。
在基建之上是咱們定義的容器層。咱們嘗試用單一技術棧解決全部問題。但通過咱們的探索,以爲不太可能實現。好的架構要匹配業務形態,業務的訴求決定了咱們不能選擇惟一的技術棧去解決全部問題,細分外賣的業務場景可獲得如下3個方向的頁面分類:
再往上,就是垂直的業務,外賣目前有流量業務、交易業務、商家業務、商品業務、廣告業務、營銷業務、閃購業務等。業務都是垂直向下依賴,直接可見容器,可見基建,可以很好地獲取到各類已經建設的能力去完成業務的需求。
最上面是承載的App端,目前有四端,包括外賣、點評、美團、閃購等等。
右側是測試發佈和線上監控,相對於常規的移動端App架構而言,容器化架構的測試發佈和監控是更爲精細化的。不只僅要關注端自己的可用性,還須要關注容器、容器承載的模塊、模塊展現的模板,模板裏面的樣式這些的可用性。
容器化架構相對常規的移動端架構而言,它從管理移動端的代碼轉變成管理移動端的容器建設代碼和業務遠端開發代碼,多出了容器和業務遠端下發。這不只僅是對技術上的挑戰,對長期作客戶端開發同窗,也須要一個思惟轉變的跳轉。
一致性的挑戰:容器須要在多個宿主應用之中運行,宿主應用的環境一致性直接影響了容器的一致性。咱們的策略是兩手準備,一方面利用外賣業務的優點推進宿主應用的環境對齊;另外一方面將容器建設成SDK,經過SDK將長期保持容器的一致性,也經過SDK內部的設計屏蔽應用之間的差別;對於Android和iOS平臺,咱們經過分層的設計,儘量屏蔽平臺的差別。綜上所述,一致性的挑戰在於(1)容器運行的宿主應用的環境一致性;(2)不一樣應用不一樣版本容器的一致性;(3)Android和iOS平臺容器的對業務的一致性。
動態發佈的挑戰:長期以來,客戶端同窗的開發概念裏面只有App版本的概念,而當咱們逐漸把業務代碼作成遠端下發時,將會新增一個線上動態發版的概念。當咱們在發佈業務的時候,相對以往的工做,多出須要去考慮這個業務的版本,能夠運行的容器對應的App上下界版本。另外,發版的週期也會新增業務的發版週期,不只僅是App的發版週期。這二者在一塊兒將會產生新的火花,業務的版本和App的版本如何適配的問題,業務動態發版的週期和App的發版週期如何適配的問題。外賣這邊的解決方式是建設主版本迭代+周迭代的模型。
監控運維的挑戰:以往的移動端架構,咱們更加關注的是端自己的可用性,然而當咱們演進到容器化架構的時候,僅僅關注端的可用性已經遠遠不能肯定業務是可用的了。咱們須要從端的可用性延伸出下載鏈路、加載鏈路,使用鏈路上的可用性,針對每一個重要的環境,都作好監控運維。
3.1.1 MRN容器簡介
React Native框架自己只是一個運行時環境中的渲染引擎,能夠將同一套JS代碼分別在Android和iOS系統上最終以Native的方式渲染頁面,從而爲App提供了基礎的跨端能力。但從工程化的角度來看,若是想在App中大規模地應用RN技術,除了RN框架自己外,還須要在開發、構建、測試、部署、運維等諸多方面的配合。MRN(Meituan React Native)是美團基於React Native框架改造並完善而成的一套動態化方案,在RN的基礎上提供了容器化能力、動態化能力、多端複用能力和工程化保障。MRN在開發效率、穩定性、性能體驗、動態化和監控運維等多方面進行了能力升級和擴展,知足了美團RN開發工程化的須要。目前,MRN已接入美團40多個App,核心框架及生態工具備超過100位內部代碼貢獻者,總PV超過4億。
3.1.2 Roo組件庫
下面再介紹一下外賣建設的兩個UI相關的技術項目,Roo組件庫和組件樣式動態配置。
外賣在2018年末開始試驗MRN容器在外賣業務上的應用,並在2019年上半年進行了大面積的頁面落地。目前,外賣已有近60個RN頁面上線,佔外賣頁面比例超80%,其中包括Tab頁面「個人」、提單選擇紅包頁、訂單評價頁等高PV頁面。MRN容器的接入,給外賣App的容器化、動態化、人效提高、包大小瘦身等方面都作出了不小的貢獻。
3.2.1 Titans容器簡介
Titans容器是美團系App統一的Web容器組件,基於蘋果提供的WebView組件,將WebView容器化,統一了WebView的UI展現和交互方式,規範了橋協議的使用範式,同時預置了諸多基礎能力和業務能力。Titans容器大大提升了Web頁面的開發效率和用戶體驗上的一致性。
Titans容器在外賣業務中的使用場景很是豐富,其中最重要的使用場景是各類運營頁和活動頁,例如點擊首頁頂部Banner的廣告落地頁、爲你優選、限時秒殺等活動運營頁等;還有客服頁、幫助反饋頁、商家入駐頁、美團公益頁等功能性頁面;做爲一級入口頁面的美團會員頁面,也是一個基於Enlight的Titans容器。
外賣容器化建設,首先須要要區分的是核心頁面和非核心頁面。外賣業務中對核心頁面的定義是頁面DAU>美團DAU的5%或者是下單關鍵路徑。爲何要先按照是否爲核心頁面進行拆分呢?重點就在於改造的成本。核心頁面的業務複雜度決定了它不容易作全頁面的動態化,它比較適合作局部的動態化方案。核心頁面的複雜度在於業務自己複雜,最重要的是核心頁面每每會有多個垂直業務團隊共同的開發維護,你們各自有重點關注的模塊,作全頁面的動態化,沒法作到有效的物理隔離。
而對於非核心頁面,業務功能和交互相對簡單,組織關係也較爲肯定,更適合作標準的MRN和Titans容器化。因此咱們的策略是核心頁面作到支撐頁面模塊級別的業務動態和複用,非核心頁面能夠作到頁面級別的動態化和複用。頁面容器化的核心含義就是把一個頁面劃分爲若干個模塊,每一個模塊成爲一個業務容器,每一個容器的填充既能夠用Native的方式實現,也能夠用Mach實現(Mach是外賣自研的頁面局部動態化技術),能夠支持iOS/Android/小程序三端跨平臺運行。頁面自己則化身爲容器的管理者,負責子容器的編排和佈局,並支持其動態化。
頁面容器化設計中主要分爲三個階段,模塊有序化、模塊編排化、漸進式業務落地。
從App頁面開發的角度看,一個完整的頁面能夠按照不一樣的功能及不一樣業務屬性劃分出多個不一樣的模塊。
業務構建泛指由多個業務模塊組合拼裝爲一個業務頁面的過程,涉及頁面自己(UIViewController/Activity)以及各個業務模塊的構造過程,先後端業務數據以及頁面和業務模塊之間的數據交互過程,業務模塊內部的數據處理以及視圖刷新流程。
模塊標準化指的將業務構建涉及到的多個過程經過規範化的方式肯定下來,造成惟一的標準。模塊標準化一方面可以在解決業務共性問題的基礎上提供業務難點專項解決方案,另外一方面可以在框架基礎上造成能力約束,減小重複建設、低質量建設的問題。
業務構建模塊標準化中咱們抽象了四層,下面將分別進行解讀。
經過業務構建模塊標準化的建設,業務模塊已是標準化的了,能夠在跨頁面間自由組合,這爲頁面容器化打下了基礎。
在頁面容器化中最基礎的能力有如下幾點:頁面中業務模塊可編排能力,動態上線前端AB實驗的能力,增量上線動態模塊的能力。實現這些能力最重要的就是進行先後端數據協議標準化建設。客戶端根據數據協議中的模塊惟一標識匹配並構造業務模塊,在完成模塊數據的填充後會根據數據協議中的模塊佈局信息完成模塊的佈局。針對Mach動態模塊,咱們建立了基於模板ID的模塊匹配和數據填充流程,能夠支持Mach動態模板的增量上線。在數據協議中針對前端AB實驗咱們預留了AB實驗和通參字段,在數據填充階段經過容器化接口傳入動態模塊中,用於支持AB實驗的動態上線。
在容器化上線的過程當中屬於接口的大版本升級,爲了保證容器的高可用性,客戶端從模塊級別和API級別實現了兩套降級容災方案。
模塊級別的降級方案主要針對Mach動態模塊,與Native模塊不一樣,Mach動態模塊須要預先下載動態模板才能正常地完成模塊的載入和渲染。爲了保證動態模塊的加載成功率,咱們一方面在接口上線前利用Eva(美團內部系統)對Mach模板的下載進行預熱。另外一方面,咱們設計了動態模塊的主動降級方案,針對動態模塊的動態上線使用Native模塊進行兜底降級,對於跟版動態模塊使用App內置模板的方案進行兜底降級。
API級別的容災方案主要爲了保障客戶端在新接口不穩定的狀況下能夠自行降級到舊接口。針對這個問題,咱們對線上老接口設計了數據結構映射方案,在客戶端經過配置化的方式能夠把老接口的數據結構映射爲新接口的數據結構。這樣在上層業務無感知的狀況下,能夠作到容災方案的上下線。
經過頁面容器化,使得頁面只須要關心頁面級的構造方式,而無需關心某一模塊內部如何實現動態化的。把頁面與頁面的模塊分離,也符合目前外賣客戶端的組織結構,有利於業務組間的協做。同時,頁面容器化使得外賣核心頁面具有了符合外賣業務場景下的動態能力,漸進式把Native靜態模塊過渡到具有動態能力的模塊,從模塊的維度使整個頁面具有了動態能力。這種漸進式的遷移方案把容器遷移跟業務模塊的遷移分隔開,大大下降了頁面容器化改造的風險。
質量和性能指標是衡量咱們App開發質量和用戶體驗的重要依據,是咱們一直都很是關注的重點數據。在非容器化時代,咱們大多數的指標都和App的使用環節緊密相關,由於在非容器化時代,邏輯鏈路相對簡單,例如咱們打開一個新頁面時,咱們首先建立頁面實例,而後發起網絡請求,同時頁面會經歷一系列生命週期方法,最後渲染。這時咱們可能會關注網絡請求的成功率和請求時間,頁面的渲染時間,和過成功是否發生Crash,這個過程相對更短暫,指標更少,因此監控起來也更容易。
外賣的容器化大大提高了外賣業務的複用能力、動態能力、模塊化和開發效率,但同時也帶來了更長的邏輯鏈路,鏈路從時間維度上劃分是:下載鏈路、加載鏈路、使用鏈路。例如咱們在使用MRN容器的時候,會涉及到bundle的啓動下載或預熱下載,bundle解壓縮,MRN容器引擎初始化,bundle加載,JS的加載、執行,頁面渲染等步驟,其中的每一個步驟均可能存在性能問題和質量風險。所以,咱們須要升級咱們的衡量指標系統來應對容器化帶來的新的挑戰。
由於容器化的使用造成了一個串行的鏈路,因此若是某個關鍵節點失敗,會致使容器功能不可以使用,關鍵指標的任務就是從上述衆多的指標當中篩選出這些關鍵節點。例如在下載鏈路中bundle下載的成功率和API的成功率,加載鏈路中bundle加載的成功率和模塊匹配的成功率,下載或加載失敗都沒法再進行鏈路中的後續步驟,針對上面的成功率指標,咱們會添加分鐘級別的實時監控告警,作到及時發現,快速響應和緊急修復。
在使用鏈路中模塊渲染的成功率、Native Crash率、JS錯誤率也屬於關鍵指標,這些任務的失敗也會致使容器的不可用,針對這些指標咱們也會採用實時監控措施,而且添加降級手段,例如回滾bundle版本,或者把MRN和Mach容器降級爲Native容器。
上面講到了容器化架構的各項衡量指標,那麼把這些指標具體落到實處的工做就是線上的運維監控工做。工欲善其事,必先利其器,對於監控運維工做,必定要有合適的監控工具輔助配合才能事半功倍,公司內有不少優秀的監控統計工具可供使用,這裏的難點就是如何根據監控的須要判斷選擇合適的工具。還有就是合理的劃分監控維度和數據指標的優先級,例如對於可以影響到鏈路穩定性的關鍵指標,咱們須要作到分鐘級的監控,一旦出現問題就能及時收到告警,對於非關鍵指標,則經過生成日報的方式,方便開發者的統計和分析。
工具的使用上主要分爲大盤工具、具體異常工具、灰度降級工具、告警工具等(如下是美團內部使用的工具)。
業務覆蓋維度監控能夠分爲全局監控和局部(單業務)監控。
時間維度監控:能夠按天、小時、分鐘的時間維度。天級別的監控主要是一些非關鍵路徑指標,例如一些性能指標,頁面加載時間、頁面FPS、JS渲染時間等,咱們能夠按天維度的生成數據報表,已郵件的數據發送日報。當App灰度上線時,咱們會開始小時級別的監控,每過半小時經過IM軟件向廣播一些關鍵指標,方便開發者跟蹤線上數據的穩定性。分鐘級別的監控則是針對關鍵指標,觀察分鐘維度上的變化,若是關鍵指標超過閾值,或者波動過大,就會及時產生告警。
下面咱們以一個開發者的視角去看一下外賣容器化架構的監控運維繫統。從獲取信息的方式上能夠分爲主動查詢和被動推送,開發者能夠經過監控工具監控全局和局部數據的變化趨勢,也能夠分析具體異常Case;也能夠從IM工具,郵件等收到相關的推送數據,以便及時響應。在控制運維上,開發者能夠經過Eva、Horn等美團內部的灰度系統進行灰度發佈,當灰度期發現問題的時候,能夠及時地經過中止灰度,版本回滾,關閉入口的方式進行降級容災處理。
容器化使外賣業務具有了強大的動態化能力,但動態化能力又和須要相應的發佈能力來支持,發佈能力是咱們業務開發質量和效率的重要保障,也是咱們容器化建設工做過程當中的重點環節,這一節主要介紹一下外賣容器化的發佈能力。
從發佈能力類型的角度看主要能夠分爲三種類型:(1)容器內容的發佈,包括髮布整個頁面或者發佈頁面中的局部模塊;(2)配置下發,經過API或其餘配置平臺,下發佈局協議、AB測試、樣式配置、功能配置、模板配置、容器配置等,大大提升了業務的靈活度和線上驗證能力;(3)灰度、降級下發,經過UUID,用戶畫像等信息作到灰度發佈,降級回滾等控制能力。
從發佈資源的的角度看主要分爲兩種:一種是普通的資源,例如發佈一個Web頁面,或者經過發佈新版API來控制頁面局部容器的展現與否和展現的位置,同時咱們也能夠進行一些AB Test操做;另外一種是bundle資源,主要是針對MRN容器和Mach容器,每一個MRN容器和Mach容器的資源都會先被打包成一個bundle,而後經過發佈系統下發到終端,而後終端解析bundle中的代碼和資源,最終渲染頁面。
從發佈階段的角度看,能夠分爲測試階段、上線階段、灰度階段和全量階段,其中上線階段是最終的環節,咱們增長了不少校驗和保護手段來儘可能保證上線操做的正確性。
雖然咱們具隨時備動態發佈能力,但正常的版本迭代仍是會存在中,因此外賣這邊的節奏是周動態迭代+雙週版本迭代,這保證了咱們的開發工做有個一清晰的週期。在動態發佈階段中最關鍵的階段操做上線階段。以MRN爲例,目前外賣業務中應有70多個bundle,再算上測試環境的bundle就有接近150個bundle,只是管理這些bundle就是一個複雜的工做,何況在進行上線操做時仍是涉及發佈的目標App、App版本的上下界、MRN版本的上下界等,一不當心就會形成操做失誤,因此進行上線操做時須要很是謹慎。
咱們針對操做上線階段進行了事務流水線,經過流水線創建保護措施,一個bundle的上線要經歷一個流水線的若干操做。首先,操做人根據上線SOP手冊進行若干檢查操做,同時編寫標準格式的發佈說明,而後周知相關核心人員後在操做系統上發起上線申請,Leader和QA收到申請後會進行檢查並審批,審批經過後還要避開App使用的高峯期或節假日上線,上線後經過灰度發佈觀察各項數據指標,指標正常後全量發佈。
bundle是咱們最常發佈的資源類型,這裏再結合發佈工具講解一下bundle的發佈過程。MRN和Mach都是以bundle的形式下發到設備終端的,咱們在發佈bundle的時候主要會用到兩個工具,打包工具Talos和發佈工具Eva(美團內部工具)。一個bundle的工程文件主要由三個部分組成:配置文件、源代碼和資源文件,其中配置文件用於指導Talos對工程文件進行打包,多個bundle能夠共享一份配置文件。當咱們準備發佈一個bundle時,先找到該bundle在Talos的發佈模板,選擇發佈環境(測試或線上),而後進行一鍵打包,而後Talos會進行一系列流水線操做,包括Clone代碼、配置環境、進行Lint檢查、構建和上傳等。Talos打包完畢後將bundle上傳到Eva系統,而後Eva負責bundle的分包、上線、下線、灰度等操做,最終下發到終端設備上。
將來,咱們還將引入美團住宿的MRN-DevOps來進一步的屏蔽當前多系統的問題,下降整個週期管理的成本,特別是發佈前的人工檢查成本,逐漸實現RD在一個平臺上操做從研發到發佈運維的全部實現。儘量地減小人工成本,提高自動化。
上面介紹的是以bundle資源形式的發佈流程,過程較爲清晰簡單。下面再結合外賣首頁,介紹一下局部容器化的發佈方式。外賣首頁是典型的流式列表,在局部容器化的架構下,首頁就是由一個個矩形容器以ListView方式佈局的,容器分兩種,Native容器和Mach容器,Mach容器是一個通用容器,咱們能夠編寫不一樣的樣式模板,下發到終端後交由通用Mach容器來渲染,以此達到只使用通用容器展現不一樣UI樣式的目的,這裏涉及了Mach的發佈系統。
首頁各子容器至關於一塊塊積木,它們的位置排布、展現與否、模板的選擇等最終交由API控制,API具有了控制首頁佈局,樣式展現的能力,而再也不是單純的數據源。同時,首頁也涉及了AB能力、灰度降級策略等其實配置項下發系統。能夠看到外賣首頁的容器化是由多種發佈能力配合支撐的,是外賣發佈能力體系的「集大成者」。
好的架構是要隨着業務的發展,不斷演變去適應業務的發展。美團外賣從一個很小規模,每日單量只有幾千的業務,逐漸地走到今天,每日單量峯值超過4000萬,組織架構也從一個十幾我的的團隊,逐漸發展到如今多角色、多垂直業務方向,上千人共同協做的團隊。移動端上的架構,爲了適應業務的發展要求,也經歷了組件化、平臺化、RN混合化,再到如今向容器化的變遷。
容器化架構相對於傳統的移動端架構而言,充分地利用瞭如今的跨端技術,將動態化的能力最大化的賦予業務。經過動態化,帶來業務迭代週期縮短、編譯的加速、開發效率的提高等好處。同時,也解決了咱們面臨着的多端複用、平臺能力、平臺支撐、單頁面多業務團隊、業務動態訴求強等業務問題。
固然,容器化架構帶來好處的同時,對線上的可用性、容器的可用性、支撐業務的線上發佈上提出了更加嚴格的要求。咱們經過監控下載、加載、使用鏈路上的可用性,來保障線上動態業務的可用性。針對容器,咱們利用成熟的測試基建,建設容器的自動化測試來保障容器的可用性。針對發佈,咱們建設迭代流程,配合發佈流水線,將線上的發佈變得更爲可控。
截止到目前爲止,外賣業務通過了幾十個動態化業務上線窗口,累積共發版百次以上。將來半年,咱們還將進一步從業務需求入手,將業務需求細分歸類,讓產品側逐漸創建容器和動態化需求的概念,可以從源頭上,逐漸的將業務進行劃分,最終使得每一個業務需求,均可以歸類抽象成能夠動態下發的業務和容器能力建設,從而進一步的完善容器化架構的能力和支持更多的的業務場景。
美團外賣長期招聘Android、iOS、FE高級/資深工程師和技術專家,歡迎有興趣的同窗投遞簡歷到wangxiaofei03@meituan.com。
想閱讀更多技術文章,請關注美團技術團隊(meituantech)官方微信公衆號。