亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 服務器 > Web服務器 > 正文

詳解如何通過tomcat的ManagerServlet遠程部署項目

2024-09-01 13:55:24
字體:
來源:轉載
供稿:網友

介紹

之前在郵政實習時,leader讓我閱讀tomcat的源代碼,嘗試自己實現遠程部署項目的功能,于是便有了這此實踐。
在Tomact中有一個Manager應用程序,它是用來管理已經部署的web應用程序,在這個應用程序中,ManagerServlet是他的主servlet,通過它我們可以獲取tomcat的部分指標,遠程管理web應用程序,不過這個功能會受到web應用程序部署中安全約束的保護。

當你請求ManagerServlet時,它會檢查getPathInfo()返回的值以及相關的查詢參數,以確定被請求的操作。它支持以下操作和參數(從servlet路徑開始): 

 

請求路徑 描述
/deploy?config={config-url} 根據指定的path部署并啟動一個新的web應用程序(詳見源碼)
/deploy?config={config-url}&war={war-url}/ 根據指定的pat部署并啟動一個新的web應用程序(詳見源碼)
/deploy?path=/xxx&war={war-url} 根據指定的path部署并啟動一個新的web應用程序(詳見源碼)
/list 列出所有web應用程序的上下文路徑。格式為path:status:sessions(活動會話數)
/reload?path=/xxx 根據指定path重新加載web應用
/resources?type=xxxx 枚舉可用的全局JNDI資源,可以限制指定的java類名
/serverinfo 顯示系統信息和JVM信息
/sessions 此方法已過期
/expire?path=/xxx 列出path路徑下的web應用的session空閑時間信息
/expire?path=/xxx&idle=mm Expire sessions for the context path /xxx which were idle for at least mm minutes.
/sslConnectorCiphers 顯示當前connector配置的SSL/TLS密碼的診斷信息
/start?path=/xx 根據指定path啟動web應用程序
/stop?path=/xxx 根據指定path關閉web應用程序
/threaddump Write a JVM thread dump
/undeploy?path=/xxx 關閉并刪除指定path的Web應用程序,然后刪除底層WAR文件或文檔基目錄。

 

我們可以通過ManagerServlet中getPathInfo()提供的操作,將自己的項目遠程部署到服務器上,下面將貼出我的實踐代碼,在實踐它之前你只需要引入httpclient包和commons包。

封裝統一的遠程請求管理類

封裝此類用于方便client請求ManagerServlet:

