免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1750 | 回复: 6
打印 上一主题 下一主题

想用c写一个Httpserver,大家给点思路啊!! [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2004-10-22 19:54 |只看该作者 |倒序浏览
想写个最简单的多线程httpserver,只要能实现文件传送的功能,大家有什么思路或类似的代码么?小弟在这先谢了!!!

论坛徽章:
2
技术图书徽章
日期:2013-09-04 15:21:51酉鸡
日期:2013-11-01 21:20:20
2 [报告]
发表于 2004-10-22 21:37 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

到网上找一个,小的httpd很多

论坛徽章:
0
3 [报告]
发表于 2004-10-23 02:41 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

早些年写的,供参考。

  1. #include <system.h>;
  2. #include <stdio.h>;
  3. #include <string.h>;
  4. #include <stdlib.h>;
  5. #include <sys/stat.h>;
  6. #ifndef WINDOW_SYS
  7.   #include <unistd.h>;
  8.   #include <sys/socket.h>;
  9.   #include <netinet/in.h>;
  10.   #include <arpa/inet.h>;
  11.   #include <sys/select.h>;
  12.   #include <sys/times.h>;
  13.   #include <sys/time.h>;
  14.   #include <strings.h>;
  15.   #include <signal.h>;
  16. #endif
  17. #include <asyncapi.h>;
  18. #include <tbase.h>;
  19. #include <paramgr.h>;
  20. #include <cini.h>;

  21. char * Gen_Head[] =
  22.   {"Cache-Control",            // Section 14.9
  23.    "Connection",               // Section 14.10
  24.    "Date",                     // Section 14.19
  25.    "Pragma",                   // Section 14.32
  26.    "Transfer-Encoding",        // Section 14.40
  27.    "Upgrade",                  // Section 14.41
  28.    "Via"};                     // Section 14.44
  29. #define Gen_Head_Num 7

  30. int GetGen_HeadType(char * type)
  31. {
  32. int id;
  33.   for(id=0;id<Gen_Head_Num;id++)
  34.     if(!strcmp(Gen_Head[id],type))
  35.       return id;
  36.   return -1;
  37. }

  38. char * Req_Method[] =
  39.   {"OPTIONS",                  // Section 9.2
  40.    "GET",                      // Section 9.3
  41.    "HEAD",                     // Section 9.4
  42.    "POST",                     // Section 9.5
  43.    "PUT",                      // Section 9.6
  44.    "DELETE",                   // Section 9.7
  45.    "TRACE"};                   // Section 9.8
  46. #define Req_Method_Num 7

  47. int GetReq_MethodType(char * type)
  48. {
  49. int id;
  50.   for(id=0;id<Req_Method_Num;id++)
  51.     if(!strcmp(Req_Method[id],type))
  52.       return id;
  53.   return -1;
  54. }

  55. char * Req_Head[] =
  56.   {"Accept",                   // Section 14.1
  57.    "Accept-Charset",           // Section 14.2
  58.    "Accept-Encoding",          // Section 14.3
  59.    "Accept-Language",          // Section 14.4
  60.    "Authorization",            // Section 14.8
  61.    "From",                     // Section 14.22
  62.    "Host",                     // Section 14.23
  63.    "If-Modified-Since",        // Section 14.24
  64.    "If-Match",                 // Section 14.25
  65.    "If-None-Match",            // Section 14.26
  66.    "If-Range",                 // Section 14.27
  67.    "If-Unmodified-Since",      // Section 14.28
  68.    "Max-Forwards",             // Section 14.31
  69.    "Proxy-Authorization",      // Section 14.34
  70.    "Range",                    // Section 14.36
  71.    "Referer",                  // Section 14.37
  72.    "User-Agent"};              // Section 14.42

  73. #define Req_Head_Num 17

  74. int GetReq_HeadType(char * type)
  75. {
  76. int id;
  77.   for(id=0;id<Req_Head_Num;id++)
  78.     if(!strcmp(Req_Head[id],type))
  79.       return id;
  80.   return -1;
  81. }

  82. char * Res_Head[] =
  83.   {"Age",                      // Section 14.6
  84.    "Location",                 // Section 14.30
  85.    "Proxy-Authenticate",       // Section 14.33
  86.    "Public",                   // Section 14.35
  87.    "Retry-After",              // Section 14.38
  88.    "Server",                   // Section 14.39
  89.    "Vary",                     // Section 14.43
  90.    "Warning",                  // Section 14.45
  91.    "WWW-Authenticate"};        // Section 14.46

  92. #define Res_Head_Num 17

  93. int GetRes_HeadType(char * type)
  94. {
  95. int id;
  96.   for(id=0;id<Res_Head_Num;id++)
  97.     if(!strcmp(Res_Head[id],type))
  98.       return id;
  99.   return -1;
  100. }

  101. char * Res_Entity[] =
  102.    {"Allow",                   // Section 14.7
  103.     "Content-Base",            // Section 14.11
  104.     "Content-Encoding",        // Section 14.12
  105.     "Content-Language",        // Section 14.13
  106.     "Content-Length",          // Section 14.14
  107.     "Content-Location",        // Section 14.15
  108.     "Content-MD5",             // Section 14.16
  109.     "Content-Range",           // Section 14.17
  110.     "Content-Type",            // Section 14.18
  111.     "ETag",                    // Section 14.20
  112.     "Expires",                 // Section 14.21
  113.     "Last-Modified"};          // Section 14.29

  114. #define Res_Entity_Num 12

  115. int GetRes_EntityType(char * type)
  116. {
  117. int id;
  118.   for(id=0;id<Res_Entity_Num;id++)
  119.     if(!strcmp(Res_Entity[id],type))
  120.       return id;
  121.   return -1;
  122. }

  123. typedef struct
  124. {
  125.   int  Code;
  126.   char Msg[40];
  127. }StatusCodeStru;

  128. StatusCodeStru StatusCode[]=
  129.   {{100,"Continue"},
  130.    {101,"Switching Protocols"},
  131.    {200,"OK"},                                       /* 成功 */
  132.    {201,"Created"},                                  /* 成功 新的资源建立(POST) */
  133.    {202,"Accepted"},                                 /* 请求被接受,但处理未完成 */
  134.    {203,"Non-Authoritative Information"},
  135.    {204,"No Content"},                               /* 成功,但无信息返回 */
  136.    {205,"Reset Content"},
  137.    {206,"Partial Content"},
  138.    {300,"Multiple Choices"},
  139.    {301,"Moved Permanently"},                        /* 所请求的资源已经被指派为新的固定 URL */
  140.    {302,"Moved Temporarily"},
  141.    {303,"See Other"},                                /* 文档没有修改 (GET) */
  142.    {304,"Not Modified"},
  143.    {305,"Use Proxy"},
  144.    {400,"Bad Request"},                              /* 错误的请求 */
  145.    {401,"Unauthorized"},                             /* 未被授权,该请求要求用户认证 */
  146.    {402,"Payment Required"},
  147.    {403,"Forbidden"},                                /* 不明原因的禁止 */
  148.    {404,"Not Found"},                                /* 没有找到 */
  149.    {405,"Method Not Allowed"},
  150.    {406,"Not Acceptable"},
  151.    {407,"Proxy Authentication Required"},
  152.    {408,"Request Time-out"},
  153.    {409,"Conflict"},
  154.    {410,"Gone"},
  155.    {411,"Length Required"},
  156.    {412,"Precondition Failed"},
  157.    {413,"Request Entity Too Large"},
  158.    {414,"Request-URI Too Large"},
  159.    {415,"Unsupported Media Type"},
  160.    {500,"Internal Server Error"},                    /* 内部服务器差错 */
  161.    {501,"Not Implemented"},                          /* 没有实现 */
  162.    {502,"Bad Gateway"},                              /* 错误的网关 */
  163.    {503,"Service Unavailable"},                      /* 服务暂时失效 */
  164.    {504,"Gateway Time-out"},
  165.    {505,"HTTP Version not supported"}};

  166. //定义文件类型配对信息
  167. typedef struct
  168. {
  169.   char  Ext[8];
  170.   char  Type[32];
  171. }FileTypeStru;

  172. FileTypeStru File_Type[]={
  173.             {"","*/*"},
  174.             {"htm","text/html"},
  175.             {"txt","text/text"},
  176.             {"html","text/html"},
  177.             {"gif","image/gif"},
  178.             {"jpg","image/jpg"},
  179.             {"png","image/png"},
  180.     };
  181. #define File_Type_Num   7

  182. char * GetFile_Type(char * ext)
  183. {
  184. int id;
  185.   for(id=0;id<File_Type_Num;id++)
  186.     if(!strcmp(File_Type[id].Ext,ext))
  187.       return File_Type[id].Type;
  188.   return File_Type[0].Type;
  189. }


  190. /*定义配置信息*/
  191. struct HttpSetStru
  192. {
  193.   /* 存放信息的路径 */
  194.   char  DocPath[128];
  195.   /* 基础路径 */
  196.   char  BasePath[128];
  197.   /* 显示的默认文档 */
  198.   char  DefaultDoc[128];
  199.   /* 用于存放临时文件的路径 */
  200.   char  TmpPath[128];
  201.   /* 用于存放CGI程序的路径 */
  202.   char  CgiPath[128];
  203.   /* 服务端口编号 */
  204.   short Port;
  205.   /* 主机名称 */
  206.   char  HostName[128];
  207. }HttpSet;

  208. /*
  209. 函数功能:判断该路径是否CGI路径
  210. 传入参数:
  211.   path: 要判断的路径
  212. 返回参数:
  213.   1:是CGI路径
  214.   0:不是CGI路径
  215. */
  216. int IsCgiPath(char * path)
  217. {
  218.   if(!strcmp(path,HttpSet.CgiPath)) return 1;
  219.   return 0;
  220. }

  221. /*定义HTTP请求头结构*/
  222. class TRequest_HTTP
  223. {
  224. public:
  225.   /* 客户端的连接 */
  226.   int  mClientSocket;
  227.   /* 客户端的 IP 地址 */
  228.   char mClientIp[30];
  229.   /* 存储的信息 */
  230.   char mLine[5120];
  231.   /* 存储的字节数 */
  232.   char mBytes;
  233.   /* 请求类型 */
  234.   char mType[30];
  235.   /* 请求的 URI */
  236.   char mUri[300];
  237.   /* 版本标识 */
  238.   char mVersion[200];
  239.   /* 存放请求头传入的参数 */
  240.   TParaMgr mReqHead;
  241.   /* 存放返回的参数 */
  242.   TParaMgr mRetHead;
  243.   /* 状态编码 */
  244.   int mStatusCode;
  245.   /* 错误编码 */
  246.   int mErrorCode;
  247. private:
  248.   /* 是不是第一行 */
  249.   int  mIsFirstLine;
  250.   /* 文件名称 */
  251.   char mFileName[200];
  252.   /* 路径名称 */
  253.   char mPathName[200];
  254. public:
  255.   /* 构造函数 */
  256.   TRequest_HTTP();
  257.   /* 返回信息给客户端 */
  258.   void RetToClient();
  259.   /* 添加信息 */
  260.   void BrowseDataCome(char * line,int bytes);
  261.   /* 分析每个实体域 */
  262.   int  AnalysisHeadLine(char * line);
  263.   /* 分析请求的信息 */
  264.   int  AnalysisRequestLine(char * line);
  265.   /* 分析URI */
  266.   void AnalysisUri(char * uri);
  267.   /* 设置状态码及信息 */
  268.   void SetStatusCode(int code,char * msg);
  269.   /* 设置客户端的信 息 */
  270.   void SetClientInfo(int socket,char * ip);
  271.   /* 压缩生成返回的头 */
  272.   char * PackRetHead(int * size);
  273.   /* 处理 GET 的请求句柄 */
  274.   void Process_Get();
  275.   /* 处理请求的句柄 */
  276.   void Process_Request();
  277. };

  278. class T_Http
  279. {
  280. public:
  281.   TMemList * mSiteLst;

  282. public:
  283.   T_Http();
  284.   ~T_Http();

  285.   TRequest_HTTP * GetSite(int socketid);
  286.   int AddSite(int socketid,char * address);
  287.   int DeleteSite(int socketid);
  288.   int Disconnect_Handle(int sock,char * ip);
  289.   int DataArrived_Handle(int sock,char * buf,int length);
  290.   int ConnectIn_Handle(int sock,char * ip,int servsocket);
  291.   int LoadHttpSet();
  292.   void ShowHttpSet();
  293. };

  294. T_Http IHttp;

  295. /*******************************************************************
  296. *                    下面是T_Http类的成员函数
  297. *******************************************************************/
  298. T_Http::T_Http()
  299. {
  300.   mSiteLst=new TMemList();
  301. };

  302. T_Http::~T_Http()
  303. {
  304. TRequest_HTTP * obj;

  305.   while(mSiteLst->;GoHead())
  306.   {
  307.     obj = (TRequest_HTTP * )mSiteLst->;GetPtr();
  308.     mSiteLst->;Remove(obj);
  309.     delete obj;
  310.   };
  311.   delete mSiteLst;
  312. };

  313. /*
  314. 函数功能:获取1个存储信息的数组编号,如果目前不存在,则把它加进去,当然不应当超过限制
  315. 传入参数:
  316. 返回参数:
  317. */
  318. TRequest_HTTP * T_Http::GetSite(int socketid)
  319. {
  320. TRequest_HTTP * obj;

  321.   mSiteLst->;GoHead();
  322.   while(1)
  323.   {
  324.     obj = (TRequest_HTTP * )mSiteLst->;GetPtr();
  325.     if(obj==NULL) break;
  326.     if(obj->;mClientSocket==socketid)
  327.       return obj;
  328.     if(!mSiteLst->;Next())
  329.       break;
  330.   };
  331.   return NULL;
  332. };

  333. int T_Http::AddSite(int socketid,char * address)
  334. {
  335. TRequest_HTTP * obj;

  336.   obj=new TRequest_HTTP();
  337.   obj->;mClientSocket=socketid;
  338.   strcpy(obj->;mClientIp,address);
  339.   mSiteLst->;AddTail(obj);
  340.   return 1;
  341. };

  342. /*
  343. 函数功能:删除一个连接
  344. 传入参数:
  345. 返回参数:
  346. */
  347. int T_Http::DeleteSite(int socketid)
  348. {
  349. TRequest_HTTP * obj;

  350.   obj=GetSite(socketid);
  351.   if(obj==NULL)
  352.     return 0;
  353.   mSiteLst->;Remove(obj);
  354.   delete obj;
  355.   return 1;
  356. };

  357. int T_Http::Disconnect_Handle(int sock,char * ip)
  358. {
  359.   return DeleteSite(sock);
  360. };

  361. int T_Http::DataArrived_Handle(int sock,char * buf,int length)
  362. {
  363. TRequest_HTTP * obj;

  364.   obj=GetSite(sock);
  365.   if(obj==NULL)
  366.     return 0;
  367.   obj->;BrowseDataCome(buf,length);
  368.   return 1;
  369. };

  370. int T_Http::ConnectIn_Handle(int sock,char * ip,int servsocket)
  371. {
  372.   return AddSite(sock,ip);
  373. };

  374. /*
  375. 函数功能:提取webserver的配置参数
  376. 传入参数:
  377. 返回参数:
  378.   1:成功
  379.   0:失败

  380. [system]
  381. port = 88
  382. docpath = doc
  383. basepath = /bill1/goddess/goddess/http
  384. tmppath = tmp
  385. cgipath = cgi
  386. defaultdoc = index.html
  387. */
  388. int T_Http::LoadHttpSet()
  389. {
  390.   int   port;
  391.   TIni  ini;

  392.   if(!ini.LoadFile("webserver.ini"))
  393.     return 0;
  394.   sscanf(ini.GetKey("system","port"),"%d",&port);
  395.   HttpSet.Port=(short)port;
  396.   strcpy(HttpSet.DocPath,ini.GetKey("system","docpath"));
  397.   strcpy(HttpSet.BasePath,ini.GetKey("system","basepath"));
  398.   strcpy(HttpSet.DefaultDoc,ini.GetKey("system","defaultdoc"));
  399.   strcpy(HttpSet.TmpPath,ini.GetKey("system","tmppath"));
  400.   strcpy(HttpSet.CgiPath,ini.GetKey("system","cgipath"));
  401.   return 1;
  402. }

  403. /*
  404. 函数功能:显示webserver的配置参数
  405. 传入参数:
  406. 返回参数:
  407. */
  408. void T_Http::ShowHttpSet()
  409. {
  410.   printf("PORT: %d\n",HttpSet.Port);
  411.   printf("DOC PATH:%s\n",HttpSet.DocPath);
  412.   printf("BASE PATH:%s\n",HttpSet.BasePath);
  413.   printf("DEFAULT DOC:%s\n",HttpSet.DefaultDoc);
  414.   printf("TMP PATH:%s\n",HttpSet.TmpPath);
  415.   printf("CGI PATH:%s\n",HttpSet.CgiPath);
  416. }

  417. /*******************************************************************
  418. *                    下面是TRequest_HTTP类的成员函数
  419. *******************************************************************/
  420. /*
  421. 函数功能:构造函数
  422. 传入参数:
  423. 返回参数:
  424. */
  425. TRequest_HTTP::TRequest_HTTP()
  426. {
  427.   mBytes = 0;
  428.   mIsFirstLine = 1;
  429.   mRetHead.SetValidCheck(0);
  430. }

  431. /*
  432. 函数功能:请求信息的处理
  433.   每一行都已0x0d0x0a结尾,连续的2个0x0d0xa表示客户端请求数据完毕
  434. 传入参数:
  435.   line: 到达的请求数据
  436.   bytes:到达数据的字节数
  437. 返回参数:
  438. */
  439. void TRequest_HTTP::BrowseDataCome(char * line,int bytes)
  440. {
  441. int    id;

  442.   for(id=0;id<bytes;id++)
  443.   {
  444.     //凡是0x0d都忽略掉
  445.     if(line[id]==0x0d)
  446.       continue;
  447.     if(mBytes==0)
  448.     {
  449.       mLine[mBytes++] = line[id];
  450.       continue;
  451.     };
  452.     //如果是0x0a则需要判断是否是请求结束标志
  453.     if(line[id]==0x0a)
  454.     {
  455.       if(mLine[0]==0x0a)
  456.       {
  457.         //连续的2个0x0a,这里已经是最后一行了
  458.         Process_Request();
  459.         //处理结束
  460.         mBytes = 0;
  461.         mIsFirstLine = 1;
  462.         return;
  463.       };
  464.       //一行结束,进行处理
  465.       mLine[mBytes] = 0;
  466.       if(mIsFirstLine)
  467.       {
  468.         //第一行中包含了请求类型
  469.         AnalysisRequestLine(mLine);
  470.         mIsFirstLine = 0;
  471.       }
  472.       else
  473.       {
  474.         //后面行中为其它信息
  475.         AnalysisHeadLine(mLine);
  476.       };
  477.       mBytes=0;
  478.       mLine[mBytes++] = line[id];
  479.       continue;
  480.     };
  481.     if(mLine[0]==0x0a)
  482.       mBytes=0;
  483.     mLine[mBytes++] = line[id];
  484.   };
  485. }

  486. /*
  487. 函数功能:设置客户端的信 息
  488. 传入参数:
  489. 返回参数:
  490. */
  491. void TRequest_HTTP::SetClientInfo(int socket,char * ip)
  492. {
  493.   mClientSocket = socket;
  494.   strcpy(mClientIp,ip);
  495. }

  496. /*
  497. 函数功能:设置状态码及信息
  498. 传入参数:
  499. 返回参数:
  500. */
  501. void TRequest_HTTP::SetStatusCode(int code,char * msg)
  502. {
  503.   mStatusCode = code;
  504. }

  505. /*
  506. 函数功能:解析URI
  507. 传入参数:
  508.   uri: 要解析的文件名
  509. 返回参数:
  510. */
  511. void TRequest_HTTP::AnalysisUri(char * uri)
  512. {
  513. char * pos;
  514. char tmpbuf[512];

  515.   printf("uri is '%s'\n",uri);
  516.   //pos=uri;
  517.   /*
  518.   pos = strstr(uri,HttpSet.HostName);
  519.   if(pos==NULL)
  520.     pos = uri;
  521.   else
  522.   {
  523.     pos += strlen(HttpSet.HostName);
  524.     sprintf(tmpbuf,"%s/%s",HttpSet.BasePath,pos);
  525.   };
  526.   pos = strstr(tmpbuf,HttpSet.CgiPath);
  527.   if(pos!=NULL)
  528.   {
  529.     strcpy(mPathName,HttpSet.CgiPath);
  530.     strcpy(mFileName,pos+strlen(HttpSet.CgiPath));
  531.     return;
  532.   };
  533.   pos = strstr(tmpbuf,HttpSet.DocPath);
  534.   if(pos!=NULL)
  535.   {
  536.     strcpy(mPathName,HttpSet.DocPath);
  537.     strcpy(mFileName,pos+strlen(HttpSet.DocPath));
  538.     return;
  539.   };
  540.   */
  541.   if(*uri=='/')
  542.   {
  543.     strcpy(mPathName,HttpSet.DocPath);
  544.     strcpy(mFileName,uri+1);
  545.   };
  546.   if(mFileName[0]==0)
  547.     strcpy(mFileName,HttpSet.DefaultDoc);
  548.   printf("path is '%s'\n",mPathName);
  549.   printf("file is '%s'\n",mFileName);
  550.   printf("------------------------------------\n");
  551. }

  552. /*
  553. 函数功能:分析每个实体域
  554. 传入参数:
  555. 返回参数:
  556. */
  557. int TRequest_HTTP::AnalysisHeadLine(char * line)
  558. {
  559. char * pos;
  560. char type[201];
  561. int id=0;

  562.   printf("head line: '%s'\n",line);
  563.   pos = line;
  564.   while(*pos==' '||*pos==9)
  565.     pos++;
  566.   while(*pos!=':'&&*pos!=0&&id<200)
  567.     type[id++]=*pos++;
  568.   type[id]=0;
  569.   while(*pos==':')
  570.     pos++;
  571.   while(*pos==' ')
  572.     pos++;
  573.   if(GetGen_HeadType(type))
  574.   {
  575.     mReqHead.PushPara(type,pos);
  576.     return 1;
  577.   };
  578.   if(GetReq_HeadType(type))
  579.     mReqHead.PushPara(type,pos);
  580.   else
  581.     return 0;
  582.   return 1;
  583. }

  584. /*
  585. 函数功能:分析请求的信息
  586. 传入参数:
  587. 返回参数:
  588. */
  589. int TRequest_HTTP::AnalysisRequestLine(char * line)
  590. {
  591. char type[120],uri[300],version[100];

  592.   sscanf(line,"%s %s %s",type,uri,version);
  593.   // 请求类型
  594.   if(GetReq_MethodType(type)<0)
  595.   {
  596.     SetStatusCode(400,"Bad Request");
  597.     return 0;
  598.   };
  599.   strcpy(mType,type);
  600.   strcpy(mUri,uri);
  601.   AnalysisUri(mUri);
  602.   strcpy(mVersion,version);
  603.   return 1;
  604. }

  605. /*
  606. 函数功能: 压缩生成返回的头
  607. 传入参数:
  608.   size: 返回型参数,用于存放生成的buf的大小
  609. 返回参数:
  610.   压缩生成的内存空间指针
  611. */
  612. char * TRequest_HTTP::PackRetHead(int * size)
  613. {
  614. char * pos=NULL,tmpbuf[300],paraval[300],* end;
  615. int poolsize,len,id;

  616.   *size=0;
  617.   poolsize=2000;

  618.   pos=(char *)IMem.Malloc(poolsize);
  619.   end=pos;

  620.   for(id=0;id<mRetHead.mParaNum;id++)
  621.   {
  622.     mRetHead.GetParaValue(id,paraval);
  623.     if(id==0)
  624.       sprintf(tmpbuf,"%s%s",mRetHead.GetParaName(id),paraval);
  625.     else
  626.       sprintf(tmpbuf,"%s: %s",mRetHead.GetParaName(id),paraval);
  627.     len = strlen(tmpbuf);
  628.     printf("len is %d tmp is '%s'\n",len,tmpbuf);
  629.     if((*size+len+2) >; poolsize)
  630.     {
  631.       char * newbuf;
  632.       newbuf=(char *)IMem.Malloc(poolsize+2000);
  633.       memcpy(newbuf,pos,poolsize);
  634.       IMem.Free(pos);
  635.       pos=newbuf;

  636.       poolsize+=2000;
  637.       end = pos+(*size);
  638.     };
  639.     *size += (len+2);
  640.     strcpy(end,tmpbuf);
  641.     end += len;
  642.     *end = 0x0d;
  643.     end ++;
  644.     *end = 0x0a;
  645.     end ++;
  646.   };
  647.   //最后加上0x0d0x0a表示结束
  648.   *size += (2);
  649.   *end = 0x0d;
  650.   end ++;
  651.   *end = 0x0a;
  652.   end ++;
  653.   return pos;
  654. }

  655. /*
  656. 函数功能:处理客户端 GET 请求的函数
  657.     需要判断是否调用 CGI 的目录路径
  658. 传入参数:
  659. 返回参数:
  660. */
  661. void TRequest_HTTP::Process_Get()
  662. {
  663. char   fullname[512],cgicmd[512];
  664. char   type[20];
  665. char   ext[220],*headpool;
  666. struct stat statbuf;
  667. int    size,id;
  668. FILE * fp;
  669. #define MAXDATALEN  2038  //最大包长
  670. int   datalen;
  671. char  databuf[MAXDATALEN];

  672.   printf("path is '%s'\n",mPathName);
  673.   printf("file is '%s'\n",mFileName);

  674.   #ifndef UNIX_SYS
  675.   IString.ReplaceStr(mPathName,"/","\\");
  676.   IString.ReplaceStr(mFileName,"/","\\");
  677.   #endif

  678.   #ifdef UNIX_SYS
  679.   sprintf(fullname,"%s/%s",mPathName,mFileName);
  680.   #else
  681.   sprintf(fullname,"%s\\%s",mPathName,mFileName);
  682.   #endif

  683.   size=strlen(mFileName);
  684.   headpool=mFileName;
  685.   headpool+=(size-1);
  686.   for(id=size;id>;0;id--)
  687.   {
  688.     if(*headpool=='.'||*headpool=='/'||headpool==mFileName)
  689.       break;
  690.     headpool--;
  691.   };
  692.   if(*headpool=='.')
  693.     headpool++;
  694.   strcpy(ext,headpool);
  695.   if(IsCgiPath(mPathName))
  696.   {
  697.     sprintf(cgicmd,"system %s >;%s/%d.html",HttpSet.TmpPath,mClientSocket);
  698.     system(cgicmd);
  699.     #ifdef UNIX_SYS
  700.     sprintf(fullname,"%s/%d.html",HttpSet.TmpPath,mClientSocket);
  701.     #else
  702.     sprintf(fullname,"%s\\%d.html",HttpSet.TmpPath,mClientSocket);
  703.     #endif
  704.   }
  705.   else
  706.     #ifdef UNIX_SYS
  707.     sprintf(fullname,"%s/%s",mPathName,mFileName);
  708.     #else
  709.     sprintf(fullname,"%s\\%s",mPathName,mFileName);
  710.     #endif
  711.   if(stat(fullname,&statbuf))
  712.   {
  713.     SetStatusCode(404,"Not Found");
  714.     mRetHead.PushPara("HTTP","/1.0 404 Not Found");
  715.     return;
  716.   }
  717.   mRetHead.PushPara("HTTP","/1.0 200 OK");
  718.   SetStatusCode(200,"OK");
  719.   printf("cont leng is %d\n",statbuf.st_size);
  720.   strcpy(type,GetFile_Type(ext));
  721.   mRetHead.PushPara("Server","Orient WebServer");
  722.   mRetHead.PushPara("Connection","keep-alive");
  723.   mRetHead.PushPara("Content-Type",type);
  724.   mRetHead.PushPara("Accept-Ranges","bytes");
  725.   mRetHead.PushPara("Content-Length",statbuf.st_size);
  726.   headpool=PackRetHead(&size);
  727.   NetItf.SendData(mClientSocket,headpool,size);
  728.   IMem.Free(headpool);
  729.   fp=fopen(fullname,"rb");
  730.   while((datalen=fread(databuf,1,MAXDATALEN,fp))>;0)
  731.   {
  732.     NetItf.SendData(mClientSocket,databuf,datalen);
  733.     if(datalen<MAXDATALEN)
  734.     {
  735.       //所有数据都已经发送完了
  736.       break;
  737.     };
  738.   }
  739.   fclose(fp);
  740. }

  741. /*
  742. 函数功能:所有请求的处理句柄
  743.   现在只处理GET请求
  744. 传入参数:
  745. 返回参数:
  746. */
  747. void TRequest_HTTP::Process_Request()
  748. {
  749.   Process_Get();
  750.   mReqHead.ResetPara();
  751.   mRetHead.ResetPara();
  752. }

  753. #include <asyncapi.h>;

  754. int SocketDisconnect_Handle_Http(int sock,char * ip)
  755. {
  756.   return IHttp.DeleteSite(sock);
  757. }

  758. int DataArrived_Handle_Http(int sock,char * buf,int length)
  759. {
  760.   return IHttp.DataArrived_Handle(sock,buf,length);
  761. }

  762. int SocketConnectIn_Handle_Http(int sock,char * ip,int servsocket)
  763. {
  764.   IHttp.AddSite(sock,ip);
  765.   NetItf.AddClient(sock,ip);
  766.   NetItf.SetDisconnectHandle(sock,SocketDisconnect_Handle_Http);
  767.   NetItf.SetDataArriveHandle(sock,DataArrived_Handle_Http);
  768.   NetItf.SetRecvBufSize(sock,0);
  769.   return 1;
  770. }

  771. void Exit_Handle_Http(int para)
  772. {
  773.   exit(0);
  774. }

  775. int main_Web(int argc, char **argv)
  776. {
  777. int socket;

  778.   IHttp.LoadHttpSet();
  779.   IHttp.ShowHttpSet();
  780.   /*显示版权信息*/
  781.   printf("*******************************************************************************\n");
  782.   printf("                       Orient Simple Web Server startup\n");
  783.   printf("                                 www.onlyit.cn\n");
  784.   printf("                                  Version 1.0\n");
  785.   printf("*******************************************************************************\n");
  786.   socket=INet.Server(HttpSet.Port);
  787.   if(socket<0)
  788.   {
  789.     puts("Unable to create sock for http serv");
  790.     exit(0);
  791.   };
  792.   NetItf.AddServer(socket,"127.0.0.1");
  793.   NetItf.BeginSocketAsync(socket);
  794.   NetItf.SetRecvBufSize(socket,16384);
  795.   NetItf.SetConnectInHandle(socket,SocketConnectIn_Handle_Http);
  796.   NetItf.SetDisconnectHandle(socket,SocketDisconnect_Handle_Http);
  797.   NetItf.SetDataArriveHandle(socket,DataArrived_Handle_Http);
  798.   NetItf.NetEventMgr_Balance();
  799.   return 1;
  800. }

复制代码

论坛徽章:
0
4 [报告]
发表于 2004-10-23 02:42 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

从后往前看简单一点。

论坛徽章:
0
5 [报告]
发表于 2004-10-23 19:58 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

google thttpd

论坛徽章:
0
6 [报告]
发表于 2004-10-25 12:45 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

楼上是小鱼儿,哈哈.

论坛徽章:
0
7 [报告]
发表于 2004-10-25 17:24 |只看该作者

想用c写一个Httpserver,大家给点思路啊!!

看看apache源代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP