从零手写实现 nginx-18-nginx.conf header 信息操作

模式苔原
• 阅读 156

前言

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

我们为 java 开发者实现了 java 版本的 nginx

https://github.com/houbb/nginx4j

如果你想知道 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 的操作处理

目标

我们希望可以通过配置对 header 进行相关操作,比如添加 cors 响应。

这里应该分为两个部分:

1)请求头的统一处理

2)响应头的统一处理

处理又包含 增删改。

指令

抱歉,之前的回答有点冗长了。其实,对于请求头和响应头的增删改操作,各自都有一个主要指令即可。

请求头的增删改指令

  • 增加或修改请求头:使用 proxy_set_header 指令。
  • 删除请求头:同样使用 proxy_set_header 指令,但将值设为空。
proxy_set_header Header-Name Value;  # 增加或修改
proxy_set_header Header-Name "";     # 删除

响应头的增删改指令

  • 增加或修改响应头:使用 add_header 指令。
  • 删除响应头:使用 proxy_hide_header 指令。
add_header Header-Name Value;        # 增加或修改
proxy_hide_header Header-Name;       # 删除

示例

请求头的增删改
server {
    listen 80;
    server_name example.com;

    location / {
        # 增加或修改请求头
        proxy_set_header X-Real-IP $remote_addr;
        
        # 删除请求头
        proxy_set_header X-Unwanted-Header "";

        proxy_pass http://backend;
    }
}
响应头的增删改
server {
    listen 80;
    server_name example.com;

    location / {
        proxy_pass http://backend;

        # 增加或修改响应头
        add_header X-Response-Time $request_time;
        
        # 删除响应头
        proxy_hide_header X-Unwanted-Header;
    }
}

这样就可以简洁地实现对请求头和响应头的增删改操作。

核心代码

配置的初始化

首先我们解析配置文件

# 默认匹配
location / {
    proxy_set_header X-DEFINE-PARAM myDefineParam;
    proxy_set_header X-DEFINE-HOST 127.0.0.1;
    # 增加或修改响应头 这里就提现了一些占位符的强大之处。下一次可以考虑支持
    add_header X-Response-Time 2024-06-08;
    # 删除响应头
    proxy_hide_header X-Unwanted-Header;
}

核心逻辑如下:

private List<NginxUserServerLocationConfig> getHttpServerLocationList(final NgxConfig conf, final NgxBlock serverBlock) {
        List<NginxUserServerLocationConfig> resultList = new ArrayList<>();
        // value
        List<NgxEntry> entryList = serverBlock.findAll(NgxBlock.class, "location");
        if(CollectionUtil.isNotEmpty(entryList)) {
            for(NgxEntry entry : entryList) {
                NgxBlock ngxBlock = (NgxBlock) entry;
                // 参数
                NginxUserServerLocationConfig locationConfig = new NginxUserServerLocationConfig();
                locationConfig.setName(ngxBlock.getName());
                locationConfig.setValue(ngxBlock.getValue());
                locationConfig.setValues(ngxBlock.getValues());

                NginxLocationPathTypeEnum typeEnum = NginxLocationPathTypeEnum.getTypeEnum(locationConfig);
                locationConfig.setTypeEnum(typeEnum);

                // 参数
                List<NginxUserConfigParam> paramList = new ArrayList<>();
                Collection<NgxEntry> ngxEntries = ngxBlock.getEntries();
                if(CollectionUtil.isNotEmpty(ngxEntries)) {
                    for(NgxEntry ngxEntry : ngxEntries) {
                        // 暂时跳过一些注释之类的处理
                        if(!(ngxEntry instanceof NgxParam)) {
                            continue;
                        }

                        NgxParam ngxParam = (NgxParam) ngxEntry;
                        String name = ngxParam.getName();
                        List<String> values = ngxParam.getValues();
                        String value = ngxParam.getValue();

                        NginxUserConfigParam nginxUserConfigParam = new NginxUserConfigParam();
                        nginxUserConfigParam.setName(name);
                        nginxUserConfigParam.setValue(value);
                        nginxUserConfigParam.setValues(values);

                        paramList.add(nginxUserConfigParam);
                    }
                }
                locationConfig.setDirectives(paramList);

                resultList.add(locationConfig);
            }
        }

        // 排序。按照匹配的优先级,从高到底排序
        if(CollectionUtil.isNotEmpty(resultList)) {
            Collections.sort(resultList, new Comparator<NginxUserServerLocationConfig>() {
                @Override
                public int compare(NginxUserServerLocationConfig o1, NginxUserServerLocationConfig o2) {
                    return o1.getTypeEnum().getOrder() - o2.getTypeEnum().getOrder();
                }
            });
        }

        return resultList;
    }

根据配置文件,构建对应的处理类信息。

这里统一设置了优先级,就按照给出的标准:

public enum NginxLocationPathTypeEnum {

    EXACT("EXACT", "精确匹配 (`=`)", 1000),
    PREFIX("PREFIX", "前缀匹配 (^~)", 2000),
    REGEX("REGEX", "正则匹配 (~ 或 ~*)", 3000),
    COMMON_PREFIX("COMMON_PREFIX /prefix", "普通前缀匹配", 4000),
    DEFAULT("DEFAULT", "默认匹配 /", 5000),
    ;

}

配置的处理

在处理请求的时候,我们暂时主要处理两个部分

1)请求头

2)响应头

