cygwin daemon

前一段時間遇到一個問題,最近才解決,主要也是對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 " ")
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章