该简易的J2EE WEB容器缺失很多功能,却可以提供给大家学习HTTP容器大致流程。

注:容器功能很少,只供学习。

1. 支持静态内容与Servlet,不支持JSP

2. 仅支持304/404

3. 该设计参考Jetty容器

GIT地址:https://git.oschina.net/redcode/jerry.git

一、HTTP请求处理流程:

HTTP包的解析直接使用Socket读取InputStream,再根据HTTP协议读取HTTP请求头于数据体,HTTP GET请求头类似如下:

GET / HTTP/1.1
Accept: */*
Accept-Language: zh-CN
User-Agent:
Accept-Encoding: gzip, deflate
Host: www.baidu.com
Connection: Keep-Alive

1. 如GET / HTTP/1.1代表是GET 请求,请求路径为/,协议版本为HTTP 1.1,中间使用空格分隔,请求头每个属性一行,使用\n换行(WINDOWS为\r\n)。

当解析Socket的InputStream的时候首先读取第一行,代码类似如下:

BufferedReader br = new BufferedReader( new InputStreamReader(socket.getInputStream()) );
String reqCmd = br.readLine();
if(reqCmd == null){
return null; //数据包不正常,忽略
}
String[] cmds = reqCmd.split("\\s");

2. POST 请求包类似如下:

POST /login HTTP/1.1
Accept: */*
User-Agent:
Host:
Pragma: no-cache
Cookie:
Content-Length: 25 count=1&viewid=lNe3tRpyVj

请求头后换行,再封装POST请求数据:count=1&viewid0=lNe3tRpyVj

解析POST请求包时,读取请求头后再读取数据,存入Map中。检查请求类型如下:

//Request method check
if(!HttpMethod.isAccept(cmds[0])) {
return null;
}

接受的请求类型枚举:

public enum HttpMethod {
GET,
POST; public static boolean isAccept(String method) {
for(HttpMethod m : HttpMethod.values()) {
if(m.name().equals(method)) {
return true;
}
}
return false;
} public static HttpMethod getMethod(String method){
for(HttpMethod m : HttpMethod.values()) {
if(m.name().equals(method)) {
return m;
}
}
return null;
}
}

POST 请求需读取 Content-Length 属性,即需要知道POST包中的参数包大小,当TCP包被拆分通过几条链路到达目的地时,根据包长度使得服务端能合理的等待数据到来。

//Read headers
String line;
int contentLength = 0;
HashMap<String,String> headers = new HashMap<String, String>();
while( (line = br.readLine()) != null
&& !line.equals("") ) { int idx = line.indexOf(": ");
if(idx == -1) {
continue;
}
if(HttpHeaders.CONTENT_LENGTH.equals(line)) {
contentLength = Integer.parseInt(line.substring(idx+2).trim());
}
headers.put(line.substring(0, idx), line.substring(idx+2));
}

二、总体设计说明:

1.Main函数开始说明应该的设计方法,有些机制可用于其他软件的设计。

部署结构如下:

%HOME%/lib/*    ----依赖包

%HOME%/conf/*  -----配置文件夹

%HOME%/startup.sh  ---启动SHELL

%HOME%/logs/*    ----日志文件夹

%HOME%/webapps/*  ----页面部署路径

这个设计方法很类似于TOMCAT。ECLIPSE包结构截图如下:

工程启动类 org.mike.jerry.launcher.Main

lib类加载器 org.mike.jerry.launcher.ClassPath

服务加载类 org.mike.jerry.launcher.Bootstrap,该类中读取配置并启动服务端口监听。

配置文件conf/config.properties  默认配置80端口,启动后使用 http://127.0.0.1即可访问。

2. 请求接受与线程池

真正处理请求即为org.mike.jerry.server.SocketConnector ,启动与接受请求:

protected ServerSocket newServerSocket(String host, int port,int backlog) throws IOException{
ServerSocket ss= host==null?
new ServerSocket(port,backlog):
new ServerSocket(port,backlog,InetAddress.getByName(host)); return ss;
} public void accept() throws IOException {
log.info("Server started ..."); while(started){
Socket socket = serverSocket.accept();
ConnectorEndPoint connector = new ConnectorEndPoint(socket);
connector.dispatch();
}
}

每次请求开启一个ConnectorEndPoint线程处理,该线程从线程池中获取(org.mike.jerry.server.util.thread.ThreadPool),处理如下:

 /* Request Handler
*/
protected class ConnectorEndPoint extends SocketEndPoint implements Runnable { public ConnectorEndPoint(Socket socket) throws IOException {
super(socket);
socket.setSoTimeout(7000);
} public void dispatch() {
threadPool.dispatch(this);
} @Override
public void run() {
......
}
}

 3. HTTP包解析器

