使用java开发简单的自动部署

背景:

作为一名开发人员,不可避免地需要部署项目。

在没使用Jenkins之前,每次部署都使人感觉到很繁琐。

那有没有办法自己弄一个程序来减轻一下部署的工作呢?

就是运行一下程序,想要部署的项目就直接部署到服务器中。

准备阶段

一、了解部署项目的步骤过程
  1. 将打包好的jar文件上传到服务器上

  2. 查看项目是否在运行,运行的话就停止

  3. 启动项目程序

二、前置条件:
  1. 一个简单的springboot项目

  2. 一台配置好环境的服务器

实际开发:

一、简单了解Jsch

JSch是Java Secure Channel的缩写。JSch是一个SSH2的纯Java实现。它允许你连接到一个SSH服务器,并且可以使用端口转发,X11转发,文件传输等,当然你也可以集成它的功能到你自己的应用程序。

这里只需使用JSch实现的SFTP功能

SFTP是Secure File Transfer Protocol的缩写,安全文件传送协议。可以为传输文件提供一种安全的加密方法。SFTP 为 SSH的一部份,是一种传输文件到服务器的安全方式。SFTP是使用加密传输认证信息和传输的数据,所以,使用SFTP是非常安全的。但是,由于这种传输方式使用了加密/解密技术,所以传输效率比普通的FTP要低得多。

ChannelSftp类是JSch实现SFTP核心类,它包含了所有SFTP的方法

二、拆分步骤

要使用JSch将项目部署到服务器中,可拆分为如下步骤:

  1. 连接到服务器

  2. 创建新项目文件夹

  3. 将项目上传到新创建的文件夹中

  4. 查看进程并关闭正在运行的项目程序

  5. 启动项目程序

三、具体程序代码
  1. 添加依赖的jar包

<dependency>

   <groupId>com.jcraft</groupId>

   <artifactId>jsch</artifactId>

   <version>0.1.54</version>

  </dependency>

  1. 连接到服务器

    public void connect() throws JSchException {

        jsch = new JSch();

        session = jsch.getSession(user, host, port);

        session.setPassword(passwd);

        java.util.Properties config = new java.util.Properties();

        config.put("StrictHostKeyChecking", "no");

        session.setConfig(config);

        session.connect();

        Channel channel = session.openChannel("sftp");

        channel.connect();

        sftp = (ChannelSftp) channel;

        log.info("连接到SFTP成功。host: " + host);

    }

  1. 创建新项目文件夹

public void createDir(String createpath) {

        try {

            if (isDirExist(createpath)) {

                sftp.cd(createpath);

                return;

            }

            String pathArry[] = createpath.split("/");

            StringBuffer filePath = new StringBuffer("/");

            for (String path : pathArry) {

                if (path.equals("")) {

                    continue;

                }

                filePath.append(path + "/");

                if (isDirExist(filePath.toString())) {

                    sftp.cd(filePath.toString());

                } else {

                    // 建立目录

                    sftp.mkdir(filePath.toString());

                    // 进入并设置为当前目录

                    sftp.cd(filePath.toString());

                }

            }

            sftp.cd(createpath);

        } catch (SftpException e) {

            throw new RuntimeException("创建路径错误:" + createpath);

        }

    }

  1. 将项目上传到新创建的文件夹中
public void uploadFile(String local,String remote) throws Exception {

        File file = new File(local);

        if (file.isDirectory()) {

            throw new RuntimeException(local + " is not a file");

        }

        InputStream inputStream = null;

        try {

            String rpath = remote.substring(0,remote.lastIndexOf("/")+1);

            if (!isDirExist(rpath)){

                log.info("目录不存在,创建目录:{}",rpath);

                createDir(rpath);

            }

            inputStream = new FileInputStream(file);

            sftp.setInputStream(inputStream);

            sftp.put(inputStream, remote);

            log.info("上传文件完成");

        } catch (Exception e) {

            log.info("上传文件失败");

            throw e;

        }finally{

            if(inputStream != null){

                inputStream.close();

            }

        }

    }

  1. 远程执行Shell命令,执行完后返回结果

