卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章34766本站已运行393

从 golang 通过代理连接到 ftp

从 golang 通过代理连接到 ftp

php小编子墨为您介绍一种方法,可以通过使用代理连接到FTP服务器的Golang代码。在某些情况下,我们可能需要在访问FTP服务器时使用代理服务器,以便提高安全性或解决网络限制问题。Golang是一种强大的编程语言,具有简单易用的网络库,因此我们可以很容易地实现这个功能。接下来,我们将通过一个简单的示例代码来演示如何使用Golang通过代理连接到FTP服务器。

问题内容

我需要通过代理连接到 ftp 服务器并从 Golang 代码下载文件。我已经实现了简单的客户端:

package main

import (
    "bufio"
    "fmt"
    "github.com/jlaffaye/ftp"
    "io"
    "log"
    "net"
    "net/http"
    "os"
    "strings"
    "time"
)

var targetAddr = "ftp.dlptest.com"
var proxyAddr = "146.19.106.109:3128"
var user = "dlpuser"
var pass = "rNrKYTX9g7z3RgJRmxWuGHbeu"
var path = "/xxx.txt"

func connectViaProxy(network, address string) (net.Conn, error) {

    parts := strings.Split(address, ":")
    addr := fmt.Sprintf("%s:%s", targetAddr, parts[1])

    proxyConn, err := net.Dial(network, proxyAddr)
    if err != nil {
        return nil, err
    }

    req, err := http.NewRequest(http.MethodConnect, "http://"+addr, nil)
    if err != nil {
        proxyConn.Close()
        return nil, err
    }

    req.Header.Set("Accept", "*/*")
    req.Header.Set("User-Agent", "curl/8.1.2")
    req.Header.Set("Host", targetAddr+":21")

    err = req.Write(proxyConn)
    if err != nil {
        return nil, err
    }

    resp, err := http.ReadResponse(bufio.NewReader(proxyConn), req)
    if err != nil {
        proxyConn.Close()
        return nil, err
    }
    if resp.StatusCode != 200 {

        x, _ := io.ReadAll(resp.Body)
        fmt.Println(string(x))

        proxyConn.Close()
        return nil, fmt.Errorf("non-200 status code received from proxy: %v", resp.Status)
    }

    return proxyConn, nil
}

func main() {
    c, err := ftp.Dial(targetAddr+":21",
        ftp.DialWithTimeout(5*time.Second),
        ftp.DialWithDebugOutput(os.Stdin),
        ftp.DialWithDialFunc(connectViaProxy),
    )
    if err != nil {
        log.Fatal(err)
    }

    err = c.Login(user, pass)
    if err != nil {
        log.Fatal(err)
    }

    if r, err := c.Retr(path); err != nil {
        fmt.Println(err, ":<")
    } else {
        if reader, err := io.ReadAll(r); err != nil {
            fmt.Println(err, ":(")
        } else {
            fmt.Println(string(reader))
        }
    }

    if err = c.Quit(); err != nil {
        log.Fatal(err)
    }
}

不幸的是,我在尝试连接时收到 403 Forbidden。通常我会假设这是服务器限制,但通过curl它可以完全正常工作:

curl -vvv -x 146.19.106.109:3128 -u dlpuser:rNrKYTX9g7z3RgJRmxWuGHbeu ftp://ftp.dlptest.com/xxx.txt

有人知道为什么会发生这种情况吗?据我所知,curl 也在底层使用 http CONNECT,所以我想知道我错过了什么。

附注不用担心代码中的凭据 - 这些凭据可公开用于网络测试

解决方法

HTTP 代理上的 FTP 不是通过 CONNECT 请求完成的。相反,代理被指示使用普通的 GET 请求获取 ftp://... URL。这也是curl正在做的事情。在 Go 中,可以这样完成(为了简单起见,省略错误处理):

// create the TCP connection to the proxy
conn, err := net.Dial("tcp", "146.19.106.109:3128")

// create the request with the ftp:// URL
req, err := http.NewRequest(http.MethodGet, "ftp://ftp.dlptest.com/xxx.txt", nil)
req.SetBasicAuth("dlpuser","rNrKYTX9g7z3RgJRmxWuGHbeu")

// don't use req.Write but req.WriteProxy, since the request line must
// contain the full URL and not only the path
err = req.WriteProxy(conn)

// read response
resp, err := http.ReadResponse(bufio.NewReader(conn), req)
body, _ := io.ReadAll(resp.Body)
卓越飞翔博客
上一篇: 如何让golang区分带命名空间和不带命名空间的XML元素?
下一篇: 返回列表
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