425 无法在我的程序上打开数据连接,而 Filezilla 工作

425 Can't open data connection on my program whereas Filezilla work

本文关键字:连接 数据 工作 Filezilla 我的 程序上      更新时间:2023-10-16

我尝试制作一个FTP客户端,该客户端在活动模式下上传文件(使用linux的套接字库的C++)。当我想在本地服务器(LAN)上发送文件时,我的程序可以工作,但它当我想在远程服务器(WAN)上发送文件时,它不起作用。我有错误425无法打开数据连接。我做了很多研究一般来说,这个问题来自NAT(可能会被阻止数据连接端口)。但我尝试过Filezilla(在活动模式下),它成功了。所以我没有发现数据连接的问题在哪里。

这是我的代码:

using namespace std;
int sock;
int sock_data,new_sock;
struct sockaddr_in sin_data;
struct sockaddr_in cli_addr = {0};
char* receiving(int socket)
{
    char message[1000] = {'/0'};
    recv(socket,message,sizeof(message),0);
    cout << message << endl;
    return message;
}
void *send_data(void* args)
{
    send(sock,"STOR fichier.txtrn",sizeof("STOR fichier.txtrn"),0);
    receiving(sock);
    return(0);
}
void *accept_data_connexion(void* args)
{
    sock_data = socket(AF_INET,SOCK_STREAM,6);
    sin_data.sin_family = AF_INET;
    sin_data.sin_port = htons(58703);
    sin_data.sin_addr.s_addr = INADDR_ANY;
    socklen_t clilen = sizeof(cli_addr);
    new_sock = accept(sock_data,(struct sockaddr *)&cli_addr,&clilen);
    if(errno == 0)
        {
        cout << "Data connection is established" << endl;
        }
}
int main()
{
    struct sockaddr_in sin;
    sin.sin_family = AF_INET;
    sin.sin_port = htons(21);
    sin.sin_addr.s_addr = inet_addr("xxx,xxx,xxx,xxx");
    sock = socket(AF_INET,SOCK_STREAM,6);
    //connection
    connect(sock,(struct sockaddr *)&sin,sizeof(sin));
    receiving(sock);
    receiving(sock);
    print_box();
    //log in
    send(sock,"USER usernamern",sizeof("USER usernamern"),0);
    receiving(sock);
    send(sock,"PASS passwordrn",sizeof("PASS passwordrn"),0);
    receiving(sock);
    send(sock,"SYSTrn",sizeof("SYSTrn"),0);
    receiving(sock);
    send(sock,"PWDrn",sizeof("PWDrn"),0);
    receiving(sock);
    send(sock,"PORT xxx,xxx,xxx,xxx,229,79",sizeof("PORT xxx,xxx,xxx,xxx,229,79"),0);
    receiving(sock);
    send(sock,"TYPE I rn",sizeof("TYPE I rn"),0);
    receiving(sock);
    pthread_t th1, th2;
    void *ret;
    pthread_create (&th1,NULL, accept_data_connexion,(void*)1);
    pthread_create (&th2, NULL, send_data,(void *)2);
    (void)pthread_join (th1, &ret);
    (void)pthread_join (th2, &ret);
    ifstream fichier("fichier.txt");
    int i = 0;
    while(fichier.eof() == false)
     {
        fichier.seekg(i,fichier.beg);
        char* buffer= new char[1];
        fichier.read(buffer,1);
        write(new_sock,buffer,1);
        i++;
     }
    fichier.close();
    close(new_sock);
    close(sock_data);
    close(sock);
    return 0;
}

根本问题是accept_data_connexion()在调用accept()接收数据连接之前,没有调用bind()listen()来实际打开数据端口。

总的来说,您所展示的是一个糟糕的实现。你犯了很多错误。sizeof()的误用。在套接字API和FTP协议中都缺乏错误处理。内存泄漏。在不需要的地方使用线程。你真的需要退后一步,想想你在做什么。要么从头开始重写这些代码以正确地完成任务,要么使用第三方库为您处理细节,例如libcurl。