maven自動部署到遠程tomcat教程


使用maven的自動部署功能可以很方便的將maven工程自動部署到遠程tomcat服務器,節省了大量時間。

本文章適用於tomcat的7.x ,8.x, 9.x版本。

下面是自動部的步驟

1,首先,配置tomcat的manager

編輯遠程tomcat服務器下的conf/tomcat-users.xml,在末尾增加(其實只要拉到文件末尾,去掉注釋改一下就可以了)

<role rolename="manager-gui"/>
<role rolename="manager-script"/>
<user username="admin" password="password" roles="manager-script"/>
<user username="root" password="password" roles="manager-gui"/>

將上面的password改為自己的密碼,注意對於tomcat9來說,不能同時賦予用戶manager-script和manager-gui角色。

保存tomcat-users.xml。

在tomcat服務器的conf/Catalina/localhost/目錄下創建一個manager.xml文件,寫入如下值:

<?xml version="1.0" encoding="UTF-8"?>
<Context privileged="true" antiResourceLocking="false"
         docBase="${catalina.home}/webapps/manager">
             <Valve className="org.apache.catalina.valves.RemoteAddrValve" allow="^.*$" />
</Context>

 

保存退出。

然后在瀏覽器中輸入http://serverip:port/manager/html,此時會彈出要求輸入用戶名和密碼對話框,輸入manager-gui對應的用戶和密碼登錄管理控制台(其中serverip為服務器ip,如果服務器在本地就是localhost或者127.0.0.1,端口為tomcat端口,默認8080)。以此確認manager是否配置正確。正確結果示例如下:

 

2,在maven項目中添加配置

在pom.xml文件中,在plugins節點下添加如下plugin節點

<plugin>

    <groupId>org.apache.tomcat.maven</groupId>
    <artifactId>tomcat7-maven-plugin</artifactId>
    <version>2.2</version>

    <configuration>
        <url>http://serverip:port/manager/text</url>
        <username>admin</username>
        <password>password</password>
        <update>true</update>
        <path>/webapp</path>
    </configuration>

</plugin>

將上面的serverip和port換成自己tomcat服務器的ip和端口。密碼換成上面配置的manager-script角色的密碼。path改為項目在tomcat服務器中的部署路徑。


然后進行部署,如果是第一次部署,運行mvn tomcat7:deploy進行自動部署(對於tomcat8,9,也是使用tomcat7命令),如果是更新了代碼后重新部署更新,運行mvn tomcat7:redeploy,如果第一次部署使用mvn tomcat7:redeploy,則只會執行上傳war文件,服務器不會自動解壓部署。如果路徑在tomcat服務器中已存在並且使用mvn tomcat7:deploy命令的話,上面的配置中一定要配置<update>true</update>,不然會報錯。

如果IDE是eclipse,就在runas->run configurations中配置一個maven build,intellij類似。

3. 內存泄漏

 使用上面的方法進行部署后會出現嚴重的內存泄漏現象。tomcat的manager提供了診斷在部署時是否產生內存泄漏的功能,在上面提到的http://serverip:port/manager/html這個頁面底部有一個“Find leaks”的按鈕,如下:

點擊按鈕,網頁頭部出現如下信息說明在部署的時候有內存泄漏:

上面的消息顯示部署的test項目存在內存泄漏,如果同一項目多次重新部署,則一個項目名可能會出現多次。

部署時產生內存泄漏的原因是每次(重新)部署時,Tomcat會為項目新建一個類加載器,而舊的類加載器沒有被GC回收。maven的庫classloader-leak-prevention-servlet可以用來解決這個問題。具體方案為:

 

(1)添加maven依賴:

<dependency>
    <groupId>se.jiderhamn.classloader-leak-prevention</groupId>
    <artifactId>classloader-leak-prevention-servlet</artifactId>
    <version>2.1.0</version>
</dependency>

(2)在項目的web.xml中添加一個Listener(必須讓此Listener成為web.xml中的第一個Listener,否則不起作用

<listener>
    <listener-class>se.jiderhamn.classloader.leak.prevention.ClassLoaderLeakPreventorListener</listener-class>
</listener>

這樣部署時的內存泄漏就解決了。

注意:

1) 添加這個Listener后,默認在tomcat關閉5s后jvm會進行內存回收的操作,具體時間設置可在下面的第三個參考鏈接中找到,所以,在關閉后的5s內,再次啟動tomcat,可能會存在問題,導致啟動無效(如果出現tomcat重啟后日志顯示正常但是服務器不工作的話考慮一下是不是這個問題)。

