前言

大家好,我是老马。很高兴遇到你。

我们希望实现最简单的 http 服务信息,可以处理静态文件。

如果你想知道 servlet 如何处理的,可以参考我的另一个项目:

手写从零实现简易版 tomcat minicat

手写 nginx 系列

如果你对 nginx 原理感兴趣,可以阅读:

从零手写实现 nginx-01-为什么不能有 java 版本的 nginx?

从零手写实现 nginx-02-nginx 的核心能力

从零手写实现 nginx-03-nginx 基于 Netty 实现

从零手写实现 nginx-04-基于 netty http 出入参优化处理

从零手写实现 nginx-05-MIME类型(Multipurpose Internet Mail Extensions,多用途互联网邮件扩展类型)

从零手写实现 nginx-06-文件夹自动索引

从零手写实现 nginx-07-大文件下载

从零手写实现 nginx-08-范围查询

从零手写实现 nginx-09-文件压缩

从零手写实现 nginx-10-sendfile 零拷贝

从零手写实现 nginx-11-file+range 合并

从零手写实现 nginx-12-keep-alive 连接复用

从零手写实现 nginx-13-nginx.conf 配置文件介绍

从零手写实现 nginx-14-nginx.conf 和 hocon 格式有关系吗?

从零手写实现 nginx-15-nginx.conf 如何通过 java 解析处理?

从零手写实现 nginx-16-nginx 支持配置多个 server

从零手写实现 nginx-17-nginx 默认配置优化

从零手写实现 nginx-18-nginx 请求头+响应头操作

从零手写实现 nginx-19-nginx cors

从零手写实现 nginx-20-nginx 占位符 placeholder

从零手写实现 nginx-21-nginx modules 模块信息概览

从零手写实现 nginx-22-nginx modules 分模块加载优化

从零手写实现 nginx-23-nginx cookie 的操作处理

从零手写实现 nginx-24-nginx IF 指令

从零手写实现 nginx-25-nginx map 指令

从零手写实现 nginx-26-nginx rewrite 指令

从零手写实现 nginx-27-nginx return 指令

从零手写实现 nginx-28-nginx error_pages 指令

从零手写实现 nginx-29-nginx try_files 指令

从零手写实现 nginx-30-nginx proxy_pass upstream 指令

从零手写实现 nginx-31-nginx load-balance 负载均衡

从零手写实现 nginx-32-nginx load-balance 算法 java 实现

从零手写实现 nginx-33-nginx http proxy_pass 测试验证

从零手写实现 nginx-34-proxy_pass 配置加载处理

从零手写实现 nginx-35-proxy_pass netty 如何实现?

基本实现

思路

基于 serverSocket 实现最基本的监听。

核心

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public void start() { try { // 服务器监听的端口号 int port = nginxConfig.getHttpServerListen(); ServerSocket serverSocket = new ServerSocket(port); log.info("[Nginx4j] listen on port={}", port); while (true) { Socket socket = serverSocket.accept(); log.info("[Nginx4j] Accepted connection from address={}", socket.getRemoteSocketAddress()); handleClient(socket); } } catch (Exception e) { log.info("[Nginx4j] meet ex", e); throw new RuntimeException(e); } }

socket 的处理

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
private void handleClient(Socket socket) { try { // 基本信息 InputStream input = socket.getInputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader(input)); String header = reader.readLine(); String[] parts = header.split(" "); String method = parts[0]; String path = parts[1]; String protocol = parts[2]; // 根路径 final String basicPath = nginxConfig.getHttpServerRoot(); // 只处理GET请求 if ("GET".equalsIgnoreCase(method)) { //root path if(StringUtil.isEmpty(path) || "/".equals(path)) { log.info("[Nginx4j] current path={}, match index path", path); byte[] fileContent = tryGetIndexContent(); sendResponse(socket, 200, "OK", fileContent); return; } // other File file = new File(basicPath + path); if (file.exists()) { byte[] fileContent = Files.readAllBytes(file.toPath()); sendResponse(socket, 200, "OK", fileContent); } else { sendResponse(socket, 404, "Not Found", "File not found.".getBytes()); } } else { sendResponse(socket, 405, "Method Not Allowed", "Method not allowed.".getBytes()); } } catch (Exception e) { // 异常处理... } }

响应的处理

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
private void sendResponse(Socket socket, int statusCode, String statusMessage, byte[] content) throws IOException { OutputStream output = socket.getOutputStream(); PrintWriter writer = new PrintWriter(output, true); // 发送HTTP响应头 writer.println("HTTP/1.1 " + statusCode + " " + statusMessage); writer.println("Content-Type: text/plain"); writer.println("Content-Length: " + content.length); writer.println("Connection: close"); writer.println(); // 发送HTTP响应体 output.write(content); output.flush(); }

测试

快速开始

maven 依赖

  [xml]
1
2
3
4
5
<dependency> <groupId>com.github.houbb</groupId> <artifactId>nginx4j</artifactId> <version>0.1.0</version> </dependency>

启动测试

  [java]
1
Nginx4jBs.newInstance().init().start();

启动日志:

  [plaintext]
1
2
[DEBUG] [2024-05-24 23:40:37.573] [main] [c.g.h.l.i.c.LogFactory.setImplementation] - Logging initialized using 'class com.github.houbb.log.integration.adaptors.stdout.StdOutExImpl' adapter. [INFO] [2024-05-24 23:40:37.576] [main] [c.g.h.n.s.s.NginxServerSocket.start] - [Nginx4j] listen on port=8080

页面访问:http://127.0.0.1:8080

响应:

  [plaintext]
1
Welcome to nginx4j!

其他页面

访问 http://localhost:8080/1.txt

将返回对应的文件内容:

  [plaintext]
1
hello nginx4j!

开源地址

为了便于大家学习,已经将 nginx 开源

https://github.com/houbb/nginx4j

参考资料

https://nginx.org/en/