此处以请求头为例子,暂时如何统一处理请求头

public void dispatch(final NginxRequestDispatchContext context) {
    beforeDispatch(context);
    // 统一的处理
    doDispatch(context);
    // 统一的处理
    afterDispatch(context);
}

我们在分发请求之前,匹配我们定义的各种指令策略

/**
 * 请求头的统一处理
 * @param context 上下文
 */
protected void beforeDispatch(final NginxRequestDispatchContext context) {
    // 参数管理类
    final INginxParamManager paramManager = context.getNginxConfig().getNginxParamManager();
    //1. 当前的配置
    NginxUserServerLocationConfig locationConfig = context.getCurrentUserServerLocationConfig();
    if(locationConfig == null) {
        return;
    }
    List<NginxUserConfigParam> directives = locationConfig.getDirectives();
    if(CollectionUtil.isEmpty(directives)) {
        return;
    }
    // 处理
    for(NginxUserConfigParam configParam : directives) {
        List<INginxParamHandle> handleList = paramManager.paramHandleList(configParam, context);
        if(CollectionUtil.isNotEmpty(handleList)) {
            for(INginxParamHandle paramHandle : handleList) {
                paramHandle.beforeDispatch(configParam, context);
            }
        }
    }
}

INginxParamHandle 处理类

这里统一定义各种指令的操作。以其中一个为例

package com.github.houbb.nginx4j.config.param;

import com.github.houbb.heaven.util.lang.StringUtil;
import com.github.houbb.log.integration.core.Log;
import com.github.houbb.log.integration.core.LogFactory;
import com.github.houbb.nginx4j.config.NginxUserConfigParam;
import com.github.houbb.nginx4j.support.request.dispatch.NginxRequestDispatchContext;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpHeaders;

import java.util.List;

/**
 * 参数处理类 请求头处理
 *
 * @since 0.16.0
 * @author 老马啸西风
 */
public class NginxParamHandleProxySetHeader extends AbstractNginxParamHandle {

    private static final Log logger = LogFactory.getLog(NginxParamHandleProxySetHeader.class);

    /**
     * # 增加或修改请求头
     * proxy_set_header X-Real-IP $remote_addr;
     * # 删除请求头
     * proxy_set_header X-Unwanted-Header "";
     *
     * @param configParam 参数
     * @param context     上下文
     */
    @Override
    public void doBeforeDispatch(NginxUserConfigParam configParam, NginxRequestDispatchContext context) {
        List<String> values = configParam.getValues();

        // $ 占位符号后续处理

        String headerName = values.get(0);
        String headerValue = values.get(1);

        FullHttpRequest fullHttpRequest = context.getRequest();

        // 设置
        HttpHeaders headers = fullHttpRequest.headers();
        if (StringUtil.isEmpty(headerValue)) {
            headers.remove(headerName);
            logger.info(">>>>>>>>>>>> doBeforeDispatch headers.remove({})", headerName);
        } else {
            // 是否包含
            if (headers.contains(headerName)) {
                headers.set(headerName, headerValue);
                logger.info(">>>>>>>>>>>> doBeforeDispatch headers.set({}, {});", headerName, headerValue);
            } else {
                headers.add(headerName, headerValue);
                logger.info(">>>>>>>>>>>> doBeforeDispatch headers.set({}, {});", headerName, headerValue);
            }
        }
    }

    // 省略

    @Override
    public boolean doMatch(NginxUserConfigParam configParam, NginxRequestDispatchContext context) {
        return "proxy_set_header".equalsIgnoreCase(configParam.getName());
    }

}

这里就可以针对配置的信息,执行请求头的处理。

小结

到这里可以发现 nginx 确实非常的强大。

不过我们目前还没有实现各种占位符,感觉这部分也是 nginx 强大的原因之一。

而且各种指令的实现方式也比较多,需要后续陆续补充实现。

点赞
收藏
评论区
推荐文章
blmius blmius
4年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
美凌格栋栋酱 美凌格栋栋酱
7个月前
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(
Wesley13 Wesley13
3年前
VBox 启动虚拟机失败
在Vbox(5.0.8版本)启动Ubuntu的虚拟机时,遇到错误信息:NtCreateFile(\\Device\\VBoxDrvStub)failed:0xc000000034STATUS\_OBJECT\_NAME\_NOT\_FOUND(0retries) (rc101)Makesurethekern
Wesley13 Wesley13
3年前
FLV文件格式
1.        FLV文件对齐方式FLV文件以大端对齐方式存放多字节整型。如存放数字无符号16位的数字300(0x012C),那么在FLV文件中存放的顺序是:|0x01|0x2C|。如果是无符号32位数字300(0x0000012C),那么在FLV文件中的存放顺序是:|0x00|0x00|0x00|0x01|0x2C。2.  
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
PHP创建多级树型结构
<!lang:php<?php$areaarray(array('id'1,'pid'0,'name''中国'),array('id'5,'pid'0,'name''美国'),array('id'2,'pid'1,'name''吉林'),array('id'4,'pid'2,'n
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Linux日志安全分析技巧
0x00前言我正在整理一个项目,收集和汇总了一些应急响应案例(不断更新中)。GitHub地址:https://github.com/Bypass007/EmergencyResponseNotes本文主要介绍Linux日志分析的技巧,更多详细信息请访问Github地址,欢迎Star。0x01日志简介Lin
Python进阶者 Python进阶者
1年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这