2)這個Listener只解決部署的內存泄漏,其他問題(如jdbc等)產生的內存泄漏還需要自己解決。

參考:

http://stackoverflow.com/questions/7788280/memory-leak-when-redeploying-application-in-tomcat#answer-36295683

http://java.jiderhamn.se/2011/12/11/classloader-leaks-i-how-to-find-classloader-leaks-with-eclipse-memory-analyser-mat/

https://github.com/mjiderhamn/classloader-leak-prevention

 

4,錯誤排除。

(1) 執行tomcat7:deploy顯示Build Success成功但是沒有效果,也沒有在本地生成war包,檢查一下maven配置文件中packaging標簽是否設置為war。即:

<packaging>war</packaging>

如果不是(比如說是pom),那么改成war應該就可以了。

(2) 如果出現在本地tomcat服務器自動部署沒有任何問題,部署到遠程服務器出現下面的Cannot invoke Tomcat manager: Connection reset by peer: socket write error 錯誤:

[ERROR] Failed to execute goal org.apache.tomcat.maven:tomcat7-maven-plugin:2.2:deploy (default-cli) on project webapp: Cannot invoke Tomcat manager: Connection reset by peer: socket write error -> [Help 1]
org.apache.maven.lifecycle.LifecycleExecutionException: Failed to execute goal org.apache.tomcat.maven:tomcat7-maven-plugin:2.2:deploy (default-cli) on project clyf_wechat: Cannot invoke Tomcat manager
    at org.apache.maven.lifecycle.internal.MojoExecutor.execute(MojoExecutor.java:212)
    at org.apache.maven.lifecycle.internal.MojoExecutor.execute(MojoExecutor.java:153)
    at org.apache.maven.lifecycle.internal.MojoExecutor.execute(MojoExecutor.java:145)
    at org.apache.maven.lifecycle.internal.LifecycleModuleBuilder.buildProject(LifecycleModuleBuilder.java:116)
    at org.apache.maven.lifecycle.internal.LifecycleModuleBuilder.buildProject(LifecycleModuleBuilder.java:80)
    at org.apache.maven.lifecycle.internal.builder.singlethreaded.SingleThreadedBuilder.build(SingleThreadedBuilder.java:51)
    at org.apache.maven.lifecycle.internal.LifecycleStarter.execute(LifecycleStarter.java:128)
    at org.apache.maven.DefaultMaven.doExecute(DefaultMaven.java:307)
    at org.apache.maven.DefaultMaven.doExecute(DefaultMaven.java:193)
    at org.apache.maven.DefaultMaven.execute(DefaultMaven.java:106)
    at org.apache.maven.cli.MavenCli.execute(MavenCli.java:863)
    at org.apache.maven.cli.MavenCli.doMain(MavenCli.java:288)
    at org.apache.maven.cli.MavenCli.main(MavenCli.java:199)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:497)
    at org.codehaus.plexus.classworlds.launcher.Launcher.launchEnhanced(Launcher.java:289)
    at org.codehaus.plexus.classworlds.launcher.Launcher.launch(Launcher.java:229)
    at org.codehaus.plexus.classworlds.launcher.Launcher.mainWithExitCode(Launcher.java:415)
    at org.codehaus.plexus.classworlds.launcher.Launcher.main(Launcher.java:356)
Caused by: org.apache.maven.plugin.MojoExecutionException: Cannot invoke Tomcat manager
    at org.apache.tomcat.maven.plugin.tomcat7.AbstractCatalinaMojo.execute(AbstractCatalinaMojo.java:141)
    at org.apache.tomcat.maven.plugin.tomcat7.AbstractWarCatalinaMojo.execute(AbstractWarCatalinaMojo.java:68)
    at org.apache.maven.plugin.DefaultBuildPluginManager.executeMojo(DefaultBuildPluginManager.java:134)
    at org.apache.maven.lifecycle.internal.MojoExecutor.execute(MojoExecutor.java:207)
    ... 20 more