import java.io.File;import java.net.URL;import java.net.URLEncoder;import org.apache.commons.io.IOUtils;import org.apache.commons.lang.StringUtils;import org.apache.http.Header;import org.apache.http.HttpHost;import org.apache.http.HttpResponse;import org.apache.http.HttpStatus;import org.apache.http.auth.AuthScope;import org.apache.http.auth.Credentials;import org.apache.http.auth.UsernamePasswordCredentials;import org.apache.http.client.AuthCache;import org.apache.http.client.methods.HttpGet;import org.apache.http.client.methods.HttpRequestBase;import org.apache.http.client.protocol.ClientContext;import org.apache.http.impl.auth.BasicScheme;import org.apache.http.impl.client.BasicAuthCache;import org.apache.http.impl.client.DefaultHttpClient;import org.apache.http.impl.conn.PoolingClientConnectionManager;import org.apache.http.protocol.BasicHttpContext;public class TomcatManager {  private static final String MANAGER_CHARSET = "UTF-8";  private String username;  private URL url;  private String password;  private String charset;  private boolean verbose;  private DefaultHttpClient httpClient;  private BasicHttpContext localContext;  /** constructor */  public TomcatManager(URL url, String username) {    this(url, username, "");  }  public TomcatManager(URL url, String username, String password) {    this(url, username, password, "ISO-8859-1");  }  public TomcatManager(URL url, String username, String password, String charset) {    this(url, username, password, charset, true);  }  public TomcatManager(URL url, String username, String password, String charset, boolean verbose) {    this.url = url;    this.username = username;    this.password = password;    this.charset = charset;    this.verbose = verbose;        // 創建網絡請求相關的配置    PoolingClientConnectionManager poolingClientConnectionManager = new PoolingClientConnectionManager();    poolingClientConnectionManager.setMaxTotal(5);    this.httpClient = new DefaultHttpClient(poolingClientConnectionManager);    if (StringUtils.isNotEmpty(username)) {      Credentials creds = new UsernamePasswordCredentials(username, password);      String host = url.getHost();      int port = url.getPort() > -1 ? url.getPort() : AuthScope.ANY_PORT;      httpClient.getCredentialsProvider().setCredentials(new AuthScope(host, port), creds);      AuthCache authCache = new BasicAuthCache();      BasicScheme basicAuth = new BasicScheme();      HttpHost targetHost = new HttpHost(url.getHost(), url.getPort(), url.getProtocol());      authCache.put(targetHost, basicAuth);      localContext = new BasicHttpContext();      localContext.setAttribute(ClientContext.AUTH_CACHE, authCache);    }  }  /** 根據指定的path部署并啟動一個新的應用程序 */  public TomcatManagerResponse deploy(String path, File war, boolean update) throws Exception {    StringBuilder buffer = new StringBuilder("/deploy");    buffer.append("?path=").append(URLEncoder.encode(path, charset));    if (war != null) {      buffer.append("&war=").append(URLEncoder.encode(war.toString(), charset));    }    if (update) {      buffer.append("&update=true");    }    return invoke(buffer.toString());  }  /** 獲取所有已部署的web應用程序的上下文路徑。格式為path:status:sessions(活動會話數) */  public TomcatManagerResponse list() throws Exception {    StringBuilder buffer = new StringBuilder("/list");    return invoke(buffer.toString());  }  /** 獲取系統信息和JVM信息 */  public TomcatManagerResponse serverinfo() throws Exception {    StringBuilder buffer = new StringBuilder("/serverinfo");    return invoke(buffer.toString());  }  /** 真正發送請求的方法 */  private TomcatManagerResponse invoke(String path) throws Exception {    HttpRequestBase httpRequestBase = new HttpGet(url + path);    HttpResponse response = httpClient.execute(httpRequestBase, localContext);    int statusCode = response.getStatusLine().getStatusCode();    switch (statusCode) {      case HttpStatus.SC_OK: // 200      case HttpStatus.SC_CREATED: // 201      case HttpStatus.SC_ACCEPTED: // 202        break;      case HttpStatus.SC_MOVED_PERMANENTLY: // 301      case HttpStatus.SC_MOVED_TEMPORARILY: // 302      case HttpStatus.SC_SEE_OTHER: // 303      String redirectUrl = getRedirectUrl(response);      this.url = new URL(redirectUrl);      return invoke(path);    }    return new TomcatManagerResponse().setStatusCode(response.getStatusLine().getStatusCode())        .setReasonPhrase(response.getStatusLine().getReasonPhrase())        .setHttpResponseBody(IOUtils.toString(response.getEntity().getContent()));  }    /** 提取重定向URL */  protected String getRedirectUrl(HttpResponse response) {    Header locationHeader = response.getFirstHeader("Location");    String locationField = locationHeader.getValue();    // is it a relative Location or a full ?    return locationField.startsWith("http") ? locationField : url.toString() + '/' + locationField;  }}

封裝響應結果集

@Datapublic class TomcatManagerResponse {  private int statusCode;  private String reasonPhrase;  private String httpResponseBody;}

測試遠程部署

在測試之前請先在配置文件放通下面用戶權限:

<role rolename="admin-gui"/><role rolename="admin-script"/><role rolename="manager-gui"/><role rolename="manager-script"/><role rolename="manager-jmx"/><role rolename="manager-status"/><user username="sqdyy" password="123456" roles="manager-gui,manager-script,manager-jmx,manager-status,admin-script,admin-gui"/>

下面是測試成功遠程部署war包的代碼:

import static org.testng.AssertJUnit.assertEquals;import java.io.File;import java.net.URL;import org.testng.annotations.Test;public class TestTomcatManager {  @Test  public void testDeploy() throws Exception {    TomcatManager tm = new TomcatManager(new URL("http://localhost:8080/manager/text"), "sqdyy", "123456");    File war = new File("E://tomcat//simple-war-project-1.0-SNAPSHOT.war");    TomcatManagerResponse response = tm.deploy("/simple-war-project-1.0-SNAPSHOT", war, true);    System.out.println(response.getHttpResponseBody());    assertEquals(200, response.getStatusCode());        // output:    // OK - Deployed application at context path /simple-war-project-1.0-SNAPSHOT  }  @Test  public void testList() throws Exception {    TomcatManager tm = new TomcatManager(new URL("http://localhost:8080/manager/text"), "sqdyy", "123456");    TomcatManagerResponse response = tm.list();    System.out.println(response.getHttpResponseBody());    assertEquals(200, response.getStatusCode());        // output:    // OK - Listed applications for virtual host localhost    // /:running:0:ROOT    // /simple-war-project-1.0-SNAPSHOT:running:0:simple-war-project-1.0-SNAPSHOT    // /examples:running:0:examples    // /host-manager:running:0:host-manager    // /manager:running:0:manager    // /docs:running:0:docs  }  @Test  public void testServerinfo() throws Exception {    TomcatManager tm = new TomcatManager(new URL("http://localhost:8080/manager/text"), "sqdyy", "123456");    TomcatManagerResponse response = tm.serverinfo();    System.out.println(response.getHttpResponseBody());    assertEquals(200, response.getStatusCode());        // output:    // OK - Server info    // Tomcat Version: Apache Tomcat/7.0.82    // OS Name: Windows 10    // OS Version: 10.0    // OS Architecture: amd64    // JVM Version: 1.8.0_144-b01    // JVM Vendor: Oracle Corporation  }}

參考資料

ManagerServlet 源碼地址

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到服務器教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产成人涩涩涩视频在线观看| 欧美激情精品久久久久| 91国产在线精品| 国产精品视频导航| 国产精品久久久久久久久久小说| 成人福利网站在线观看11| 国产成人精品视频| 操人视频在线观看欧美| 亚洲人在线观看| 亚洲成人久久网| 在线观看欧美日韩| 国产精品亚洲视频在线观看| 国产精品亚洲美女av网站| 欧美激情乱人伦| 精品久久久久久久大神国产| 亚洲第一精品夜夜躁人人爽| 日韩视频在线免费| 欧美极品美女电影一区| 欧美电影在线观看网站| 久久久国产视频| 久久久久久久一区二区| 日韩国产欧美精品一区二区三区| 中文字幕精品视频| 成人午夜两性视频| 国产精品黄色av| 亚洲欧美日韩精品久久| 亚洲精品永久免费精品| 久久全球大尺度高清视频| 欧美黑人一级爽快片淫片高清| 国产美女精彩久久| 欧美成人手机在线| 久久在线免费视频| 日本一区二区三区四区视频| 久久精品中文字幕一区| 91在线精品视频| 5566日本婷婷色中文字幕97| 久久久久北条麻妃免费看| 韩国三级电影久久久久久| 国产精品久久久久一区二区| 国模精品视频一区二区| 中文字幕日本欧美| 午夜精品福利视频| 国产亚洲成av人片在线观看桃| 久久精品91久久久久久再现| 好吊成人免视频| 亚洲精品一区二区久| 亚洲在线视频观看| 亚洲3p在线观看| 精品国产一区二区三区在线观看| 日韩欧美中文免费| 国产一区私人高清影院| 国产精品视频永久免费播放| 日韩欧美视频一区二区三区| 日韩av免费看| 色综合视频网站| 九九视频这里只有精品| 久久久久久久久久久免费精品| 国产一区二区在线免费视频| 久久人人97超碰精品888| 国产一区二区黑人欧美xxxx| 91精品中文在线| 国产在线精品一区免费香蕉| 亚洲va久久久噜噜噜久久天堂| 亚州成人av在线| 精品日韩视频在线观看| 久久艹在线视频| 日韩精品高清视频| 成人激情视频在线播放| 国产欧美日韩最新| 国产精品国产自产拍高清av水多| 色天天综合狠狠色| 热久久视久久精品18亚洲精品| 欧美激情按摩在线| 精品中文字幕在线观看| 亚洲天堂成人在线| 国产精品羞羞答答| 欧美综合在线观看| 欧美性猛交xxxx免费看漫画| 亚洲精品综合精品自拍| 久久韩剧网电视剧| 成人午夜在线视频一区| 激情成人中文字幕| 国外视频精品毛片| 欧美高清在线视频观看不卡| 黄色一区二区在线| 都市激情亚洲色图| 情事1991在线| 国产精品久久久久久久久免费看| 国产99久久精品一区二区| 琪琪亚洲精品午夜在线| 欧美黑人巨大精品一区二区| 久久久久久久av| 欧美最近摘花xxxx摘花| 久久精品国产一区二区三区| 欧美激情视频网址| 欧美激情精品久久久久| 欧美日韩激情视频| 啪一啪鲁一鲁2019在线视频| 亚洲va电影大全| 一道本无吗dⅴd在线播放一区| 日韩视频一区在线| 日韩中文字幕免费看| 国产日韩欧美夫妻视频在线观看| 久久人人爽人人爽人人片av高请| 欧美国产精品日韩| 7777精品视频| 欧美第一黄网免费网站| 97久久精品在线| 国产精品免费在线免费| 欧日韩不卡在线视频| 久久久久久久久国产精品| www.日韩免费| 国内精品久久久久久| 成人精品久久一区二区三区| 57pao国产精品一区| 久久成人精品电影| 亚洲久久久久久久久久久| 91影院在线免费观看视频| 久久亚洲电影天堂| 久久久免费高清电视剧观看| 国产精品成久久久久三级| 欧美精品免费播放| 中文字幕av一区| 日韩av影片在线观看| 亚洲精品美女久久| 久久视频在线免费观看| 国产精品av免费在线观看| 色樱桃影院亚洲精品影院| 欧美成人精品一区| 亚洲天天在线日亚洲洲精| 粉嫩老牛aⅴ一区二区三区| 欧美放荡办公室videos4k| 国产精品在线看| 精品国产乱码久久久久久婷婷| 一本色道久久综合狠狠躁篇怎么玩| 日韩av片免费在线观看| 久久色免费在线视频| 96sao精品视频在线观看| 久久国产天堂福利天堂| 欧美裸体xxxx极品少妇| 久久视频在线视频| 亚洲精品美女在线观看播放| 亚洲激情在线视频| 国产亚洲精品一区二区| 欧美日韩国产中字| 中文字幕亚洲国产| 亚洲性av网站| 亚洲在线观看视频网站| 日韩中文av在线| 欧美午夜精品在线| 久久精品国产亚洲精品2020| 国产女人精品视频| 国产裸体写真av一区二区| 亚洲成人网久久久| 欧美老少配视频| 中文字幕不卡av| 亚洲欧美资源在线| 日韩精品黄色网| 91po在线观看91精品国产性色| 国产日韩中文字幕| 一本大道久久加勒比香蕉| 久久偷看各类女兵18女厕嘘嘘| 91精品国产777在线观看| 亚洲欧美一区二区精品久久久|