跳到主要內容

關於ASP.NET core的IHostedService的schedule task不會自動執行的問題

最近在測試ASP.NET core上的IHostedService製作的schedule task ,發現了一個tricky的問題。

       問題描述如下:

Startup.cs中的ConfigureServices() 裡我們會加上要重複執行的schedule task (IHostedService),預期網站運行後開始背景執行這些task,但實際在IIS上測試發現兩個問題:

 

1.      schedule task並不會自動執行,除非對網站進行訪問(任何http request),例如開啟首頁。這時那些在ConfigureServices()加入的task才會被建立並且開始運作。

 

2.      如果一直都沒有對網站進行訪問(任何http request),約20分鐘後所有task都會停止,直到下一次網站被訪問(任何http request)才會又開始所有task  

 

基於以上兩點行為,針對第一點,我們的網站必須要每次主機開機都開一次首頁或是任何對網站的request,否則schedule task不會執行。針對第二點,我們需要確保固定時間對網站進行訪問(任何http request),目前的作法是每次task執行時就發個request(如果schedule task本來就會發API request就沒問題)


ASP.NET Core的IHostedService運作行為:

這個問題目前的了解是:以IHostedService製作的schedule task,它的生命週期開始在Web server接收到任何一個http request的時候,而IIS運行Web server時並不會觸發http request,所以schedule task也就不會啟動。再來另一個狀況是,IIS有個預設的idle timeout機制,即使schedule task啟動了,如果一直都沒有對網站進行訪問(任何http request),20分鐘後會將Web server的http實體程序關閉,所以schedule task也就停止了。

非透過IIS的解決方式:

    在Server開機後透過固定開啟網站首頁的方式,讓schedule task啟動,並加入一個schedule task固定每10分鐘對Web server發送一個http request,以確保IIS不會將schedule task停止。

透過IIS的解決方式:

    在IIS管理介面上可以設定,將Web server的行為設成每次server運行時就載入主體程序。經測試可以解決IHostedService製作的schedule task不會自行啟動的問題,另外即使IIS預設的idle timeout時間到了,也會自動重啟新的主體程序,schedule task不會停止。

設定步驟如下:

[設定自動背景執行]

[控制台]-->[程式和功能]-->[開啟或關閉windows功能] à打勾 應用程式初始化

 

如下圖,在IIS管理員中進入[進階設定]

[預先載入已啟用]設為True

 

[應用程式集區],的[進階設定]

 

如下圖,將[啟動模式]設成 AlwaysRunning

 

至此完成設定


留言

這個網誌中的熱門文章

[Python] 管理多個執行緒(thread)的運作與結束

在開發應用程式中,幾乎所有的專案都會用到thread,不同的程式語言有不同˙的叫用方法,但都大同小異。Thread的使用上最重要的不是在產生一個thread,而是如何結束一個thread,或是如何結束好幾個threads,以下介紹在Python中如何叫用並管理一或多個thread: 一般來說,create thread如下: def job(): while 1: print("Child thread:", i) time.sleep(1) #In main code thread = threading.Thread.__init__(job) thread.start() 這樣可以開始一個thread執行job()中的動作,但若job中的迴圈是如上範例的無限迴圈,必須要加上能讓其退出的機制: stopped = 0 def job(): global stopped while stopped == 0: print("Child thread:", i) time.sleep(1) #In main code thread = threading.Thread.__init__(job) thread.start() 這樣在主程序先將stopped設成1,再調用 thread.join() 就可以將thread完整關閉並釋放記憶體空間。 多執行續的處理: 基於這樣的方式,下面是進化版本,創建一個thread專用的class,並同時處理多個thread的執行與結束: class EventHandleThread (threading.Thread): def __init__(self, fun, name): threading.Thread.__init__(self) self.name = name self.fun = fun self.stopped = [0] def run(self): self.fun(self.name, self.stopped) print ("Exiting " ...

[Python] 將程式設定為全螢幕顯示(full screen)和永遠在最前景、最上層(topmost)

這裡以創建一個tkinter 的 Application物件為例 EX: app = Application() app.master.title("WaferCheckApp") app.master.geometry("1100x720") 將UI顯示在最上層: app.master.attributes('-topmost', True) 將UI以全螢幕顯示: app.master.attributes('-fullscreen', True)

[Python] 自己製作文字設定檔config file,動態調整程式設定

在軟體開發中,通常不會知道實際使用行為有那些與原先預期不一樣的情況,這時如果還要針對特定的不同需求改程式是很吃力不討好的事情。所以我們常常會需要能夠動態調整一些設定,以應付不同的使用情況。 下面以Python提一個簡單的方法來動態調整程式設定: 建立一個function,在程式初始化時呼叫: EX: def initialSetting():         try:             currentDirectory = os.getcwd()             filePath = '%s\\ ToolSet.cfg ' % (currentDirectory)             file = open(filePath, 'r')             for line in file.readlines():                 if line.find("Value1") == 0:                     str1 = int(line[7:-1])                 elif line.find(" Value2 ") == 0:                      str2  = int(line[7:-1])                 elif line.find(" Value3 ")...