1. 程式人生 > 程式設計 >SpringBoot專案執行指令碼 自動拉取最新程式碼並重啟的例項內容

SpringBoot專案執行指令碼 自動拉取最新程式碼並重啟的例項內容

gitPullThenRestart.sh

# 日期: 20191230
# 作者: 何鵬舉
# 說明: 專案部署在阿里雲上,每次編譯打包上傳比較麻煩,因此編寫此指令碼,用於應用內可以點選按鈕進行自動重啟.
# 備註: 1) 阿里雲的外掛也是需要每次上傳到OSS再啟動,上傳的jar包比較大(大部分是第三方依賴),感覺不好
#    2) 使用jenkins,就一個專案再去搭建一套jenkins,必要性也不是很大

# 以下程式碼中 /root/hekele 為git clone的專案路徑,/root/app 為部署的路徑


# 重新整理環境變數
source ~/.bashrc

# 切換到git專案目錄拉取最新程式碼,並進行mvn的打包
cd /root/hekele
git pull

cd /root/hekele/java
mvn clean install


# 殺掉當前專案的程序
ps -ef|grep hekele.jar | awk '{print $2}' | while read pid
do
 kill -9 $pid
done

# 複製jar包,強制覆蓋(忽略提醒),由於CentOS的.bashrc中開啟cp的-i選項,即可使用-f也不行,所以還是先刪除再複製
rm -f /root/app/hekele.jar
cp /root/hekele/java/target/hekele.jar /root/app/

# 後臺啟動專案
nohup java -jar /root/app/hekele.jar >> /root/app/hekele.log 2>&1 &

# 檢視日誌(備註: kill掉之後,其複製jar包和啟動還是執行了,原因暫時未知... 此句話在手動啟動可用,自動重啟實測也沒影響)
tail -f /root/app/hekele.log

JavaController

@ApiOperation("拉取最新程式碼然後重啟應用")
  @GetMapping("gitPullThenRestart")
  public R gitPullThenRestart() {
    // 必須開啟一個新執行緒,從而返回給瀏覽器,否則瀏覽器得不到響應,還會再次傳送
    new Thread(() -> ExecUtil.execCommand(restartShell)).start();
    return R.ok().setErrmsg("正在重新啟動,請稍等");
  }

JavaExecUtil

/**
 * 執行系統命令
 * 
 * 
 * <br> 注意死鎖問題
 * <br> 解決: 只要主程序在waitfor之前,能不斷處理緩衝區中的資料就可以
 * 
 * @see <a href="https://blog.csdn.net/seapeak007/article/details/69668600" rel="external nofollow" >呼叫Process.waitfor導致的程序掛起</a>
 * 
 * @author he_pe
 *
 */
@Slf4j
public class ExecUtil {

  public static void execCommand(String command) {
    log.info("begin exec os command: " + command);
    Process process;
    try {
      process = Runtime.getRuntime().exec(command);
      
      //開啟新執行緒處理正常輸出 和 錯誤輸出
      newThreadLogInputStream(process.getInputStream());
      newThreadLogInputStream(process.getErrorStream());

      // 阻塞當前程序,直到命令結束
      process.waitFor();

      // 不會阻塞程序,但是呼叫時如果沒有完成會報錯
      if (process.exitValue() != 0) {
        log.error("exec os command failure: " + command);
      } else {
        log.info("exec os command success: " + command);
      }
    } catch (Exception e) {
      log.error(e.getMessage(),e);
    }
  }

  private static void newThreadLogInputStream(InputStream is){
    new Thread(() -> {
      try(InputStream inputStream = is;
        BufferedReader bufr = new BufferedReader(new InputStreamReader(inputStream))){
        String out = null;
        while ((out = bufr.readLine()) != null) {
          log.info(out);
        }
      } catch (IOException e) {
      }
    }).start();
  }

}

以上就是本知識點的全部程式碼內容,感謝大家的學習和對我們的支援。