小型 web 服务器系统,小型web服务器

HTTP是一个属于应用层的面向对象的协议,由于其简捷、快速的方式,适用于分布式超媒体信息系统。HTTP协议的主要特点可概括如下:

1.支持客户/服务器模式。

2.简单快速:客户向服务器请求服务时,只需传送请求方法和路径。请求方法常用的有GET、HEAD、POST。每种方法规定了客户与服务器联系的类型不同。由于HTTP协议简单,使得HTTP服务器的程序规模小,因而通信速度很快。

3.灵活:HTTP允许传输任意类型的数据对象。正在传输的类型由Content-Type加以标记。4.无连接:无连接的含义是限制每次连接只处理一个请求。服务器处理完客户的请求,并收到客户的应答后,即断开连接。采用这种方式可以节省传输时间。5.无状态:HTTP协议是无状态协议。无状态是指协议对于事务处理没有记忆能力。缺少状态意味着如果后续处理需要前面的信息,则它必须重传,这样可能导致每次连接传送的数据量增大。另一方面,在服务器不需要先前信息时它的应答就较快。

http(超文本传输协议)是一个基于请求与响应模式的、无状态的、应用层的协议,常基于TCP的连接方式,HTTP1.1版本中给出一种持续连接的机制,绝大多数的Web开发,都是构建在HTTP协议之上的Web应用。

HTTP URL (URL是一种特殊类型的URI,包含了用于查找某个资源的足够的信息)的格式如下:

http://host[":"port][abs_path]

http表示要通过HTTP协议来定位网络资源;host表示合法的Internet主机域名或者IP地址;port指定一个端口号,为空则使用缺省端口80;abs_path指定请求资源的URI;如果URL中没有给出abs_path,那么当它作为请求URI时,必须以“/”的形式给出,通常这个工作浏览器自动帮我们完成。

1.http请求由三部分组成,分别是:请求行、消息报头

请求行以一个方法符号开头,以空格分开,后面跟着请求的URI和协议的版本,格式如下:Method Request-URI HTTP-Version CRLF  其中 Method表示请求方法;Request-URI是一个统一资源标识符;HTTP-Version表示请求的HTTP协议版本;CRLF表示回车和换行(除了作为结尾的CRLF外,不允许出现单独的CR或LF字符)。

在接收和解释请求消息后,服务器返回一个HTTP响应消息。

2.HTTP响应也是由三个部分组成,分别是:状态行、消息报头、响应正文

状态行格式如下:HTTP-Version Status-Code Reason-Phrase CRLF

其中,HTTP-Version表示服务器HTTP协议的版本;Status-Code表示服务器发回的响应状态代码;Reason-Phrase表示状态代码的文本描述。

状态代码有三位数字组成,第一个数字定义了响应的类别,且有五种可能取值:

1xx:指示信息--表示请求已接收,继续处理

2xx:成功--表示请求已被成功接收、理解、接受

3xx:重定向--要完成请求必须进行更进一步的操作

4xx:客户端错误--请求有语法错误或请求无法实现

5xx:服务器端错误--服务器未能实现合法的请求

常见状态代码、状态描述、说明:

200 OK      //客户端请求成功

400 Bad Request  //客户端请求有语法错误,不能被服务器所理解

401 Unauthorized //请求未经授权,这个状态代码必须和WWW-Authenticate报头域一起使用

403 Forbidden  //服务器收到请求,但是拒绝提供服务

404 Not Found  //请求资源不存在,eg:输入了错误的URL

500 Internal Server Error //服务器发生不可预期的错误

503 Server Unavailable  //服务器当前不能处理客户端的请求,一段时间后可能恢复正常

b67dd0a341f00e05efd1f253828362a0.png

当前目录下文件如上图所示。

http.h http.c main.c 为主要源文件

conf,cgi_bin,log,build.sh暂时没使用

htdocs文件夹里存储网页数据来源:

1.jpg和一张html

hello

hello world

        

Makefile:ROOT_PATH=$(shell pwd)

LDFLAGS=-lpthread

FLAGS=#_DEBUG1_

CC=gcc

BIN=httpd

