Java web程序实现请求后重启服务动作

此方案只适合少数服务,是笔者在做自己的实验项目所采用,谨慎参考。

大致思路,Java执行system.exit()方法退出服务,使用定时任务结合命令、脚本、服务方式实现服务重启。

windows平台实现方式

1、schtasks命令加bat脚本方式实现,写上三个start.bat、stop.bat、restart.bat脚本,写好逻辑后使用schtasks命令延迟执行相关脚本即可。但不推荐此方式,因为此方式要针对命令的返回结果做相关的解析,很容易遇到环境问题无法处理。笔者就是bat脚本写的比较low,放弃了此方式。
2、使用windows service的方式实现,共分两个方向,第一个是使用windows SC 工具,执行SC 相关命令安装windows服务,这种方法也有一定的局限性,我就遇到过SC 工具未安装的情况,运行了安装脚本后无反应。第二个方向是使用第三方工具WinSW来实现windows服务的相关动作,主要使用此方法,兼容性更强,使用起来也顺手。此项目是github开源项目,地址: WinSW,使用方法可以参考链接: 参考博文,此中配置可能略有不足,参考github上的配置解释,配置自己想要的即可,另外推荐使用管理员打开cmd执行winsw命令,不要使用powershell执行winsw相关命令,我在执行时会遇到命令执行阻塞问题,可能是winsw的兼容有待提高。

Linux平台实现方式

使用atd服务,at命令结合脚本实现服务重启动作。需要注意,at命令依赖atd服务,如果没有安装,执行yum安装即可,atd服务的依赖源是crontab服务,所以在运行at命令时需要保证上述两个服务在运行中。

具体实现代码大概如下,仅供参考,相关变量和工具方法不公开,自己写吧。

package com.les.ls.controller;

import com.les.ls.config.ConstantsConfig;
import com.les.ls.utils.DateUtils;
import com.les.ls.utils.TerminalUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;

import javax.annotation.Resource;

@Slf4j
@CrossOrigin
@RestController
@RequestMapping("/system")
public class SystemOptionController {

    @Resource
    private ConstantsConfig constants;

    @GetMapping("/shutdown")
    public void shutdown() {
        System.exit(0);
    }

    /**
     * <p><font color="red">
     * 1. 检查 schtasks 创建的自动任务是否存在,不存在创建,存在则修改时间</br>
     * 2. schtasks /query /tn "my once"  查询任务是否存在</br>
     * 3. schtasks /create /tn "my once" /tr D:\WinSCP\WinSCP.exe /sc once /st 17:23:00 /sd 2022/10/31  创建任务</br>
     * 4. schtasks 执行远程主机的计划任务提示拒绝访问一般由两个因素:1.登录用户没有权限。2.组策略设置不得当。</br>
     * 5. 建议用administrator账户登录,并且更改本地组策略:
     * 运行gpedit.msc ——计算机配置——Windows设置——安全设置——本地策略 ——安全选项——用户账户控制:以管理员批准模式运行所有管理员——禁用</br>
     * </font></p>
     */
    @GetMapping("{version}/restart")
    public void restart(@PathVariable String version) {
        String os = TerminalUtils.getOSName();
        String[] futureDateTimes = DateUtils.getFutureDateTime(120L, DateUtils.yyyy_MM_dd_HH_mm_ss2).split(" ");
        String futureDate = futureDateTimes[0], futureTime = futureDateTimes[1];
        if (os.contains("windows")) {
            if ("v1".equals(version)) {
                //v1版本使用 schtasks命令的实现方式需要自己写bat脚本,并且确保权限问题。在自启动会有相当大的麻烦
                String queryTaskResult = TerminalUtils.execCommand("schtasks /query /tn \"" + constants.windowsRestartTaskName + "\"");
                if (queryTaskResult.contains("系统找不到")) {
                    //找不到任务,执行任务创建
                    String createTaskResult = TerminalUtils.execCommand("schtasks /create" +
                            " /tn \"" + constants.windowsRestartTaskName + "\"" +  // tn指定任务名称
                            " /tr " + constants.windowsRestartScriptName +            // tr指定任务执行内容
                            " /sc once " +                                       // sc执行策略,只执行一次
                            " /sd " + futureDate +                                 // sd执行时间
                            " /st " + futureTime);                                    // st执行日期
                    if (!createTaskResult.contains("成功创建计划任务")) {
                        log.error("任务创建失败!createTaskResult->{}", createTaskResult);
                    }
                } else {
                    //任务存在,修改任务执行时间
                    String changeTaskResult = TerminalUtils.execCommand("echo " + constants.windowsSystemPassword + "| " +
                            "schtasks /change " +
                            " /tn \"" + constants.windowsRestartTaskName + "\" " +
                            " /sd " + futureDate +
                            " /st " + futureTime);
                    if (!changeTaskResult.contains("成功")) {
                        log.error("任务修改失败!changeTaskResult->{}", changeTaskResult);
                    }
                }
                //关闭应用程序,剩下的交给任务脚本
                System.exit(0);
            } else if ("v2".equals(version)) {
                // v2版本使用WinSW 工具实现打包发布java程序为windows service服务,使用命令重启即可
                String checkResult = TerminalUtils.execCommand(constants.windowsCheckCmdForService);
                if (checkResult.contains("running")) {
                    TerminalUtils.execCommand(constants.windowsRestartCmdForService);
                } else {
                    log.error("not found service ! checkResult->{}", checkResult);
                }
            } else {
                log.error("not found version ! version->{}", version);
            }
        } else if (os.contains("linux")) {
            //检查 atd服务, 执行at命令
            String checkResult = TerminalUtils.execForLinux(null, "/bin/sh", "-c", "systemctl is-active atd");
            if (!"active".equals(checkResult)) {
                //启动atd服务
                TerminalUtils.execForLinux(null, "/bin/sh", "-c", "systemctl start atd");
                //再次检查
                checkResult = TerminalUtils.execForLinux(null, "/bin/sh", "-c", "systemctl is-active atd");
                if (!"active".equals(checkResult)) {
                    log.error("无法启动 atd 服务!");
                    return;
                }
            }
            String createTaskResult = TerminalUtils.execForLinux(null, "/bin/sh", "-c",
                    "at -f " + constants.linuxRestartScriptName + " now");
            if (createTaskResult.contains("error")) {
                log.error("执行失败!检查执行脚本是否存在!");
            }
        } else {
            log.error("系统未识别,无法处理!os->{}", os);
        }
    }

    public static void main(String[] args) throws Exception {

    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值