IRC机器人故障

时间:2016-12-28 19:25:01

标签: c network-programming bots irc

我试图在C中制作一个IRC机器人。当机器人试图连接到IRC服务器时,它进入一个无限循环,它什么都没有收到。 我不确定这是因为我加入IRC服务器的过程是否格式错误,或者我是否遗漏了一些应该发送/接收的数据。

#include<stdio.h>
#include<sys/socket.h>
#include<netdb.h>
#include<string.h>
#include<stdlib.h>
#include<time.h>


#define MAXSIZE 4096

void delay(int milliseconds)
{
    long pause;
    clock_t now;

    pause = milliseconds*(CLOCKS_PER_SEC/1000);//set delay using
    now = clock();
    while( now < pause )
        now = clock();
}

int send_data(int sockfd, char message[])
{
    send(sockfd, message, strlen(message), 0);
    printf("OUT: %s\n", message);
    return 1;
}

int recv_data(int sockfd, char *message)
{
    int n;
    n = recv(sockfd, message, MAXSIZE, 0);
    printf("IN: %s\n", message);
    return n;
}

int tcp_connect(int *sockfd, char server[], char port[])
{
    //declare variables
    struct addrinfo hints, *res;

    //zero out structures
    memset(&hints,0,sizeof(hints));
    hints.ai_family = AF_INET;
    hints.ai_socktype = SOCK_STREAM;

    //query DNS server for IP address and port
    getaddrinfo(server,port,&hints,&res);

    //create socket for data transmission
    *sockfd = socket(res->ai_family,res->ai_socktype,0);
    if (*sockfd < 0)
    {
        printf("failure to create socket\n");
        return 0;
    }

    //connect to server side port using created socket
    if (connect(*sockfd, res->ai_addr, res->ai_addrlen)!= 0)
    {
        printf("failure to connect to port\n");
        return 0;
    }

    freeaddrinfo(res);
    return 1;
}

int irc_auth(int sockfd)
{
    //create and start clock
    clock_t start_t;
    start_t = clock();

    //seed RNG with clock output
    srand(start_t);

    //generate necessary variables
    char name[15] = "bot";
    char user[35] = "USER ";
    char nick[20] = "NICK ";
    char join[20] = "JOIN #randChat\r\n";
    int i,id;

    //generate random character for ID tag A-Z
    for(i=0; i<5; i++)
    {
        id = rand() % 91;
        if(id < 65)
        {
            while(id < 65)
            {
                id = rand() % 91;
            }
        }
        name[strlen(name)] = id;
    }
    //append return and null to string
    strcat(nick,name);
    strcat(nick,"\r\n");

    //append to finish creating USER IRC command
    strcat(user,name);
    strcat(user," 8 * :");
    strcat(user,name);
    strcat(user,"\r\n");

    //send data to server
    send_data(sockfd,user);
    delay(1000);
    send_data(sockfd,nick);
    delay(1000);
    send_data(sockfd,join);

    return 1;
}
int main (int argc, char *argv)
{
//variables
    int sockfd, n, flag;
    char *mesg_in = malloc(sizeof(char) * MAXSIZE); 
    char *pos;
    char nick[30];

    char *mesg_out = malloc(sizeof(char) * MAXSIZE);

    //connect to port 6667 of irc.freenode.org using tcp
    while(flag<1)
    {
        if(tcp_connect(&sockfd,"irc.freenode.org","6667") == 1)
        {
            flag = 1;
        }
    }   

    //IRC channel authentication
    irc_auth(sockfd);

    //command loop
    while(1)
    {
        mesg_in[0] = 0;// zero out message
        //memset(mesg_in,0,strlen(mesg_in));
        n = recv_data(sockfd,mesg_in);// pull message from channel

        if (n > 0)// check to see if it recieved a command
        {
            mesg_in[n] = 0;// set null at the end of recieved data

            //respond to ping commands from server
            if(strstr(mesg_in,"PING") != NULL)
            {
                mesg_out[0] = 0;// zero out message
                pos = strstr(mesg_in," ")+1;// point to data needed
                //append to out bound message
                sprintf(mesg_out,"PONG %s\r\n",pos);
                //send outbound message
                send_data(sockfd,mesg_out);
            }
        }
    }
}

任何和所有帮助将不胜感激

2 个答案:

答案 0 :(得分:2)

无论其他问题是什么,delay()都是其中之一。您在此测试程序中的功能等待两秒,然后同时打印1 2 3,因为它只考虑程序启动时的经过时间,而不是当前时刻。

#include <stdio.h>
#include <time.h>

void delay(int milliseconds)
{
    long pause;
    clock_t now;
    pause = milliseconds*(CLOCKS_PER_SEC/1000);//set delay using
    now = clock();
    while( now < pause )
        now = clock();
}

int main (void)
{  
    delay(2000);
    printf("1\n");

    delay(2000);
    printf("2\n");

    delay(2000);
    printf("3\n");

    return 0;
}

此版本以两秒的间隔打印1 2 3

#include <stdio.h>
#include <time.h>

void delay(clock_t milliseconds)
{
    clock_t elapsed, pause, stamp;
    stamp = clock();
    pause = milliseconds * CLOCKS_PER_SEC / 1000;
    while ((elapsed = clock() - stamp) < pause);
}

int main (void)
{  
    delay(2000);
    printf("1\n");

    delay(2000);
    printf("2\n");

    delay(2000);
    printf("3\n");

    return 0;
}

请注意,在整数运算中,我在除法之前进行乘法运算。

答案 1 :(得分:0)

重新思考您的客户端并使其成为状态机,而不是由事件引擎驱动,例如epoll(),kqueue(),select()或poll()。

通常,IRC命令会生成回复,但您不知道它们何时到达,并且协议的设计使您可能希望发送不会被数据启动的命令来自服务器。

使用延迟对IRC服务器进行身份验证是禁忌,因为有许多其他命令可以作为身份验证的一部分生成,例如PING或CTCP等,您的昵称正在使用中,等等。

另外,根据RFC,NICK命令必须先在用户之前。一般来说,IRC服务器是宽容的,但不要认为这是理所当然的。正如谚语所说,&#34;慷慨接受你所接受的内容,严格遵守你所发送的内容&#34;。