SRC=$(shell ls *.c)

OBJ=$(SRC:.c=.o)

$(BIN):$(OBJ)

$(CC) -o $@ $^ $(LDFLAGS)

%.o:%.c

$(CC) -c $<

.PHONY:clean

clean:

rm -f $(OBJ) $(BIN)

.PHONY:debug

debug:

@echo $(SRC)

@echo $(OBJ)

所用函数:

ff7dfcce70e52235990d75cc584e9e37.png

返回值:成功0;失败:-1,置错误码

e66b5efcf9b08edd586cfa2726dd297a.png

923e373db6a4a8d82fb8800db881ad95.png

st.st_mode:

5cfa104f707945479edb8409f2e041de.png

e7dbdaa7a8fbe20fd8add33850ca1224.png

6702984718f3f3004f88ea28f58da699.png

返回值:成功:写到outfd的字节数;失败:-1,置错误码//http.h

#pragma once

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#define _DEF_PAGE_ "index.html"

#define _SIZE_ 1024

static void printLog(const char* const str,const char* const fun,int line);

void usage(const char*  const  proc);

int startup(char* ip,int port);

void echo_error(int sock,int errno);

int get_line(int sock,char* buf,int size);

void clear_head(int sock);

void echo_html(int sock,char* path,ssize_t size);

void accept_request(int sock);

void* handle_client(void* arg);

//http.c

#include"http.h"

//打印错误消息

static void printLog(const char* const str,const char* const fun,int line)

{

printf("%s:%s:%d\n",str,fun,line);

}

void usage(const char*  const  proc)

{

assert(proc);

printLog(proc,__FUNCTION__,__LINE__);

}

//创建套接字,完成绑定,设置为连接状态

int startup(char* ip,int port)

{

assert(ip);

int sock=socket(AF_INET,SOCK_STREAM,0);

if(sock<0)

{

printLog(strerror(errno),__FUNCTION__,__LINE__);

exit(1);

}

int opt=1;

setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));

struct sockaddr_in local;

local.sin_family=AF_INET;

local.sin_port=htons(port);

if(strncmp(ip,"any",3)==0)

{

local.sin_addr.s_addr= INADDR_ANY;

}

else

{

local.sin_addr.s_addr=inet_addr(ip);

}

if(bind(sock,(struct sockaddr*)&local,sizeof(local))<0)

{

printLog(strerror(errno),__FUNCTION__,__LINE__);

exit(1);

}

if(listen(sock,5)<0)

{

printLog(strerror(errno),__FUNCTION__,__LINE__);

exit(1);

}

return sock;

}

void echoErr(int sock,int errno)

{

return;

}

//每次读取一行

int get_line(int sock,char* buf,int size)

{

assert(buf);

int i=0;

ssize_t _s=-1;

char ch='\0';

//  printf("getLine");

while(i

{

_s=recv(sock,&ch,1,0);

if(_s>0)

{

if(ch=='\r')

{

if((_s=recv(sock,&ch,1,MSG_PEEK)))

{

if(_s>0&&ch=='\n')

recv(sock,&ch,1,0);

}

}

buf[i++]=ch;

}

else

{

buf[i++]='\n';

break;

}

}

//  printf("endddddddddddd");

buf[i]='\0';

return i;

}

//清除sock报文头部

void clear_head(int sock)

{

char buf[_SIZE_];

buf[0]='\0';

ssize_t _s=1;

while(_s>0&&strcmp(buf,"\n")!=0)

{

_s=get_line(sock,buf,sizeof(buf));

}

}

//发回响应

void echo_html(int sock,char* path,ssize_t size)

{

//  printf("echo_htmlllllllllll\n");

//  printf("path:%s\n",path);

int fd=open(path,O_RDONLY);

if(fd<0)

{

printLog(strerror(errno),__FUNCTION__,__LINE__);

exit(1);

}

//  printf("fd:%d\n",fd);

char* status_line="HTTP/1.0  200 ok\r\n\r\n";

send(sock,status_line,strlen(status_line),0);

//  printf("sock  success,size:%d\n",size);

if(sendfile(sock,fd,NULL,size)<0)

{

printf("error");

}

//  printf("sendfile  success");

close(fd);

}

//处理请求

void accept_request(int sock)

{

char buf[_SIZE_];

//  int ret;

//  while((ret=get_line(sock,buf,sizeof(buf)))>0)

//  {

//      if(ret==0)

//      {

//          printLog(strerror(errno),__FUNCTION__,__LINE__);

//          break;

//      }

//      printf("%s",buf);

//  }

int ret=-1;

int i=0,j=0;

char method[_SIZE_/2];

char url[_SIZE_];

char path[_SIZE_];

memset(method,'\0',sizeof(method));

memset(buf,'\0',sizeof(buf));

memset(path,'\0',sizeof(path));

memset(url,'\0',sizeof(url));

if(get_line(sock,buf,sizeof(buf))==0)

{

printLog("errno",__FUNCTION__,__LINE__);

return;

}

//  printf("%s",buf);

i=j=0;

while(!isspace(buf[i])&&i

{

method[j]=buf[i];//get method

++i;

++j;

}

method[j]='\0';

//  printf("method:%s\n",method);

j=0;

while(isspace(buf[i]))

{

++i;

}

while(!isspace(buf[i])&&i

{

url[j]=buf[i];

++j;

++i;

}

url[j]='\0';

//  printf("url:%s\n",url);

int cgi=0;

if(strcasecmp(method,"POST")!=0&&strcasecmp(method,"GET")!=0)

{

printf("error\n");

//  echoErr(sock,1);

return;

}

//  printf("success\n");

if(strcasecmp(method,"POST")==0)

{

cgi=1;

}

if(strcasecmp(method,"GET")==0)

{

//      printf("getttttttttttttttttttt\n");

char* query_string=url;

while(*query_string!='\0'&&*query_string!='?')

{

++query_string;

}

if(*query_string=='?')

{

*query_string='\0';

cgi=1;

}

++query_string;

sprintf(path,"htdocs%s",url);//sprintf

//      printf("path:%s\n",path);

if(path[strlen(path)-1]=='/')

{

strcat(path,_DEF_PAGE_);

}

//      printf("cgi=0,path:%s\n",path);

struct stat st;

if(stat(path,&st)<0)//not exist

{

printf("error");

return;

}

else if(S_IFDIR&st.st_mode)//dir

{

//          printf("dirrrrrrrrrrr\n");

if(strcmp(path,"htdocs/_DEF_PAGE_")!=0)

{

strcpy(path,"htdocs/");

strcat(path,_DEF_PAGE_);

//              printf("hhhhhhhhhhhhhhhhhh");

}

}

else if((st.st_mode&S_IXUSR)||(st.st_mode&S_IXGRP)||(st.st_mode&S_IXOTH))

{

//          printf("xxxxxxxxxxxxxxxxxx\n");

cgi=1;

}

if(cgi)

{

//execute_cgi(sock,path,method,query_string);

}

else

{

//          printf("echo_html\n");

fflush(stdout);

clear_head(sock);

echo_html(sock,path,st.st_size);

}

}

close(sock);

}

void* handle_client(void* arg)

{

int sock=(int)arg;

accept_request(sock);

return NULL;

}

//main.c

#include"http.h"

int main(int argc,char* argv[])

{

if(argc!=3)

{

usage(argv[0]);

return 1;

}

char* ip=argv[1];

int port=atoi(argv[2]);

int listen_sock=startup(ip,port);

struct sockaddr_in client;

socklen_t len=sizeof(client);

fflush(stdout);

while(1)

{

//  printf("kkkkkkkkkkk");

//  fflush(stdout);

int new_sock=accept(listen_sock,(struct sockaddr*)&client,&len);

if(new_sock<0)

{

printf("no client");

fflush(stdout);

continue;

}

pthread_t id;

//  printf("con");

pthread_create(&id,NULL,handle_client,(void*)new_sock);

pthread_detach(id);//set detach return val:0/errno

}

return 0;

}

注:注释为了测试,没实际意义。

运行截图

f3d8dd47f1ce26e1173aa63ecef7571f.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值