使用jsch進行sftp傳輸時遇到的問題com.jcraft.jsch.JSchException: Session.connect: java.net.SocketException: Connection reset

来源:https://www.cnblogs.com/yjry-th/archive/2022/07/12/16470795.html
-Advertisement-
Play Games

在做某個業務時,需要將文件傳輸到另一臺伺服器,指定使用sftp方式;於是在網上找到jsch包使用,原先代碼大致如下: 1 ChannelSftp channelSftp = null; 2 try { 3 JSch jsch = new JSch(); 4 jsch.getSession("ftpU ...


在做某個業務時,需要將文件傳輸到另一臺伺服器,指定使用sftp方式;於是在網上找到jsch包使用,原先代碼大致如下:

 1 ChannelSftp channelSftp = null;
 2         try {
 3             JSch jsch = new JSch();
 4             jsch.getSession("ftpUserName", "ftpHost", 22);
 5             Session sshSession = jsch.getSession("ftpUserName", "ftpHost", 22);
 6             System.out.println("Session created.");
 7             sshSession.setPassword("ftpPassword");
 8             Properties sshConfig = new Properties();
 9             sshConfig.put("StrictHostKeyChecking", "no");
10             sshSession.setConfig(sshConfig);
11             sshSession.connect();
12             System.out.println("Session connected.");
13             System.out.println("Opening Channel.");
14             Channel channel = sshSession.openChannel("sftp");
15             channel.connect();
16             channelSftp = (ChannelSftp) channel;
17 
18             //todo 上傳文件
19         } catch (Exception e) {
20             //todo 異常處理
21         } finally {
22             //斷開sftp連接
23             if (channelSftp != null) {
24                 channelSftp.disconnect();
25             }
26         }

程式運行後大約過了幾天,發現日誌產生大量連接異常的日誌,主要是兩類異常:SocketException和NoRouteToHostException

com.jcraft.jsch.JSchException: Session.connect: java.net.SocketException: Connection reset
    at com.jcraft.jsch.Session.connect(Session.java:534)
    at com.jcraft.jsch.Session.connect(Session.java:162)
    at sun.reflect.GeneratedMethodAccessor37.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.springframework.scheduling.support.ScheduledMethodRunnable.run(ScheduledMethodRunnable.java:84)
    at org.springframework.scheduling.support.DelegatingErrorHandlingRunnable.run(DelegatingErrorHandlingRunnable.java:54)
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
    at java.util.concurrent.FutureTask.runAndReset(FutureTask.java:308)
    at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$301(ScheduledThreadPoolExecutor.java:180)
    at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:294)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)



com.jcraft.jsch.JSchException: java.net.NoRouteToHostException: Cannot assign requested address (Address not available) at com.jcraft.jsch.Util.createSocket(Util.java:344) at com.jcraft.jsch.Session.connect(Session.java:194) at com.jcraft.jsch.Session.connect(Session.java:162) at sun.reflect.GeneratedMethodAccessor37.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:498) at org.springframework.scheduling.support.ScheduledMethodRunnable.run(ScheduledMethodRunnable.java:84) at org.springframework.scheduling.support.DelegatingErrorHandlingRunnable.run(DelegatingErrorHandlingRunnable.java:54) at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511) at java.util.concurrent.FutureTask.runAndReset(FutureTask.java:308) at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$301(ScheduledThreadPoolExecutor.java:180) at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:294) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) at java.lang.Thread.run(Thread.java:748) Caused by: java.net.NoRouteToHostException: Cannot assign requested address (Address not available) at java.net.PlainSocketImpl.socketConnect(Native Method) at java.net.AbstractPlainSocketImpl.doConnect(AbstractPlainSocketImpl.java:350) at java.net.AbstractPlainSocketImpl.connectToAddress(AbstractPlainSocketImpl.java:206) at java.net.AbstractPlainSocketImpl.connect(AbstractPlainSocketImpl.java:188) at java.net.SocksSocketImpl.connect(SocksSocketImpl.java:392) at java.net.Socket.connect(Socket.java:589) at java.net.Socket.connect(Socket.java:538) at java.net.Socket.<init>(Socket.java:434) at java.net.Socket.<init>(Socket.java:211) at com.jcraft.jsch.Util.createSocket(Util.java:338) ... 17 common frames omitted
 

首先我百度了下,找到有人提到修改socket連接的配置,於是我嘗試修改:

1. 修改埠釋放後的等待時間為30s。echo 30 > /proc/sys/net/ipv4/tcp_fin_timeout

2. 修改/proc/sys/net/ipv4/tcp_tw_reuse為1。echo 1 > /proc/sys/net/ipv4/tcp_tw_reuse

3. 修改/proc/sys/net/ipv4/tcp_tw_recycle為1。echo 1 > /proc/sys/net/ipv4/tcp_tw_recycle