Caused by: java.net.SocketException: Connection reset by peer: socket write error
    at java.net.SocketOutputStream.socketWrite0(Native Method)
    at java.net.SocketOutputStream.socketWrite(SocketOutputStream.java:109)
    at java.net.SocketOutputStream.write(SocketOutputStream.java:153)
    at org.apache.http.impl.io.AbstractSessionOutputBuffer.write(AbstractSessionOutputBuffer.java:181)
    at org.apache.http.impl.io.ContentLengthOutputStream.write(ContentLengthOutputStream.java:115)
    at org.apache.tomcat.maven.common.deployer.TomcatManager$RequestEntityImplementation.writeTo(TomcatManager.java:880)
    at org.apache.http.entity.HttpEntityWrapper.writeTo(HttpEntityWrapper.java:89)
    at org.apache.http.impl.client.EntityEnclosingRequestWrapper$EntityWrapper.writeTo(EntityEnclosingRequestWrapper.java:108)
    at org.apache.http.impl.entity.EntitySerializer.serialize(EntitySerializer.java:117)
    at org.apache.http.impl.AbstractHttpClientConnection.sendRequestEntity(AbstractHttpClientConnection.java:265)
    at org.apache.http.impl.conn.ManagedClientConnectionImpl.sendRequestEntity(ManagedClientConnectionImpl.java:203)
    at org.apache.http.protocol.HttpRequestExecutor.doSendRequest(HttpRequestExecutor.java:236)
    at org.apache.http.protocol.HttpRequestExecutor.execute(HttpRequestExecutor.java:121)
    at org.apache.http.impl.client.DefaultRequestDirector.tryExecute(DefaultRequestDirector.java:682)
    at org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:486)
    at org.apache.http.impl.client.AbstractHttpClient.doExecute(AbstractHttpClient.java:863)
    at org.apache.http.impl.client.CloseableHttpClient.execute(CloseableHttpClient.java:82)
    at org.apache.tomcat.maven.common.deployer.TomcatManager.invoke(TomcatManager.java:742)
    at org.apache.tomcat.maven.common.deployer.TomcatManager.deployImpl(TomcatManager.java:705)
    at org.apache.tomcat.maven.common.deployer.TomcatManager.deploy(TomcatManager.java:388)
    at org.apache.tomcat.maven.plugin.tomcat7.deploy.AbstractDeployWarMojo.deployWar(AbstractDeployWarMojo.java:85)
    at org.apache.tomcat.maven.plugin.tomcat7.deploy.AbstractDeployMojo.invokeManager(AbstractDeployMojo.java:82)
    at org.apache.tomcat.maven.plugin.tomcat7.AbstractCatalinaMojo.execute(AbstractCatalinaMojo.java:132)
    ... 23 more

使用mvn tomcat7:redeploy時出現如下情況

 

經過查詢Tomcat文檔后發現,這是由於Tomcat的遠程地址攔截器造成的結果,默認情況下,Tomcat的Manager和Host-Manager只接受本機的請求,而要讓它接受遠程的請求,需要添加上面提到的manager.xml的配置(第一步配置過了就不要加了),也就是:

<?xml version="1.0" encoding="UTF-8"?>
<Context privileged="true" antiResourceLocking="false"
         docBase="${catalina.home}/webapps/manager">
             <Valve className="org.apache.catalina.valves.RemoteAddrValve" allow="^.*$" />
</Context>

由於Tomcat的Manager可以執行項目的部署、卸載等敏感操作,如果你只想允許特定的IP地址訪問Manager,可在上面的allow屬性中設置規則。具體規則設置見下面的鏈接:

http://tomcat.apache.org/tomcat-7.0-doc/config/valve.html#Remote_Address_Filter

問題說明:http://tomcat.apache.org/tomcat-7.0-doc/manager-howto.html#Configuring_Manager_Application_Access

(3)自動部署顯示成功,war包也上傳成功,但是war不自動解壓自動部署。

如果你在tomcat的server.xml中通過設置<Context>標簽來部署相同名稱的項目的話,maven發布到該服務器的war不會被自動解壓,部署,更新,需要去掉server.xml中該項目的<Context>標簽。

 

 5. 其他

如果想要實現對部署路徑加版本,可將上面tomcat7-maven-pluginconfiguration的path設置為 

<path>/webapp#version</path>

的形式,部署后,當前項目在服務器端的路徑就是/webapp/version。舉個例子,如果path設置為 test#1.0,那么服務端項目實際的路徑就是/test/1.0。如下:

 

 

 

如果名字和版本號之間是兩個#,效果就是制定當前項目在manager網頁中顯示的版本,路徑不變,舉個例子,path設置為test##1.0,實際部署路徑為/test,但是在manager網頁中,顯示如下,注意Version一欄的值:

 

參考:

http://tomcat.apache.org/tomcat-7.0-doc/config/context.html#Naming


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM