开发整理-Javaweb应用的系统升级功能

原创 2016年11月19日 08:10:46

问题描述

  web应用有一个功能菜单是系统升级,通过调用升级脚本,将新发布的war替换原来的tomcat的webapps下的应用,然后停掉tomcate,再重启tomcate。最初实现就是通过简单的用在web项目中通过Process调用这个upgrade.sh脚本文件的。
   但是这种实现方式出现了一个问题:javaweb应用是tomcate,作为upgrade.sh脚本的父进程,当这个upgrade.sh进程试图kill掉父进程并重启的时候,父进程被迫终止后,该子进程也无法继续执行。所以这种方式实现的升级功能总是失败。

解决办法

  启动一个小应用,来执行Shell,那么这个应用一直以RMI方式监听来自Java应用的命令,根据相应的命令执行脚本。这样的话,用这个java进程去调用Shell命令杀死tomcate,并重启,以此达到系统升级的目的就不会有问题了。

流程如下:

这里写图片描述

升级程序代码

注册两个命令一个完成系统升级,一个完成Javaweb应用重启,这两个命令的本质就是调用底层的shell脚本完成相应的功能。

import java.rmi.Naming;
import java.rmi.registry.LocateRegistry;

import org.apache.log4j.Logger;
import org.apache.log4j.PropertyConfigurator;

public class ToolkitListener {
    public static void main(String[] argv) {
        Logger logger = Logger.getLogger(ToolkitListener.class);
        try {
            //初始化log4j
            PropertyConfigurator.configure(ClassLoader.getSystemClassLoader().getResource("log4j.properties"));

            // 启动RMI注册服务,指定端口为1099 (1099为默认端口)
            LocateRegistry.createRegistry(1099);

            //重启命令
            CmdInterface restartCmd = new CmdOfRestart("restart");
            Naming.rebind("restartCmd", restartCmd);

            //系统升级命令
            CmdInterface upgradeCmd = new CmdOfUpgrade("upgradeCmd");
            Naming.rebind("upgradeCmd", upgradeCmd);

            logger.info("Sensor ToolkitListener is ready.");
        } catch (Exception e) {
            logger.error("Sensor ToolkitListener is failed. " , e);
        }
    }

}

RMI调用

在Javaweb应用中,通过RMI的方式调用对应的对象的脚本执行方法。

    public static boolean rmiExecute(String name, String var){
        if(name==null){
            return false;
        }

        //调用RMI的升级对象完成升级
        log.info("rmi object is "+name+",variable is:"+var);
        try {
            switch(name){
            case "upgradeCmd":
                CmdInterface cmdOfUpgrade = (CmdInterface) Naming.lookup("upgradeCmd");
                cmdOfUpgrade.setFileName(var);
                cmdOfUpgrade.excuteShell();
                return true;
            case "restartCmd":
                CmdInterface CmdOfRestart = (CmdInterface) Naming.lookup("restartCmd");
                CmdOfRestart.excuteShell();
                return true;
            default :
                return true;
            }

        } catch (MalformedURLException e) {
            log.error("RMI调用异常",e);;
        } catch (RemoteException e) {
            log.error("RMI调用异常",e);;
        } catch (NotBoundException e) {
            log.error("RMI调用异常",e);;
        }    

        return false;
    }

升级文件校验

java web应用提供系统升级功能,由用户上传升级包,然后调用脚本完成升级。这个过程中需要对升级包进行校验,当前系统中维护了当前系统版本号和升级补丁版本号,上传升级文件中必须提供升级版本的信息,以及上次升级版本号。
web端在启动升级脚本之前,必须进行升级版本验证,只有当前系统版本和补丁版本与升级包描述的上次升级版本和上次升级补丁版本一致时,才是正确的升级补丁包,才会执行升级。

注意事项(一)

升级脚本必须有足够的执行权限才能保证升级脚本能够正确的被执行,如果是新部署的服务器,可能会出现脚本文件权限不足导致升级执行无效。所以部署项目之后,首先需要修改相关脚本的权限,chmod 777 *给与足够权限,否则可能会出现升级无效的情况。这种情况下,将后台执行的shell命令拷贝出来在服务器上执行,很容易看到命令执行失败的原因是No permmition。