4.增加可用埠  vim /etc/sysctl.conf 

net.ipv4.ip_local_port_range = 10000     65000      -----意味著10000~65000埠可用 

改完後,執行命令“sysctl -p”使參數生效。

 

修改配置後,重啟程式,發現過了一天,仍然出現上述的問題;繼續百度查看,發現有人提到連接關閉的問題,於是查了下伺服器的ssh連接數

netstat |grep ssh |wc -l

這一查,發現有5500多個連接未關閉(隨著時間推移,空閑連接會自動關閉),於是問題的原因大致定位到了:程式生成連接的速度大於連接釋放的速度;但是問題的根源還是在於,程式沒有正確關閉連接。

不僅僅要斷開channel,還要斷開session。修改後的代碼如下:

 1 ChannelSftp channelSftp = null;
 2         try {
 3             JSch jsch = new JSch();
 4             jsch.getSession("ftpUserName", "ftpHost", 22);
 5             Session sshSession = jsch.getSession("ftpUserName", "ftpHost", 22);
 6             System.out.println("Session created.");
 7             sshSession.setPassword("ftpPassword");
 8             Properties sshConfig = new Properties();
 9             sshConfig.put("StrictHostKeyChecking", "no");
10             sshSession.setConfig(sshConfig);
11             sshSession.connect();
12             System.out.println("Session connected.");
13             System.out.println("Opening Channel.");
14             Channel channel = sshSession.openChannel("sftp");
15             channel.connect();
16             channelSftp = (ChannelSftp) channel;
17 
18             //todo 上傳文件
19         } catch (Exception e) {
20             //todo 異常處理
21         } finally {
22             //斷開sftp連接
23             if (channelSftp != null) {
24                 try {
25                     channelSftp.disconnect();
26                     //關閉會話
27                     Session session = channelSftp.getSession();
28                     if (session != null) {
29                         session.disconnect();
30                     }
31                 } catch (Exception e) {
32                     //todo 異常處理
33                 }
34             }
35         }

重新運行,觀察ssh連接數,發現恢復正常,每次文件傳輸完畢後,會話及時結束。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 《軟技能-代碼之外的生存指南》讀書筆記 寫在前面 最近項目相對鬆了一些,想靜下心來看一些書,買了些DDD的書,記得這本書也是程式員必讀的書之一,就湊單也買了紙質的來看看~ 抄錄一些覺得不錯或者有感觸的話。 抄錄 第一章 為何這本書與你先前讀過的任何數據都迥然不同 你所能犯的最大錯誤就是相信自己是在為 ...
  • 1、JDBC簡介 1.1、客戶端操作MySQL資料庫的方式 使用DOS命令行方式 使用第三方客戶端來訪問MySQL:SQLyog、Navicat、.... 通過程式來訪問MySQL資料庫 而通過Java來訪問MySQL資料庫,就是JDBC的概念 1.2、JDBC的概念 什麼是JDBC Java Da ...
  • 一、JVM的位置及體繫結構 JVM作用在操作系統之上,而Java程式作用在jvm之上,其他的程式則與jvm併列 二、類載入器,及雙親委派機制 1.類載入器 作用:載入Class文件 -> new Student();實例的引用放在棧里,具體的對象放在堆里 點擊查看代碼 package com.Tan ...
  • 一、Frame視窗 點擊查看代碼 package com.Tang.gui; import java.awt.*; public class TestFrame1 { public static void main(String[] args) { MyFrame myFrame1 = new My ...
  • 目錄 一、python的字元 二、python的類型 1.python中數據類型 2.首先是數字類型的整數類型 3.浮點類型 4.複數類型 5.布爾類型 三、python運算符 1.算數運算符 2.比較運算符 3.邏輯運算符 4.位運算符 5.賦值運算符 四、python常用語句 1.分支語句 2. ...
  • 1 記憶體結構 1、簡述一下JVM的記憶體結構?(高頻) JVM在執行Java程式時,會把它管理的記憶體劃分為若幹個的區域,每個區域都有自己的用途和創建銷毀時間。如下圖所示,可以分為兩大部分,線程私有區和共用區。 線程私有區: ① 程式計數器 作用:是一塊較小的記憶體空間,可以理解為是當前線程所執行程式的字 ...
  • 一、IO流概述 1.原理 ![](https://img2022.cnblogs.com/blog/2901531/202206/2901531-20220621172751004-1385246087.png) 2.流的分類 3.流的體系,藍底框為重點掌握的 二、IO流操作 1.節點流-字元流 ( ...
  • 面向對象(上) java面向對象學習三條主線 * 1.java類及類的成員:屬性、方法、構造器、代碼塊、內部類 * 2.面向對象的三大特征:封裝,繼承,多態 * 3.其他關鍵字:this,super,static,final,abstract,interface,package,import等 面向 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...