public int execCmd(String command) throws Exception{

        log.info( "开始执行命令:" + command);

        int returnCode = -1;

        BufferedReader reader = null;

        Channel channel = null;

        channel = session.openChannel("exec");

        ((ChannelExec) channel).setCommand(command);

        channel.setInputStream(null);

        ((ChannelExec) channel).setErrStream(System.err);

        InputStream in = channel.getInputStream();

        reader = new BufferedReader(new InputStreamReader(in));

        channel.connect();

        log.info("The remote command is:{}" ,command);

        String buf ;

        while ((buf = reader.readLine()) != null) {

            log.info(buf);

        }

        reader.close();

        // Get the return code only after the channel is closed.

        if (channel.isClosed()) {

            returnCode = channel.getExitStatus();

        }

        log.info( "Exit-status:" + returnCode );

       /* StringBuffer buf = new StringBuffer( 1024 );

        byte[] tmp = new byte[ 1024 ];

        while ( true ) {

            while ( in.available() > 0 ) {

                int i = in.read( tmp, 0, 1024 );

                if ( i < 0 ) break;

                buf.append( new String( tmp, 0, i ) );

            }

            if ( channel.isClosed() ) {

                res = channel.getExitStatus();

                log.info( "Exit-status:" + res );

                System.out.println( "Exit-status:" + res );

                break;

            }

            TimeUnit.MILLISECONDS.sleep(100);

        }

        log.info( buf.toString() );*/

        channel.disconnect();

        return returnCode;

    }

  1. 远程执行Shell命令,每当有一行结果就返回

public int shellCmd(String command) throws Exception{

        long startTime=System.currentTimeMillis();

        log.info( "开始执行命令:" + command);

        int returnCode = -1;

        boolean isTimeOut = false;

        ChannelShell channel=(ChannelShell) session.openChannel("shell");

        InputStream in = channel.getInputStream();

        channel.setPty(true);

        channel.connect();

        OutputStream os = channel.getOutputStream();

        os.write((command + "\r\n").getBytes());

        os.write("exit\r\n".getBytes());

        os.flush();

        log.info("The remote command is:{}" ,command);

        byte[] tmp=new byte[1024];

        while(true) {

            while (in.available() > 0) {

                int i = in.read(tmp, 0, 1024);

                if (i < 0) break;

                log.info(new String(tmp, 0, i));

            }

            if (channel.isClosed()) {

                if (in.available() > 0) continue;

                returnCode = channel.getExitStatus();

                log.info("exit-status: " + channel.getExitStatus());

                break;

            }

            long useTime=System.currentTimeMillis()-startTime;

            if (useTime> 600000){

                log.info("[error] Execution command timeout!" );

                isTimeOut = true;

                break;

            }

            try{Thread.sleep(1000);}catch(Exception ee){}

        }

        long endTime=System.currentTimeMillis();

        float excTime=(float)(endTime-startTime)/1000;

        log.info("执行命令是否超时:{},共耗时:{}s",isTimeOut,excTime);

        os.close();

        in.close();

        channel.disconnect();

        session.disconnect();

        return returnCode;

    }

  1. 具体的执行,如需要部署simple-auto-deploy.jar

 public static void main(String[] args) {

        JSchExecutor jSchUtil = new JSchExecutor( "123", "123123","192.168.243.18");

        try {

            jSchUtil.connect();

            jSchUtil.execCmd("kill -9 `ps -ef | grep simple-auto-deploy.jar | grep -v grep | awk '{print $2}'`");

            jSchUtil.uploadFile("C:\\mywork\\Workspaces\\IdeaProjects\\simple-auto-deploy\\target\\simple-auto-deploy.jar","/data/simple-auto-deploy/simple-auto-deploy.jar");

            jSchUtil.shellCmd("nohup java -jar /data/simple-auto-deploy/simple-auto-deploy.jar >/dev/null &");

        } catch (Exception e) {

            e.printStackTrace();

        }finally {

            jSchUtil.disconnect();

        }

    }

总结

这样一个简单的自动部署就出来了,运行main方法,程序就自动部署到服务器中。

代码可参考:https://github.com/visionsws/learnJava 中的 simple-auto-deploy

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,651评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,802评论 6 342
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,059评论 25 707
  • 悄悄地,是一个不眠夜, 心情拨动了特殊的音弦, 脑袋里回荡着舒心的乐, 心里的鬼胎尚不在人间, 感性和天真又浪漫天边。
    稻草翁挥帽阅读 161评论 0 5
  • 同属“天朝三少”,洗车小哥无疑是比快递小哥、外卖小哥更为Bug的一种存在。外卖小哥无非是风一样的男子,野蛮冲撞过路...
    UCen宇晨阅读 326评论 0 0