<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆-57  評論-117  文章-1  trackbacks-0

    副標題:利用ant腳本 自動構建svn增量/全量 系統程序升級包

    首先請允許我這樣說,作為開發或測試,你一定要具備這種本領。你可以手動打包、部署你的工程,但這不是最好的方法。最好的方式就是全自動化的方式。開發人員提交了代碼后,可以自動構建、打包、部署到測試環境。測試通過后進入到模擬環境或是直接發布的生產環境,這個過程可以是全自動的。但這個自動化的方式有一些公司用到了,但也有很多公司還不知道,他們的攻城師天天在做反復、沒有多大意義的、浪費生命的事情。當然這種方式一般針對做自己的產品,如電商、通信行業。而給其他公司或企業做項目的比較少利用這種方式,當然也可以利用這種方式,只是不能直接發布到客戶的服務器而已。當然有些公司是專門有人做這部分事情的!

    說了這么多也沒有什么惡意,只是覺得這種方式可以大大提高效率,降低人力、物力、財力而已。勿噴,O(∩_∩)O哈哈~嘿嘿~

    在部署項目或打包項目中,通常大家都是手動部署或打包的多。很多公司把這一任務交個了我們做開發的,其實這部分應該誰做呢?本質上應該是測試的來完成,但一些公司的測試不會做這個,慢慢的就變成了開發的事情。有些公司是人手比較少、不健全,所以一部分人自己承擔了這個事情。在我們手動打包的時,其實這是一個重復的、沒有技術含量的、耗費體力的活兒。一般步驟就是更新svn上代碼、修改好相關的配置、編譯class、發布到tomcat(web工程)、測試啟動無誤、手動打包。

    而有一種比較簡單快速的方式就是利用meven或ant來完成這些工作,只要我們編寫好腳本后。給相關的工作人員去運行這部分腳本就可以完成打包,甚至是部署項目,這些都是so easy~!我個人也比較贊成使用這種方式,比較簡單、快速、重用性好,最難的可能編寫build腳本,但這個腳本其實也是很簡單的,它就想dos命令行,只不過它是用xml方式來完成命令行的而已。所以測試會寫ant的build腳本這個也是應該的,如果你不會的話,還是建議你學習學習。反正是百益無一害的事情,何樂而不為呢!

     

    一、基本流程

    利用ant打包項目或打增量包的基本流程

    image

    值得一提的是jar包這個部分,這個步驟是為下面編譯增量包做準備的。因為增量包導出的增量文件,它依賴于整個項目的其他代碼,如果沒有這些代碼的支持是編譯不通過。然而又不能直接通過diff得到增量的class,所以只能導出增量文件后,通過引用全部工程的代碼的class再進行編譯即可。

     

    二、運行環境

    1、安裝jdk,不會自己上網查其他的

    2、如果你還沒有安裝ant,那么你可以參考:http://www.cnblogs.com/hoojo/archive/2013/06/14/java_ant_project_target_task_run.html

    會介紹一下ant的安裝和使用的方法。

    3、這里需要用到svn的ant相關工具包、命令支持。你需要下載svnant-1.3.1.zip,將里面的lib庫放置在你的ant腳本的編譯運行環境中。

    4、因為某些項目使用到了泛型、annotation注解,使用javac有些代碼是編譯不通過的,所以這里使用了jdt的編譯方式。參考:使用eclipse JDT compile class 會有很詳細的介紹。

    需要用到

    jdtCompilerAdapter.jar
    org.eclipse.jdt.compiler.tool_1.0.1.v_793_R33x.jar
    org.eclipse.jdt.core_3.3.3.v_793_R33x.jar
    org.eclipse.jdt.debug.ui_3.2.102.v20071002_r332.jar
    復制到ant_home/lib目錄下,如果是利用eclipse運行腳本就需要把它加載到運行環境中。可以參考上面的:使用eclipse JDT compile class

     

    三、編寫ant的build腳本

    1、首先看看ant工程的目錄結構

    image

    簡單介紹下目錄結構:

    src下面的ExportIncrementFiles.java是導出增量文件要用的,它在build命令increment中執行。它會讀取diff 比較后的文件中的內容,并導出文件

    dest 是checkout出來最新的svn的工程

    dist 是編譯上面dest目錄中的工程,也是svn全量war的工程目錄和jar

    increment_dest 是增量工程,也就是上面的ExportIncrementFiles工具導出的工程

    increment_dist 是編譯上面increment_dest 的工程,也是增量包的工程目錄

    因為每個人的項目工程目錄結構不一樣,所以這個腳本并不會通用,我這里指針對自己的項目進行測試。

    lib中是運行環境需要的jar庫,其中主要的就是svnlib 這個你可以去下載 svnant-1.3.1.zip 以及JDT編譯class的jar包,這個可以通過eclipse中的plugin中的jar包找到,可以參考:使用eclipse JDT compile class

    increment.export.jar就是ExportIncrementFiles的class打成的jar包,這個是自己打的包,可以直接應用class也可以的,在increment命令引用,jar下載:http://download.csdn.net/detail/ibm_hoojo/6501165

    build.properties是當前build的配置文件

    build.xml就是主要的ant腳本

    選中部分是打的war包,這個就可以部署了

    patch.txt就是svn的diff 比較出的增量文件的目錄路徑列表

     

    2、ExportIncrementFiles.java 導出增量文件

    package com.hoo.util;
     
    import java.io.BufferedReader;
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.FileReader;
     
    /**
     * <b>function:</b> 導出在增量的文件
     * @author hoojo
     * @createDate 2013-11-2 下午10:00:01
     * @file ExportIncrementFiles.java
     * @package com.hoo.util
     * @project AntTest
     * @blog http://blog.csdn.net/IBM_hoojo
     * @email hoojo_@126.com
     * @version 1.0
     */
    public class ExportIncrementFiles {
     
        /**
         * <b>function:</b> 導出增量文件
         * @author hoojo
         * @createDate 2013-11-2 下午10:15:43
         * @param configPath 增量文件路徑配置目錄
         * @param baseDir 基本路徑 目標位置
         * @param destDir 增量文件保存位置
         * @throws Exception
         */
        private static void export(String configPath, String baseDir, String destDir) throws Exception {
            String srcFile = baseDir + configPath;
            String desFile = destDir + configPath;
            
            int lastIndex = desFile.lastIndexOf("/");
            String desPath = desFile.substring(0, lastIndex);
            
            File srcF = new File(srcFile);
            if(srcF.exists()){//如果不存在這樣的源文件,就不再拷貝,這個用來解決版本之間有刪除文件的情況。
                File desF = new File(desFile);
                File desP = new File(desPath);
                if(!desP.exists()) {
                    desP.mkdirs();
                }
                System.out.println(srcFile);
                FileInputStream fis = new FileInputStream(srcF);
                FileOutputStream fos = new FileOutputStream(desF);
                
                byte[] buf = new byte[1024];
                int len = 0;
                while((len = fis.read(buf)) != -1) {
                    fos.write(buf,0,len);
                }
                fos.flush();
                fos.close();
                fis.close();
            }
        }
        
        /**
         * <b>function:</b> 主函數 執行導出增量包任務
         * @author hoojo
         * @createDate 2013-11-2 下午10:00:01
         * @param args 參數1 增量包導出文件路徑,參數2 要導出的文件的所在目標位置,參數3 增量包導出保存的位置路徑
         */
        public static void main(String[] args) {
     
            if (args.length > 0) {
                if (args.length == 1 && "help".equals(args[0])) {
                    System.out.println("args[0] is Export Increment Files content path");
                    System.out.println("args[1] is Export Increment Files target path");
                    System.out.println("args[2] is Increment Files Export loaction");
                } else {
                    String configPath = args[0];
                    String baseDir = args[1];
                    String destDir = args[2];
                    
                    try {
                        BufferedReader br = new BufferedReader(new FileReader(configPath));
                        String s = null;
                        while((s = br.readLine()) != null) {
                            s = s.trim();//去掉路徑前面的空格
                            String str = destDir + s;
                            if(!destDir.equals(str)){//過濾空行
                                export(s, baseDir, destDir);
                            }
                        }
                        br.close();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }

    main函數參數看注釋,主要就是讀取patch.txt增量文件路徑,導出文件到指定目錄的。

     

    3、build 腳本

    <?xml version="1.0" encoding="UTF-8" ?>
    <!-- createDate 2013-10-28 -->
    <!-- author hoojo & http://blog.csdn.net/IBM_hoojo & http://hoojo.cnblogs.com -->
    <project default="checkout" basedir=".">
        <property file="build.properties"/>
        
        <!-- svn 比較項目最新路徑 -->
        <property name="svn.url" value="${svn._url}"/>
        <!-- svn 備份路徑-->
        <property name="bak.svn.url" value="${bak.svn._url}"/>
        
        <property name="svn.username" value="${svn.username}"/>
        <property name="svn.password" value="${svn.password}"/>
        
        <!-- 項目名稱 -->
        <property name="webapp" value="${webapp.name}"/>
        <!-- 目標項目的Web 名稱(WEB-INF上一級的目錄名稱) -->
        <property name="webroot" value="${web.root}"/>
        
        <!-- svn改動文件列表信息 -->
        <property name="compare.path.file" value="${increment.file}"/>
        
        <!-- svn導出/切出文件存放目錄 -->
        <property name="dest.path" location="dest/${webapp}"/>
        <!-- svn導出/切出文件編譯后存放目錄 -->
        <property name="dist.path" location="dist/${webapp}"/>
        <!-- svn增量文件保存目錄 -->
        <property name="increment.dest.path" location="increment_dest/${webapp}"/>
        <!-- svn增量文件編譯后保存目錄 -->
        <property name="increment.dist.path" location="increment_dist/${webapp}"/>
        
        <!-- 利用jdt編譯class 解決泛型不能轉換的問題 需要將
            jdtCompilerAdapter.jar
            org.eclipse.jdt.compiler.tool_1.0.1.v_793_R33x.jar
            org.eclipse.jdt.core_3.3.3.v_793_R33x.jar
            org.eclipse.jdt.debug.ui_3.2.102.v20071002_r332.jar
            復制到ant_home/lib目錄下
        <property name="build.compiler" value="org.eclipse.jdt.core.JDTCompilerAdapter"/>
        -->
        <path id="svnant.classpath">
            <fileset dir="${basedir}">
                <include name="**/*.jar"/>
            </fileset>
        </path>
        
        <path id="buildpath">
            <fileset dir="${dest.path}">
                <include name="**/lib/*.jar"/>
            </fileset>
            <fileset dir="C:/Program Files/Java/jdk1.6.0_13">
                <include name="**/*.jar"/>
            </fileset>
        </path>
        
        <typedef resource="org/tigris/subversion/svnant/svnantlib.xml" classpathref="svnant.classpath"/>
        <svnSetting id="svn.settings" javahl="false" svnkit="true" username="${svn.username}" password="${svn.password}" failonerror="true"/>
        
        <target name="init" description="init clean dirs">
            <echo message="${svn.username}"/>
            <echo message="${svn.password}"/>
            <echo message="${webapp}"/>
            <echo message="${webroot}"/>
            <echo message="${compare.path.file}"/>
            
            <delete dir="${dest.path}" failonerror="false" deleteonexit="true" excludes="**/lib"/>
            <delete dir="${dist.path}" failonerror="false" deleteonexit="true" excludes="**/lib"/>
            <delete file="${compare.path.file}" failonerror="false"/>
            <delete dir="${increment.dest.path}" failonerror="false" deleteonexit="true"/>
            <delete dir="${increment.dist.path}" failonerror="false" deleteonexit="true"/>
        </target>
        
        <!-- that is to test i svnant is available //-->
        <target name="tool-available" depends="init">
            <echo message="run task test svnant is available"></echo>
            <available resource="org/tigris/subversion/svnant/svnantlib.xml" classpathref="svnant.classpath" property="available.svnant"/>
            <echo message="SVN-ANT is available = ${available.svnant}"></echo>
        </target>
        
        <!-- 比較差異 增量文件 -->
        <target name="diff" description="deff/compare project">
             <svn refid="svn.settings">
                <diffSummarize oldUrl="${bak.svn.url}" newUrl="${svn.url}" outFile="${compare.path.file}" recurse="true"/>
             </svn>
        </target>
            
        <!-- 下載 切成 導出 服務器上最新代碼 -->
        <target name="checkout" depends="tool-available" description="checkout/export project code ${svn.url} ">
            <echo message="checkout/export project code ${svn.url}"></echo>
            <svn refid="svn.settings">
                 <export srcUrl="${svn.url}" destPath="${dest.path}" revision="HEAD" force="true"/>
            </svn>
       </target>
        
        <!-- javac編譯 -->
        <target name="compile">
            <buildnumber/>
            <echo>compile ${dest.path} ......</echo>
            <delete dir="${dist.path}" failonerror="false" deleteonexit="true" excludes="**/lib"/>
            <mkdir dir="${dist.path}/classes"/>
            
            <javac nowarn="true" debug="${javac.debug}" debuglevel="${javac.debuglevel}" destdir="${dist.path}/classes" source="${javac.source}" target="${javac.target}" encoding="utf-8" fork="true" memoryMaximumSize="512m" includeantruntime="false">
                <src path="${dest.path}/src"/>
                <!--
                <compilerarg value="-Xlint:unchecked"/>
                <compilerarg value="-Xlint:deprecation"/>
                <compilerarg value="-Xlint"/>
                -->
                <classpath refid="buildpath"/>
                <classpath refid="svnant.classpath"/>
            </javac>
        </target>
        
        <!-- 利用JDT編譯 -->
        <target name="compile_jdt">
            <buildnumber/>
            <echo>compile ${dest_path} ......</echo>
            <delete dir="${dist_path}" failonerror="false" deleteonexit="true" excludes="**/lib"/>
            <mkdir dir="${dist_path}/classes"/>
            
            <javac compiler="org.eclipse.jdt.core.JDTCompilerAdapter" nowarn="true" debug="${javac.debug}" debuglevel="${javac.debuglevel}" destdir="${dist_path}/classes" source="${javac.source}" target="${javac.target}" encoding="utf-8" fork="true" memoryMaximumSize="512m" includeantruntime="false">
                <src path="${dest_path}/src"/>
                
                <classpath refid="buildpath"/>
                <classpath refid="svnant.classpath"/>
            </javac>
        </target>
        
        <!-- 利用JDT編譯SVN 最新項目 -->
        <target name="compile_svn">
            <!-- 回調任務 -->
            <antcall target="compile_jdt">
                <param name="dest_path" value="${dest.path}"/>
                <param name="dist_path" value="${dist.path}"/>
            </antcall>
        </target>
        
        <!-- 將全部項目的class 建立jar包 -->
        <target name="jar" depends="compile_svn">
            <jar destfile="${basedir}/lib/${webapp}.jar" level="9" compress="true" encoding="utf-8" basedir="${dist.path}/classes">
                <manifest>
                    <attribute name="Implementation-Version" value="Version: 2.2"/>
                </manifest>
            </jar>
        </target>
        
        <!-- 導出增量文件 -->
        <target name="increment" depends="diff">
            <java classname="com.hoo.util.ExportIncrementFiles" classpath="${basedir}/lib/increment.export.jar" fork="true">
                <arg value="${compare.path.file}"/>
                <arg value="${dest.path}/"/>
                <arg value="${increment.dest.path}/"/>
            </java>
        </target>
     
        <!-- 利用JDT編譯增量文件 -->
        <target name="compile_increment">
            <antcall target="compile_jdt">
                <param name="dest_path" value="${increment.dest.path}"/>
                <param name="dist_path" value="${increment.dist.path}"/>
            </antcall>
        </target>
        
        <!-- 全部打包 -->    
        <target name="war">
            <echo>create war file.......</echo>
            
            <copy todir="${dist_path}" failonerror="false">
                <fileset dir="${dest_path}/${webroot}" includes="**"/>
            </copy>
            <move todir="${dist_path}/WEB-INF/classes" failonerror="false">
                <fileset dir="${dist_path}/classes" />
            </move>
            <copy todir="${dist_path}/WEB-INF/classes" failonerror="false">
                <fileset dir="${dest_path}/src/main/" includes="**/*.xml, **/*.properties, **/*.xsd"/>
                <fileset dir="${dest_path}/src/test/" includes="**/*.xml, **/*.properties, **/*.xsd"/>
                <fileset dir="${dest_path}/src/resource/" includes="**/*.xml, **/*.properties, **/*.xsd"/>
            </copy>
            
            <!--得到當前日期--> 
            <tstamp> 
                <format property="DSTAMP" pattern="yyyyMMdd" locale="zh"/> 
                <format property="TSTAMP" pattern="HHmmss" locale="zh"/> 
            </tstamp> 
            
            <war destfile="${basedir}/${webapp}_${DSTAMP}_${TSTAMP}.war" basedir="${dist_path}" webxml="${dist_path}/WEB-INF/web.xml"/>        
        </target>
        
        <!-- 全部打包 -->    
        <target name="war_svn">
            <antcall target="war">
                <param name="dest_path" value="${dest.path}"/>
                <param name="dist_path" value="${dist.path}"/>
            </antcall>        
        </target>
        
        <!-- 全部打包 -->    
        <target name="war_increment">
            <copy todir="${increment.dist.path}/WEB-INF" file="${dest.path}/${webroot}/WEB-INF/web.xml"/>
            <antcall target="war">
                <param name="dest_path" value="${increment.dest.path}"/>
                <param name="dist_path" value="${increment.dist.path}"/>
            </antcall>        
        </target>
        
        <!-- svn 全量包 -->
        <target name="svn_war" depends="checkout, compile_svn, war_svn"/>
        <!-- 增量包 -->
        <target name="increment_war" depends="checkout, increment, jar, compile_increment, war_increment"/>
    </project>

     

    4、build的配置文件內容

    #Mon, 04 Nov 2013 11:18:12 +0800
    svn._url=http://172.31.100.100/svn/iMVS_DataComm2
    bak.svn._url=http://172.31.100.100/svn/iMVS_DataComm
    svn.username=hoojo
    svn.password=mypass
    webapp.name=iMVS_DataComm
    web.root=WebRoot
    increment.file=patch.txt
     
    javac.debuglevel=source,lines,vars
    javac.target=1.6
    javac.source=1.6
    javac.debug=true

     

    運行svn_war任務可以打全部的包,也就是svn最新地址的項目工程包。

    運行increment_war任務可以打增量包,也會形成一個war文件。

    如果你需要發布到tomcat目錄,可以寫一個任務copy相關war包到tomcat的webapps的目錄下,這個很簡單~如果你需要調用tomcat的相關任務或命令,你需要在build腳本中加入

    <target name="_def_tomcat_tasks">
      <!-- tasks: deploy,undeploy,reload,stop,start,list,roles,resources -->
      <taskdef name="deploy"    classname="org.apache.catalina.ant.DeployTask" />
      <taskdef name="list"      classname="org.apache.catalina.ant.ListTask" />
      <taskdef name="reload"    classname="org.apache.catalina.ant.ReloadTask" />
      <taskdef name="resources" classname="org.apache.catalina.ant.ResourcesTask" />
      <taskdef name="roles"     classname="org.apache.catalina.ant.RolesTask" />
      <taskdef name="start"     classname="org.apache.catalina.ant.StartTask" />
      <taskdef name="stop"      classname="org.apache.catalina.ant.StopTask" />
      <taskdef name="undeploy"  classname="org.apache.catalina.ant.UndeployTask" />
    </target>

    關于這些命令使用方法有興趣的可以自己研究研究。

     

    四、總結

    整個流程稍微有點復雜,只要思路清晰這個build腳本還是很容易編寫的。前提是你要懂得Java發布、編譯、部署的流程,很多人用eclipse或MyEclipse來發布工程,好像很簡單。其實在工具背后也是使用這些腳本完成的,知道在用戶目錄下有一個 .m2的目錄么,這個就是eclipse工具的meven的緩存和配置的內容。所以當我們不使用這些ide的情況下,你怎么編譯、部署你的項目呢~!這篇文章只是一個拋磚引玉的效果,希望能給大家一個啟示。在這之前,我在網上搜集了些資料也沒有找到打增量包的比較好的方法,全都是手動方式。

    同時,在一些自動集成或持續集成的智能工具中,也大量的使用到了這方面的技術。如果你想更智能的完成這個項目的發布、部署的話,這里只是其中的第一步。有興趣的朋友可以研究下Continuous integration或Hudson等相關自動化集成技術應用。



    作者:hoojo
    出處:
    blog:http://blog.csdn.net/IBM_hoojo
             http://hoojo.cnblogs.com
    本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。


    版權所有,轉載請注明出處 本文出自:
    分享道版權所有,歡迎轉載,轉載請注明出處,謝謝
    posted on 2013-11-05 09:01 hoojo 閱讀(17044) 評論(2)  編輯  收藏 所屬分類: IDE/UtilsJavaEEOthersRCP【富客戶端技術】RIA 【富互聯網程序】

    評論:
    # re: 不會用ant打包、部署項目的工程師,不是一個好程序員(測試) 2013-11-05 12:31 | 魏五鎖業
    作為開發或測試,你一定要具備這種本領  回復  更多評論
      
    # re: 不會用ant打包、部署項目的工程師,不是一個好程序員(測試)[未登錄] 2014-11-19 13:30 | Jack
    用tortoise導出提交文件,在用myeclipse更新,對比src目錄考出.classes文件會方便的多  回復  更多評論
      
    主站蜘蛛池模板: 亚洲最新视频在线观看| 亚洲人成精品久久久久| 青娱乐在线免费观看视频| 国产啪亚洲国产精品无码 | 亚洲Aⅴ无码专区在线观看q| 免费观看激色视频网站bd| 亚洲精品又粗又大又爽A片| 国产偷国产偷亚洲高清日韩| 亚洲成A人片77777国产| 在线观看免费人成视频色9| 一级毛片a女人刺激视频免费| 亚洲理论片在线中文字幕| 亚洲无码日韩精品第一页| 蜜桃精品免费久久久久影院| 国产永久免费高清在线| 亚洲乱亚洲乱妇24p| 色欲aⅴ亚洲情无码AV蜜桃| 久久亚洲私人国产精品vA| 亚洲激情在线视频| 国产亚洲?V无码?V男人的天堂| 自拍偷自拍亚洲精品第1页| 无码欧精品亚洲日韩一区夜夜嗨 | 成人免费观看一区二区| 黄网站免费在线观看| 免费无遮挡无遮羞在线看 | 亚洲成AV人片在线播放无码| 成在线人永久免费视频播放| 最近2019免费中文字幕视频三| 九九综合VA免费看| 国产在线观看免费视频软件| 成人免费观看一区二区| 国产福利免费在线观看| 国语成本人片免费av无码| 99在线观看精品免费99| 中文字幕视频在线免费观看| 免费在线人人电影网| 国产午夜精品久久久久免费视 | 伊人久久国产免费观看视频| 亚洲精品美女久久7777777| 亚洲丰满熟女一区二区v| 亚洲午夜在线电影|