Function Compute在處理使用者請求時,自動分配一個或多個執行個體,每個執行個體提供一個安全和隔離的運行時環境。傳統應用遷移至Serverless架構時,由於函數執行個體的建立和銷毀具有瞬態性,可能導致監控資料更新不及時、指標資料延遲或丟失等問題,為瞭解決以上痛點,Function Compute基於執行個體生命週期增加多種回調操作,確保監控資料的即時性和完整性。
函數執行個體的生命週期
函數執行個體會根據函數當前的請求調用量動態地按需構建與銷毀,每個函數執行個體的生命週期包括執行個體構建(Creating)、請求調用(Invoke)和執行個體銷毀(Destroy)三個階段,如下圖所示。
執行個體構建(Creating)
執行個體構建是指Function Compute根據您的函數配置為您建立函數執行個體。在執行個體構建階段,Function Compute會按照順序執行以下三項任務:
執行個體建立(Instance Create),包括載入代碼、層載入或拉取鏡像和啟動執行個體。
運行時啟動(Runtime Init)。
執行函數配置的Initializer回調(Init Hook)。更多資訊,請參見Initializer回調。
執行個體構建階段示意圖如下所示。
執行個體構建一般在以下兩種情況下發生。
彈性擴容
收到調用請求時,若當前的函數執行個體已經滿載,會構建新的執行個體來處理請求,並立即執行執行個體構建流程,緊接著就會執行請求調用流程。彈性擴容可能會造成冷啟動,解決方案請參見Function Compute冷啟動最佳化最佳實務。
預留配置調整
如果您配置了預留執行個體,對於這些預留執行個體,Function Compute立即啟動執行個體構建流程。如果當前未收到調用請求,則後續的請求調用流程會與執行個體構建流程相隔較長時間。更多資訊,請參見配置預留執行個體。
對於一個函數執行個體,函數配置的Initialize回調方法只會執行一次,如果執行失敗,會重新啟動一個函數執行個體執行Initializer流程,Initializer回調失敗的執行個體會直接銷毀。
請求調用(Invoke)
執行個體運行期間,會調用您的函數處理常式以處理來自內外部的函數調用請求。在調用階段,對於Function Compute支援的內建運行時,一個執行個體在同一時間只會處理一個請求;對於自訂運行時或自訂容器運行時,一個執行個體在同一時間可以處理多個請求。您可以通過設定單一實例多並發實現,具體操作,請參見建立Web函數。
Function Compute只在實際請求和回調程式執行時計費,在請求以外的時間段內執行個體會被冷凍,因此不計費,詳情請參見計費說明。
執行個體銷毀(Destroy)
如果函數執行個體在一段時間內沒有收到任何調用,則觸發此階段。在銷毀階段,Function Compute會先執行PreStop回調方法。您可以在PreStop回調方法中執行一些清理任務。
執行個體銷毀一般在以下三種情況下發生。
執行個體閑置:如果執行個體在一段時間內沒有收到任何調用請求,Function Compute會自動回收該執行個體。
預留配置調整:當您縮減預留執行個體數量時,Function Compute會立即為您銷毀多餘的執行個體。
執行個體異常:如果執行個體在構建或運行階段出現了異常,Function Compute會銷毀該執行個體。
執行個體凍結機制
在沒有調用請求時,Function Compute會將執行個體凍結(Freeze),當新的請求來到時,Function Compute會將執行個體解凍(Thaw)。如下圖所示。
執行個體凍結主要發生在以下兩種情況。
執行個體初始化階段完成後,調用階段前。
函數某一次調用階段結束後,下一次調用階段前。
在一次調用階段完成後,Function Compute將凍結函數執行個體,程式中的後台進程、線程或協程無法繼續運行,非同步日誌也可能沒有寫入成功。
使用限制
所有運行時均支援Initialize和PreStop回調方法。
PreStop回調方法的輸入參數沒有event參數。
PreStop回呼函數無傳回值,在函數末尾增加返回邏輯無效。
如果使用Java Runtime,您需要將fc-java-core更新至1.4.0及以上版本,否則無法使用PreStop擴充回呼函數。
當函數執行返回時,Function Compute將凍結函數執行個體,使用者不可假設調用返回時所有非同步進程、線程、協程等執行完成,也不可假設本次非同步寫入的日誌被重新整理。
計費說明
執行個體生命週期回調方法的請求數不計費,其他費用與執行個體調用階段的計費邏輯相同。計費時間長度的統計如下圖所示。關於計費方式的具體資訊,請參見計費概述。
前提條件
已完成函數的建立,具體請參見建立函數。
配置執行個體生命週期回調方法
通過控制台配置回調
當您使用控制台建立函數時,Function Compute不支援配置PreStop回調,您需要在更新函數時配置該回呼函數。
登入Function Compute控制台,在左側導覽列,單擊函數。
在頂部功能表列,選擇地區,然後在函數頁面,單擊目標函數。
在函數詳情頁,選擇配置頁簽,在左側導覽列,選擇生命週期,然後單擊編輯。
在編輯生命週期面板,設定回調程式和回調逾時時間,然後單擊部署。
說明每一個擴充函數都需要配置單獨的回調程式和逾時時間,其中回調程式格式為[檔案名稱].[擴充函數名]。例如在Python Runtime中,建立函數時指定的PreStop回調為index.preStop,那麼檔案名稱為index.py,PreStop函數名為preStop。
配置回呼函數後,您需要在代碼執行中實現對應的函數。
單擊代碼頁簽,在代碼編輯地區,輸入擴充函數代碼。
例如,您配置的PreStop回調程式為
index.preStop
,則需要實現preStop函數。不同語言運行時實現函數執行個體生命週期回調的方法請參見函數執行個體生命週期回調方法。說明線上IDE支援PHP、Python、Node.js和自訂運行時;但不支援Java、Go和.NET這類編譯性語言,以及自訂鏡像。
單擊代碼編輯器上方的部署代碼,然後單擊測試函數。
通過Serverless Devs配置回調
使用Serverless Devs配置PreStop擴充函數時,s.yaml
設定檔範例程式碼片段如下所示:
codeUri: './code.zip'
......
instanceLifecycleConfig:
preStop:
handler: index.PreStop
timeout: 60
如果您需要關閉某個擴充函數,需要將擴充函數的handler
參數顯示置空,否則後端預設不更新。例如關閉PreStop函數,您需要按照以下配置進行部署更新,此時PreStop函數的timeout
參數已無效。
codeUri: './code.zip'
......
instanceLifecycleConfig:
preStop:
handler: ""
timeout: 60
不同語言運行時實現函數執行個體生命週期回調的方法請參見函數執行個體生命週期回調方法。
通過SDK配置回調
您可以通過SDK部署和更新擴充函數。本文介紹如何擷取在建立函數時配置preStop函數的SDK範例程式碼。
進入CreateFunction - 建立函數頁面,單擊調試,進入OpenAPI門戶。
在參數配置頁簽,根據需要建立函數的基本資料填寫輸入參數。
參數配置完成後,單擊SDK 樣本頁簽,擷取對應語言的SDK範例程式碼。
不同語言運行時實現函數執行個體生命週期回調的方法請參見函數執行個體生命週期回調方法。
函數執行個體生命週期回調方法
Function Compute中所有運行時均支援Initialize和PreStop兩種回調方法。運行時實現函數執行個體生命週期回調的方法請參考以下內容。
運行時 | 描述 | 參考文檔 |
Node.js | 通過Node.js實現並應用函數執行個體生命週期回調方法。 | |
Python | 通過Python實現並應用函數執行個體生命週期回調方法。 | |
PHP | 通過PHP實現並應用函數執行個體生命週期回調方法。 | |
Java | 通過Java運行時實現函數執行個體生命週期回調的方法。 | |
C# | 通過C#運行時實現函數執行個體生命週期回調的方法。 | |
Go | 通過Go實現函數執行個體生命週期回調的方法。 | |
自訂運行時 | 通過自訂運行時實現函數執行個體生命週期回調的方法。 | |
自訂鏡像 | 通過自訂鏡像運行時實現函數執行個體生命週期回調的方法。 |
查詢回呼函數相關日誌
配置函數執行個體生命週期回調並執行代碼實現對應的回呼函數後,您可以查詢執行個體生命週期回呼函數的相關日誌。
登入Function Compute控制台,在左側導覽列,單擊函數。
在頂部功能表列,選擇地區,然後在函數頁面,單擊目標函數。
在函數詳情頁,選擇日誌頁簽,然後在調用請求頁簽,單擊目標請求行右側操作列的進階日誌。
您可以使用複製的執行個體ID,查詢所有生命週期回呼函數的Start/End日誌;還可以使用
執行個體ID+函數執行個體生命週期回調關鍵字
查詢指定回呼函數的Start/End日誌,例如,c-62833f38-20f1629801fa4bd***** and PreStop
。此外,您還可以根據Start/End日誌中的RequestId查詢請求的日誌資訊。如果使用者日誌中沒有RequestId,可以單擊該日誌中的表徵圖擷取上下文日誌。