C语言socket编程--每日签到

  前几天写了个python的每日签到,你运行还得借助crontab,很是不爽.....正好前几天看了个关于c编写daemon进程,加上自己那点可怜的socket知识,于是我们重操旧页,C语言版的每日签到。就是要签到~~~

  关于daemon程序,也叫守护进程,一般运行在后台,我喜欢悄悄的运行。。。。哈哈哈,这个init_daemon.c被大家在网上转来转去,原则都是一样的

Daemon程序设计主要原则包括:

(1)       程序运行后调用fork,并让父进程退出。子进程获得一个新的进程ID,但继承了父进程的进程组ID。

(2)       调用setsid创建一个新的session,使自己成为新session和新进程组的leader,并使进程没有控制终端(tty)。

(3)       设置文件创建mask为0,避免创建文件时权限的影响。

(4)       关闭不需要的打开文件描述符。因为Daemon程序在后台执行,不需要于终端交互,通常就关闭STDIN、STDOUT和STDERR。其它根据实际情况处理。

(5)       Daemon无法输出信息,可以使用SYSLOG或自己的日志系统进行日志处理。(可选)

(6)       编写管理Daemon的SHELL脚本,使用service对Daemon进行管理和监控。(可选)

  

#include <unistd.h>
#include <signal.h>
#include <sys/param.h>
#include <sys/stat.h>
void init_daemon(void)
{
    int pid;
    int i;

    if(pid=fork())
        exit(0);//是父进程,结束父进程
    else if(pid< 0)
        exit(1);//fork失败,退出
   //是第一子进程,后台继续执行
    
    setsid();//第一子进程成为新的会话组长和进程组长
    //并与控制终端分离
    if(pid=fork())
    exit(0);//是第一子进程,结束第一子进程
    else if(pid< 0)
    exit(1);//fork失败,退出
    //是第二子进程,继续
   //第二子进程不再是会话组长
    
    for(i=0;i< NOFILE;++i)//关闭打开的文件描述符
    close(i);
    chdir("/tmp");//改变工作目录到/tmp
    umask(0);//重设文件创建掩模
    return;
}

这个其实就是个引子,下面的是实体。。

#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <time.h>
#include <errno.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define IPSTR "1.1.1.1"
#define PORT 80
#define BUFSIZE 1024

