curl Unsupported Request Method and Protocol

curl的奇特問題

最近在使用curl的時候發(fā)現(xiàn)一個奇怪的問題。使用curl訪問sftp服務器時總是報錯心例。我執(zhí)行的命令如下:

curl -u sftptest:passwd sftp://127.0.0.1:22/home/sftptest/

然后返回了一大篇html, 主要的信息如下:

...
<title>ERROR: The requested URL could not be retrieved</title>
...
<p><b>Unsupported Request Method and Protocol</b></p>

首先需要確認的是sftp是否在curl中開啟了。執(zhí)行curl -V發(fā)現(xiàn)確實支持curl. 接下來打開verbose模式弦追,發(fā)現(xiàn)了一些有趣的信息:

*   Trying 172.17.10.80...
* TCP_NODELAY set
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed

  0     0    0     0    0     0      0      0 --:--:-- --:--:-- --:--:--     0* Connected to (nil) (172.17.18.84) port 8080 (#0)
* Server auth using Basic with user 'sftptest'
> GET sftp://sftptest:password@127.0.0.1/home/sftptest HTTP/1.1
> Host: 127.0.0.1:22
> Authorization: Basic c2Z0cHRlc3Q6MXEydzNlNHI=
> User-Agent: curl/7.19.7 (x86_64-redhat-linux-gnu) libcurl/7.52.1 OpenSSL/1.1.0e libssh2/1.7.0_DEV
> Accept: */*
> Proxy-Connection: Keep-Alive
> 
< HTTP/1.1 400 Bad Request
< Server: nps/2.3.1
< Mime-Version: 1.0
< Date: Wed, 25 Oct 2017 10:03:23 GMT
< Content-Type: text/html
< Content-Length: 4205
< X-Squid-Error: ERR_INVALID_URL 0
< Vary: Accept-Language
< Content-Language: en
< X-Cache: MISS from netentsec-nps-172.17.18.84
< Connection: close
< 
{ [data not shown]
* Curl_http_done: called premature == 0

100  4205  100  4205    0     0  29450      0 --:--:-- --:--:-- --:--:-- 29822
* Closing connection 0

有意思的是吴趴,curl 連接上我的代理服務器后释涛,使用http協(xié)議去訪問sftp://127.0.0.1. 所以我們會得到Bad request (400)的響應涝婉。

如果我把127.0.0..1從代理中排除(e.g. export no_proxy=127.0.0.1)則請求就可以成功了哥力。

挖掘真相

為什么在連接代理后,curl會使用http協(xié)議去訪問sftp服務器呢? 我在curl的repository中創(chuàng)建了issue. 從官方的回答中得到了答案吩跋。因為我正在使用老版本的curl(7.52), 這個版本不支持自動轉換隧道協(xié)議(tunneling). 在7.55.0版本后才支持這種自動轉換寞射。摘自issue的回復

Since 7.55.0, curl will enable tunneling automatically when you try to use SFTP over an HTTP proxy.

所以如果使用最新的curl, 這個問題應該就能解決了。接下來嘗試下載和編譯了最新的curl(7.56), 但是不幸的是仍然不能成功锌钮,并且返回了另一個錯誤:

* STATE: INIT => CONNECT handle 0x1a42ec8; line 1425 (connection #-5000)
* Added connection 0. The cache now contains 1 members
*   Trying 172.17.18.84...
* TCP_NODELAY set
* STATE: CONNECT => WAITCONNECT handle 0x1a42ec8; line 1477 (connection #0)
* Connected to 172.17.18.84 (172.17.18.84) port 8080 (#0)
* STATE: WAITCONNECT => WAITPROXYCONNECT handle 0x1a42ec8; line 1594 (connection #0)
* Marked for [keep alive]: HTTP default
* allocate connect buffer!
* Establish HTTP proxy tunnel to 127.0.0.1:22
* Server auth using Basic with user 'sftptest'
> CONNECT 127.0.0.1:22 HTTP/1.1
> Host: 127.0.0.1:22
> User-Agent: curl/7.57.0-DEV
> Proxy-Connection: Keep-Alive
> 
< HTTP/1.1 503 Service Unavailable
< Server: nps/2.3.1
< Mime-Version: 1.0
< Date: Thu, 26 Oct 2017 07:09:25 GMT
< Content-Type: text/html
< Content-Length: 4069
< X-Squid-Error: ERR_CONNECT_FAIL 111
< Vary: Accept-Language
< Content-Language: en

注意到新的錯誤是503 Service Unavailable. 為什么會這樣呢桥温?我們可以一步一步分析原因。

首先轧粟,可以輕易發(fā)現(xiàn)連接代理服務器是成功的:

*   Trying 172.17.18.84...
* TCP_NODELAY set
* STATE: CONNECT => WAITCONNECT handle 0x1a42ec8; line 1477 (connection #0)
* Connected to 172.17.18.84 (172.17.18.84) port 8080 (#0)
* STATE: WAITCONNECT => WAITPROXYCONNECT handle 0x1a42ec8; line 1594 (connection #0)
* Marked for [keep alive]: HTTP default
* allocate connect buffer!
* Establish HTTP proxy tunnel to 127.0.0.1:22

Tcp連接的狀態(tài)變化為: CONNECT => WAITCONNECT => WAITPROXYCONNECT => Establish HTTP proxy tunnel. 這個狀態(tài)變化顯然是正確的策治,代表連接代理服務器成功脓魏。

接下來嘗試連接sftp服務器:

* Server auth using Basic with user 'sftptest'
> CONNECT 127.0.0.1:22 HTTP/1.1
> Host: 127.0.0.1:22
> User-Agent: curl/7.57.0-DEV
> Proxy-Connection: Keep-Alive

但是得到的響應卻是:

< HTTP/1.1 503 Service Unavailable

也就是說兰吟,從代理服務器發(fā)起sftp連接到sftp服務器失敗 - 并且是sftp服務器服務不可用。這下真相逐漸浮出水面: 因為代理服務器嘗試去連接127.0.0.1:22, 但是這個地址對于代理服務器而言茂翔,是連接自己的22端口混蔼,而不是去連接sftp服務器!如果把這里的127.0.0.1替換為sftp服務器的內網地址珊燎,則可以成功:

* STATE: INIT => CONNECT handle 0x2238ec8; line 1425 (connection #-5000)
* Rebuilt URL to: sftp://sftptest@172.26.9.94/
* Added connection 0. The cache now contains 1 members
*   Trying 172.17.18.84...
* TCP_NODELAY set
* STATE: CONNECT => WAITCONNECT handle 0x2238ec8; line 1477 (connection #0)
* Connected to 172.17.18.84 (172.17.18.84) port 8080 (#0)
* STATE: WAITCONNECT => WAITPROXYCONNECT handle 0x2238ec8; line 1594 (connection #0)
* Marked for [keep alive]: HTTP default
* allocate connect buffer!
* Establish HTTP proxy tunnel to 172.26.9.94:22
* Server auth using Basic with user 'sftptest'
> CONNECT 172.26.9.94:22 HTTP/1.1
> Host: 172.26.9.94:22
> User-Agent: curl/7.57.0-DEV
> Proxy-Connection: Keep-Alive
> 
< HTTP/1.1 200 Connection established
< 
* Proxy replied 200 to CONNECT request
* CONNECT phase completed!
* STATE: WAITPROXYCONNECT => SENDPROTOCONNECT handle 0x2238ec8; line 1573 (connection #0)
* CONNECT phase completed!
* SFTP 0x223f170 state change from SSH_STOP to SSH_INIT
* SFTP 0x223f170 state change from SSH_INIT to SSH_S_STARTUP
* STATE: SENDPROTOCONNECT => PROTOCONNECT handle 0x2238ec8; line 1608 (connection #0)
* SFTP 0x223f170 state change from SSH_S_STARTUP to SSH_HOSTKEY
* SSH MD5 fingerprint: 4cf03683e054a3398c91d76a16715e6b
* SSH host check: 2, key: <none>
* SFTP 0x223f170 state change from SSH_HOSTKEY to SSH_SESSION_FREE
* Marked for [closure]: SSH session free
* SFTP 0x223f170 state change from SSH_SESSION_FREE to SSH_STOP
* multi_done
* SSH DISCONNECT starts now
* SSH DISCONNECT is done
* Closing connection 0
* The cache now contains 0 members

結論

要解決curl使用sftp訪問127.0.0.1或localhost的問題惭嚣,可以采用:

  • 方法1 (推薦): 把127.0.0.1或localhost加入no_proxy. 這樣無論使用哪個版本的curl, 都能正常工作。
  • 方法2: 目標地址使用內網ip地址悔政,而不要使用127.0.0.1, 并且保證curl的版本在7.55以上晚吞。 但是這樣做的話,curl會先連接代理服務器谋国,再發(fā)起sftp請求槽地,性能會有損耗。
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末芦瘾,一起剝皮案震驚了整個濱河市捌蚊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌近弟,老刑警劉巖缅糟,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異祷愉,居然都是意外死亡窗宦,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門二鳄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來赴涵,“玉大人,你說我怎么就攤上這事泥从【湔迹” “怎么了?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵躯嫉,是天一觀的道長纱烘。 經常有香客問我杨拐,道長,這世上最難降的妖魔是什么擂啥? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任哄陶,我火速辦了婚禮,結果婚禮上哺壶,老公的妹妹穿的比我還像新娘屋吨。我一直安慰自己,他們只是感情好山宾,可當我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布至扰。 她就那樣靜靜地躺著,像睡著了一般资锰。 火紅的嫁衣襯著肌膚如雪敢课。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天绷杜,我揣著相機與錄音直秆,去河邊找鬼。 笑死鞭盟,一個胖子當著我的面吹牛圾结,可吹牛的內容都是我干的。 我是一名探鬼主播齿诉,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼筝野,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了鹃两?” 一聲冷哼從身側響起遗座,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎俊扳,沒想到半個月后途蒋,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡馋记,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年号坡,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片梯醒。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡宽堆,死狀恐怖,靈堂內的尸體忽然破棺而出茸习,到底是詐尸還是另有隱情畜隶,我是刑警寧澤,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站籽慢,受9級特大地震影響浸遗,放射性物質發(fā)生泄漏。R本人自食惡果不足惜箱亿,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一跛锌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧届惋,春花似錦髓帽、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至晨缴,卻和暖如春译秦,著一層夾襖步出監(jiān)牢的瞬間峡捡,已是汗流浹背击碗。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留们拙,地道東北人稍途。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像砚婆,于是被迫代替她去往敵國和親械拍。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,864評論 2 354

推薦閱讀更多精彩內容

  • Spring Cloud為開發(fā)人員提供了快速構建分布式系統(tǒng)中一些常見模式的工具(例如配置管理装盯,服務發(fā)現(xiàn)坷虑,斷路器,智...
    卡卡羅2017閱讀 134,656評論 18 139
  • 國家電網公司企業(yè)標準(Q/GDW)- 面向對象的用電信息數(shù)據交換協(xié)議 - 報批稿:20170802 前言: 排版 ...
    庭說閱讀 10,967評論 6 13
  • 轉載自:http://www.cnblogs.com/txw1958/archive/2013/01/19/286...
    php_bruce閱讀 2,267評論 1 5
  • 一埂奈、什么是CURL迄损? cURL 是一個利用URL語法規(guī)定來傳輸文件和數(shù)據的工具,支持很多協(xié)議账磺,如HTTP芹敌、FTP、...
    茶藝瑤閱讀 4,602評論 0 6
  • 一垮抗、什么是CURL氏捞? cURL 是一個利用URL語法規(guī)定來傳輸文件和數(shù)據的工具,支持很多協(xié)議冒版,如HTTP液茎、FTP、...
    伊Summer閱讀 1,268評論 0 4