前一段時間遇到一個問題,最近才解決,主要也是對cygwin不夠了解。
問題描述
服務器端安裝了cygwin的sshd服務器,想在git push后時執行一個時間比較長的腳本,當然不想在前臺執行,所以采用后臺執行,結果,當然是不行,執行一半就被強制關閉了。
問題分析
之前一直不明白為什么,后來想了想Linux上的東西,總算明白了,cygwin調用的也是bash,shell退出后,shell運行的程序就退出了。除非運行的是daemon
守護進程
Linux下弄daemon,網上有很多,但是windows下機制不一樣啊,沒有什么init進程來接管無父進程的子進程。其實想想windows下的服務就是守護進程,再一搜,cygwin果然還是提供這種機制的。
cygrunsrv 可以將cygwin的程序安裝成windows的服務,服務的stdout和stderr都被重定向到/var/log/<servicename>.log文件,cygwin的具體用法參見幫助即可。
問題解決
編一個程序,每次接收到SIGUSR1時,啟動運行一個shell腳本,做我想做的事情。然后在git repo的post-update鉤子中,用kill向daemon發送消息
- signal處理類似標準linux,包含signal.h,然后用signal函數安裝一個回調函數,注意事項也和Linux中一樣,比如不要在signal函數中做IO操作之類的。可以這樣做,主循環平時pause(),收到signal時,置個標志位,主循環檢測到標志位后啟動shell腳本
- 啟動shell腳本關系到新建一個進程,linux中是fork,但是cygwin中使用fork太慢(官方文檔說的),推薦用spawn(),這個函數的頭文件在process.h(找了半天),用法參見MSDN,因為這實際是Windows的一個函數
- cygrunsrv默認采用SYSTEM用戶來安裝daemon,所以運行時會和直接在cygwin中有差別,主要是環境變量,因此可以在安裝daemon時指定 --user <user> --password <password>,然后調用shell時再采用一個wrapper腳本,執行 bash --login <the-script-name-you-really-want-run>
-
啟動腳本后如果什么也不管,那腳本運行完后,會產生一個<defunct>的僵尸進程,如果實在不想知道啟動的腳本的運行結果,可以直接忽略SIGCLD,將其回調函數設置為SIG_IGN,如果想了解腳本的返回值,就用waitpid或wait來獲得返回值,這和linux下是一樣的,不過我這次是直接忽略的SIGCLD,沒有試waitpid
-
post-update鉤子中,kill如何找到daemon呢,cygwin又提供了一個ps程序可以找到daemon的pid,命令如下: kill -s SIGUSR1 $(ps -a | grep <daemon-name> | cut -f1 -d " ")
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