void init_daemon(void); //守护进程初始化
int main(int argc, char **argv)
{
    FILE *fp;
    time_t t;
    int sockfd, ret, i, h;
    struct sockaddr_in servaddr;
    /*
      sockaddr_in(在netinet/in.h中定义):
      struct sockaddr_in{
      short sin_family;/*地址族一般来说是 AF_INET
      unsigned short sin_port; /* Port number(必须采用网络数据格式,普通数字可以用htons()转换成网络格式的数字)
      }
    */
    char str1[4096], str2[4096], buf[BUFSIZE], *str;
    socklen_t len;
    fd_set   t_set1;
    struct timeval  tv;
    /*
      struct timeval{
          time_t  tv_sec; //seconds
          suseconds_t tv_usec; // 毫米 microseconds
      }
     */
    init_daemon(); //守护进程初始化
    while(1){
        if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0 ) {
            printf("创建网络连接失败,本线程即将终止---socket error!\n");
            exit(0);
        };
        /*填写sockadd_in结构*/
        bzero(&servaddr, sizeof(servaddr));
        servaddr.sin_family = AF_INET;
        servaddr.sin_port = htons(PORT);
        if (inet_pton(AF_INET, IPSTR, &servaddr.sin_addr) <= 0 ){
            //inet_pton 将点分十进制IPSTR转换成一个16进制地址,转换错误返回0
            /*另一种填写servaddr.sin_addr.addr = inet_addr(IPSTR);*/
            printf("地址错误--inet_pton error!\n");
            exit(0);
        };
        /*尝试连接*/
        if (connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0){
            printf("连接到服务器失败,connect error!\n");
            exit(0);
        }
        //printf("与远端建立了连接\n");
        //发送数据
        memset(str2, 0, 4096); 
        //memset(void *s, int c, size_t n)将参数s的内存前n个字符串用参数c填入,常用做内存控件初始化
        strcat(str2, "qdxq=kx&qdmode=1&todaysay=love+is+beautiful.so+beautiful&fastreply=1");
        str=(char *)malloc(128);
        len = strlen(str2);
        sprintf(str, "%d", len);
        
        memset(str1, 0, 4096);
        strcat(str1, "POST http://xxx.xxx.com/plugin.php?identifier=dps_sign&module=sign&operation=qiandao&infloat=1&inajax=1 HTTP/1.1\n");
        strcat(str1, "Host: xxx.xxxx.com\n");
        strcat(str1, "Content-Type: application/x-www-form-urlencoded\n");
        strcat(str1, "User-Agent: Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/30.0.1599.114 Safari/537.36\n");
        strcat(str1, "Cookie:你的cookie\n");
        strcat(str1, "Content-Length: ");
        strcat(str1, str);
        strcat(str1, "\n\n");
        
        strcat(str1, str2);
        strcat(str1, "\r\n\r\n");
        // printf("%s\n",str1);
        
        ret = write(sockfd,str1,strlen(str1));
        /*  write(int fd, const void *buf, size_t count)*/
        if (ret < 0) {
            printf("发送失败!错误代码是%d,错误信息是'%s'\n",errno, strerror(errno));
            exit(0);
        }else{
             //printf("消息发送成功,共发送了%d个字节!\n\n", ret);
        }
        /*接受从server端传来的信息*/
        //FD_ZERO(&t_set1); //清空套接字集合
        //FD_SET(sockfd, &t_set1); //将sockfd加入t_set1集合中
        //对fd_set还有种常见用法FD_CLR(fd,&set)将fd从集合中清楚
        // printf("%d",sizeof(t_set1));exit(0); result:128Byte = 128*8个fd
        //可监控的fd个数取决于sizeof(fd_set)
        /*
          while(1){
              sleep(2);
              tv.tv_sec = 0;
              tv.tv_usec = 0;
              h = 0;
              h = select(sockfd +1, &t_set1, NULL, NULL, &tv);
              //printf("\n%d",h);exit(0); //results:1
          
             //if (h == 0) continue;
             if (h < 0) {
                  close(sockfd);
                  printf("在读取数据报文时SELECT检测到异常,该异常导致线程终止!\n");
                  return -1;
              };
          
              if (h > 0){
              memset(buf, 0, 4096);
              i= read(sockfd, buf, 4095);
              if (i == 0){
                  close(sockfd);
              printf("读取数据报文时发现远端关闭,该线程终止!\n");
              return -1;
           }
          
           printf("%s\n", buf);
         }
        
         }*/
        close(sockfd); 
        // return 0;
        /*写入日志*/
        
        if((fp=fopen("sign.log","a"))>=0){
            t = time(0);
            fprintf(fp, "l137 was here at %s", asctime(localtime(&t)));
            fclose(fp);
        }
        sleep(14400); //每四个小时签到,妈妈再也不担心,今天没签到了
    }
    return 0;
}

具体的请求你可以用burp抓个提交包,然后把你的cookie对应的填进去。。http就是一种特殊的socket,大家商量了一下,你用80端口,我也80端口,你按照这个格式发数据,OK,我按照说好的格式解析数据。。。。。

编译的时候gcc -g init_daemon.c bbs_sign.c -o l137

当然我们也可以自己写一个简单的Makefile,简洁大方上档次。。

CC = gcc
all:myapp

myapp: 
    $(CC) -g init_daemon.c bbs_sign.c -o l137
clean:
    rm -rf  l137

找个小服务器跑下,小程序安安稳稳的运行在后台。。。。。

 

    

posted @ 2013-11-04 19:36  l4wl137  阅读(1251)  评论(0编辑  收藏  举报