注意事项(二)

RMI注册的是本地IP和本地默认RMI端口,127.0.0.1:1099,所以/ect/hosts文件对RMI程序会有影响,项目中有一个IP设置的功能,系统初始化之前已经将服务器主机名称固定了,所以IP修改操作,也必须保证/ect/hosts文件中的主机名称和127.0.0.1的对应关系,否则IP变更后,RMI根据原来的hosts文件的IP查找主机名称时由于IP已经修改了,连接超时,导致升级操作无法进行。
解决办法:IP修改操作的Shell脚本中,完成IP设置后,重置/ect/hosts文件为标准文件,在项目部署的/bin目录下定义一个standardhosts文件,每次IP设置重启网卡之前,重置下该文件内容:

#update /etc/hosts
cat /deploy_path/bin/standardhosts>/etc/hosts

注意事项(三)

编写启动该程序的脚本starttoolkit.sh 内容为java -jar xxx.jar,保证它跟项目tomcate一起开机启动,添加到开机启动项文件tomcat中,以后台程序的方式启动:

  nohup sh /deploy_path/bin/starttoolkit.sh &
版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

java web项目war包自动升级部署方案

前言之前,我们公司部署以及升级都是由运维去管理的,联想到很多开源平台都支持自动升级,索性我也做个自动升级war的功能。 这里没有用docker镜像发包,灰度发包等,只适用于单个tomcat的部署环境...

web程序在线更新实现

说说以前的一个web程序在线更新实现,给没有实现过的人们提供一点思路,这是我一个项目的实现,也许方式不是很好,但是它实现了基本的需求。          公司的开发了一个web产品,这个web产品部...
  • wckjlu
  • wckjlu
  • 2017年04月13日 10:32
  • 202

Delphi7高级应用开发随书源码

  • 2003年04月30日 00:00
  • 676KB
  • 下载

web开源框架大汇总

Struts 项目简介信息 Struts是一个基于Sun J2EE平台的MVC框架,主要是采用Servlet和JSP技术来实现的。由于Struts能充分满足应用开发的需求,简单易用,敏捷迅速,在过...

HBuilder webApp开发(十)在线差异化升级

从4月份公司确定使用H5开发app,主要考虑了两点: 1.一套代码可以生成安卓和iOS 两个平台的app; 2.在线升级(差异化升级和整包升级)。 到现在公司的两款app也基本做完了,期间还做了...

java中客户端的代码升级

场景:当产品已经打包给客户使用,程序某处出现问题,而这个问题是来自某个类的代码,需要把这个java文件修改后更新给客户。 解决:一种办法是等到下一个升级版本,把bug的解决方案包含到新版本中; 另...

给多台机器的Java应用程序写一个自动更新的程序

问题:现有70几台前置机要逐步安装Java应用程序,现已经在10台机器上已经安装了Java应用程序。如果代码有变动的话,只能通过远程桌面连接到前置机的电脑进行手工操作。这样一天时间都荒废在这种无聊的时...
  • jdk2006
  • jdk2006
  • 2011年08月21日 06:35
  • 5720

Delphi7高级应用开发随书源码

  • 2003年04月30日 00:00
  • 676KB
  • 下载

给定A, B两个整数,不使用除法和取模运算,求A/B的商和余数

给定A, B两个整数,不使用除法和取模运算,求A/B的商和余数。 1.   最基本的算法是,从小到大遍历: for (i = 2 to A -1)          if (i * B > A)...

利用K-means聚类算法根据经纬度坐标对中国省市进行聚类

K-means聚类算法是一种非层次聚类算法,在最小误差的基础上将数据划分了特定的类,类间利用距离作为相似度指标,两个向量之间的距离越小,其相似度就越高。程序读取全国省市经纬度坐标,然后根据经纬度坐标进...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:开发整理-Javaweb应用的系统升级功能
举报原因:
原因补充:

(最多只允许输入30个字)