HTTP包解析类由org.mike.jerry.http.HttpRequestDecoder工作,HTTP请求处理都位于org.mike.jerry.http包中。

请求解析工作有几点:

1. 读取请求头,区分GET POST,获取请求头属性,GET读取URL中的符号“?”并解析参数,POST需要根据Content-Length再读取请求体中的请求参数。

把解析完成的数据存入Request中,根据Servlet设计规范,Request中需要存储请求体放入ServletInputStream in中,以供容器使用者在Servlet中能读取到InputStream.

2. 请求读取完毕后 把Resuqet交与 ResourceHandler 处理,读取所需要请求的资源。

4. 读取资源

资源的读取中,默认请求为/的会固定读取/index.html文件,该属性本应该在web.xml中配置,不过为了学习简易,硬编码于此。

1. 首先检查这路径是否在Servlet中有匹配的,如果没有,则进行下一步。

2. 从webapps文件夹中读取请求的文件,如果不存在,则返回404,如果存在,则进行下一步。

3. 读取请求中的ETag码,这个标志类似于MD5、SHA1等文件摘要,用于标志文件是否改变,如果未改变,则返回304,节省服务器资源(CPU、磁盘与网络等)

,只是MD5与SHA1计算文件摘要需要的CPU周期较长,固计算方法修改如下:

public String getWeakETag() {
try{
StringBuilder b = new StringBuilder(32);
b.append("M/\""); int length=uri.length();
long lhash=0;
for (int i=0; i<length;i++)
lhash= 31*lhash + uri.charAt(i); B64Code.encode(file.lastModified()^lhash, b);
B64Code.encode(length^lhash, b);
b.append('"');
return b.toString();
} catch (IOException e) {
throw new RuntimeException(e);
}
}

5. 如果文件发生改变,则重新读取文件字节流,放入响应包Response中。

 

5. 响应HTTP包封装

5.1 响应头输出: 首先获取socket输出流,再写出头信息,127.0.0.1抓包工具可使用rawcap,得到pcap包后使用wireshark查看,格式类似于:

HTTP/1.1 200 OK
ETag: M/"AJMRnIhabgYAJMQ2H/NnL0"
Date: Wed, 5 Nov 2014 09:58:17 GMT
Content-Length: 1102
Last-Modified: Wed, 2 Jul 2014 23:01:08 GMT
Connection: Keep-Alive
Content-Type: text/html
Server: M
Cache-Control: private

相应代码如:

         OutputStream out = socket.getOutputStream();

                //config status message
String respStat = HttpStatus.getMessage(response.getStatus()); StringBuilder headers = new StringBuilder();
headers.append(response.getHttpVersion() + " "
+ response.getStatus() + " " + respStat + StringUtil.CRLF); //write headers
for(Map.Entry<String, String> header : response.getHeaders().entrySet()){
headers.append(header.getKey() + ": " + header.getValue() + StringUtil.CRLF);
} headers.append(StringUtil.CRLF);//响应头写入完毕必须空一行,这也是协议规定,以区分响应体
out.write(headers.toString().getBytes())

写入响应头后再写入响应体,也就是请求的资源内容。

简易 HTTP Server 实现(JAVA)的更多相关文章

  1. FATAL Fatal error during KafkaServerStable startup. Prepare to shutdown (kafka.server.KafkaServerStartable) java.io.FileNotFoundException: /tmp/kafka-logs/.lock (Permission denied)

    1.启动kafka的时候,报错如下所示: [-- ::,] INFO zookeeper state changed (SyncConnected) (org.I0Itec.zkclient.ZkCl ...

  2. ArcGIS Server 10 Java 版的Rest服务的部署方法

    使用ArcGIS Server 10 Java版发布GIS服务,当使用ArcGIS Manager创建好服务后,然后打开“ArcGIS Services Directory”的链接时发现网页报出了找不 ...

  3. 【Tech】CAS多机部署Server和Java Client端

    昨天尝试把cas的java client端部署到另外一台机器,结果就有问题了.(localhost部署cas server和java client端参见:http://www.cnblogs.com/ ...

  4. mysql/sql server和java之间的数据类型对应关系

    Mysql************************************当前列 ClassName ColumnType DisplaySize TypeName0: java.lang.I ...

  5. kafka启动时出现FATAL Fatal error during KafkaServer startup. Prepare to shutdown (kafka.server.KafkaServer) java.io.IOException: Permission denied错误解决办法(图文详解)

    首先,说明,我kafk的server.properties是 kafka的server.properties配置文件参考示范(图文详解)(多种方式) 问题详情 然后,我启动时,出现如下 [hadoop ...

  6. Language Server for Java™ 1.0 在VS Code上正式发布!

    Nick Zhu form Senior Program Manager, Developer Division at Microsoft 今天,我们很高兴与大家宣布:Language Server ...

  7. ArcGIS Server 10 Java 版的Rest服务手动配置方法

    Java版的Manager中发布的服务默认只发布了该服务的SOAP接口,而REST接口需要用户在信息服务器,如Tomcat. Apache.WebLogic等中手工配置.由于在Java版的Server ...

  8. JAVA 连接 SQL Server 2008:java.lang.ClassNotFoundException: com.microsoft.jdbc.sqlserver.SQLServerDriver

    新项目需要修改Java开发的MES系统...Java忘的也差不多了...简单尝试以下JAVA连接SQL Server吧,没想到坑还是很多的.以前直接连oracle时没有这么多麻烦啊....可能微软和o ...

  9. WebSphere Application Server切换JAVA SDK版本

    最近在Windows Server 2008 R2服务器中搭建了一套IHS+WAS8.5集群环境,测试一个简单的demo应用没有问题,可是在部署正式应用时总是报类版本错误.换了好几个JDK对项目进行编 ...

随机推荐

  1. springmvc 之 DispatcherServlet

    DispatcherServlet说明 使用Spring MVC,配置DispatcherServlet是第一步. DispatcherServlet是一个Servlet,所以可以配置多个Dispat ...

  2. 解决Yii2邮件发送问题(结果返回成功,但接收不到邮件)

    刚刚用了一下yii邮件发送功能,虽然结果返回成功,但接收不到邮件.配置文件代码如下: 'components' => [ 'db' => [ 'class' => 'yii\db\C ...

  3. 基于Spring的最简单的定时任务实现与配置(一)

    朋友的项目中有点问题.他那边是Spring架构的,有一个比较简单的需要定时的任务执行.在了解了他的需求之后,于是提出了比较简单的Spring+quartz的实现方式. 注意本文只是讨论,在已搭建完毕的 ...

  4. ionic 的缓存 和局部刷新

    最近两天在做项目时,发现ionic的缓存功能非常方便好用,提高了再低端手机特别是android比较低版本上的流畅性!可是,后来发现,整体的缓存整个页面并不是一个一劳永逸的办法,结合局部刷新功能,感觉就 ...

  5. mysql+keepalived 双主热备高可用

    理论介绍:我们通常说的双机热备是指两台机器都在运行,但并不是两台机器都同时在提供服务.当提供服务的一台出现故障的时候,另外一台会马上自动接管并且提供服务,而且切换的时间非常短.MySQL双主复制,即互 ...

  6. dubbo 请求调用过程分析

    服务消费方发起请求 当服务的消费方引用了某远程服务,服务的应用方在spring的配置实例如下: <dubbo:referenceid="demoService"interfa ...

  7. undefined variable _session php

    解决方法: if (version_compare(PHP_VERSION, '5.4.0', '<')) { if(session_id() == '') {session_start();} ...

  8. js将时间戳转成格式化的时间

    function getLocalTime(nS){ return new Date(parseInt(nS) * 1000).toLocaleString().replace(/年|月/g, &qu ...

  9. C++基础之引用与指针的区别与联系、常引用使用时应注意的问题

    什么是引用? 引用就是对变量起一个别名,而变量还是原来的变量,并没有重新定义一个变量.例如下面的例子:   #include<iostream> using namespace std; ...

  10. MySQL数据库Raid存储方案

    作为一名DBA,选择自己的数据存储在什么上面,应该是最基本的事情了.但是很多DBA却容易忽略了这一点,我就是其中一个.之前对raid了解的并不多,本文就记录下学习的raid相关知识. 一.RAID的基 ...