`
brandNewUser
  • 浏览: 455415 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

使用nginx进行ab站点的过程简单分析

阅读更多

由于业务需要,在官网上部署两套前端页面,通过特定的字段(例如手机号码)进行分流,来达到a/b站的要求,后续对a/b站最终数据进行分析,选出哪部分页面对用户体验来说会更优秀。

 

nginx请求分流

考虑利用nginx的分流功能:

http://neoremind.com/2012/03/nginx%E6%A0%B9%E6%8D%AEcookie%E5%88%86%E6%B5%81/

 

在mac下使用brew install nginx,安装完成后的目录为:/usr/local/Cellar/nginx/1.10.3(根据不同的版本会有所不同)/,nginx配置文件所在目录:/usr/local/etc/nginx。

 

在Postman中需要安装下载Postman Interceptor 扩展程序,此时就可以通过发送Headers中的内容,来达到发送Cookie的目的:

 



 

 

在nginx中,可以根据该cookie进行匹配判断,决定要发送的服务器upstream:

match cookie
set $stream stream0;
if ($http_cookie ~* "phone=([^;]+)(1$)"){
    set $stream stream1;
}
if ($http_cookie ~* "phone=([^;]+)(2$)"){
    set $stream stream2;
}

 

在上面的示例中,仅能匹配单个http_cookie的最后一行,如果我们想要根据手机尾号进行用户划分的话,必须要匹配多个属性:

 

match cookie
        set $stream stream0;
        if ($http_cookie ~* "phone=([^;]+)([5-9]$)"){
            set $stream stream1;
        }
        if ($http_cookie ~* "phone=([^;]+)([0-4]$)"){
            set $stream stream2;
        }

 

进行范围查找,如果在5-9之间,对应stream1,否则对应stream2,如果没有该cookie,需要给定一个默认值stream0。

 

上述情况出现在用户已经登录的情况下,如果请求是处于注册/登录的过程中,此时并没有cookie数据,但这两种操作都是通过POST请求,在form表单中存在对应的字段手机号(phone),考虑是否可以根据request body中的字段进行填充。

 

nginx中的变量介绍主要如链接中:

https://moonbingbing.gitbooks.io/openresty-best-practices/content/openresty/inline_var.html

 

可以在日志中将 $request_body 打印出来,只要加上 $request_body 属性即可,如果我们加上的数据为“phone=111”

 

------WebKitFormBoundaryq2rbBAdTrAuTi6IG\x0D\x0AContent-Disposition: form-data; name=\x22phone\x22\x0D\x0A\x0D\x0A111\x0D\x0A------WebKitFormBoundaryq2rbBAdTrAuTi6IG--\x0D\x0A

 

可见这些字段是已经经过了额外的转义处理,如果想要分析request body中的字段比较麻烦,nginx只有在修改插件运行的情况下(对nginx本身进行编程),才能访问到request body中的字段。

 

因此我们的方案调整为,注册/登录完成后写Cookie,但不能马上刷新缓存,但可以通过页面上的ajax请求success回调,去强制重刷整个页面来获取a/b站点对应js/css资源,但可能造成额外的流量损耗。

 

 

内部域名解析/转换

 

但我们部署的服务理论上是在两台docker容器上,并无固定ip,是通过不同的内部域名进行处理的,因此在upstream出现域名时,就会发生无法转发的问题,即定义的 http://${url}并不进行替换。

 

upstream main {
      server web1.local:80;
      server web2.local:80;
      server web3.local:80;
    }

 

通过问题查找,参考下面的一篇文章:

http://serverfault.com/questions/598202/make-nginx-to-pass-hostname-of-the-upstream-when-reverseproxying

 

曾经尝试了第一种方式,设置proxy_set_header,并没有起作用:

 

    proxy_set_header Host            $host;
    proxy_set_header X-Forwarded-For $remote_addr;

 

第二种方式理论上应该可行,是通过开放多个端口的方式,建立几个virtual server,但由于我们将系统部署在lain(docker的一种实践)上,限制条件比较多,只能开放一个web端口,因此该方式在lain环境上不可行。

server {
  listen      8001 default_server;
  server_name web1.example.com;
  location / {
    proxy_pass       http://web1.local:80;
    proxy_set_header Host web1.local:80;
  }
}

server {
  listen      8002 default_server;
  server_name web2.example.com;
  location / {
    proxy_pass       http://web2.local:80;
    proxy_set_header Host web2.local:80;
  }
}

server {
  listen      8003 default_server;
  server_name web3.example.com;
  location / {
    proxy_pass       http://web3.local:80;
    proxy_set_header Host web3.local:80;
  }
}

upstream main {
  server 127.0.0.1:8001;
  server 127.0.0.1:8002;
  server 127.0.0.1:8003;
}

server {
  listen      80;
  server_name example.com;
  location / {
    proxy_pass http://main;
  }
}

Tengine提供此支持,http://tengine.taobao.org/document_cn/http_upstream_dynamic_cn.html,但通过测试发现tengine支持的这种方式可能只能利用外网可解析的域名来处理,如果是内网域名仍然是与没有配置该模块的结果相同。

upstream stream80 {
        dynamic_resolve fallback=next fail_timeout=30s;
        #server www.xxx.cn;
        server xxx.xxapp.xyz;
    }

 

转移到xxx.xxapp.xyz,此为内部解析的域名:

 

[Snip20170302_2.png](http://pic.findyou.xin/38257ca397a84fc896e0d9f617f99cc0.png)

 

我们将转移到 www.xxx.cn,会发现已经进行了转换(错误是由于servername名称不匹配)

 

[Snip20170302_2.png](http://pic.findyou.xin/ba9885a13dcb4797ab9f50e473deae58.png)

 

基本判断tengine的这个模块应该是可用的,但域名解析可能用到了一些特殊的条件或算法,导致无法解析我们内网的域名,所以在只能部署单个对外端口的docker容器下,暂时不能解决内网upstream带server_name的问题(最终考虑将其部署在虚拟机上,开启多个端口来解决该问题,也就是参考链接中的第二条)。

 

  • 大小: 22.4 KB
分享到:
评论

相关推荐

    nginx的七个实验

    实验会使用工具(如ab或JMeter)模拟并发请求,分析Nginx的性能表现,以及如何调整配置以优化并发处理。 7. **Nginx日志分析** 日志分析是监控服务器状态和优化性能的关键步骤。Nginx的日志格式灵活,可以自定义。...

    nginx外网访问内网站点配置操作

    站点是前后端分离:vue+springboot 前端内网地址:192.168.1.10:81 API内网地址:192.168.1.12:8080 外网域名:abc.ab.com 外网IP:10.114.X.X 需求: 通过域名可以访问站点且站点静态资源且可访问API请求数据 方案...

    在Linux虚拟机中搭建Nginx服务器以及SSL、反向代理和负载均衡的实现[汇编].pdf

    6. **负载测试**:可以使用工具如`ab`(Apache Bench)对Nginx进行压力测试,以评估其性能和稳定性。 接下来是基于OpenSSL部署HTTPS网站: 1. **OpenSSL和SSL/TLS协议简介**:SSL/TLS协议用于在互联网上提供安全...

    上品网站优化方案1

    描述中提到了AB测试长连接,这是一种评估不同网络连接方式对性能影响的方法,例如使用`ab`工具(ApacheBench)进行压力测试。`-k`参数表示开启长连接,`-c`和`-n`分别定义并发用户数和总的请求次数。 当前上品网站...

    Zend Studio 9.0.1 注册码

    1. **下载与安装**:首先访问官方站点或其他可信来源下载**Zend Studio 9.0.1**安装包,然后按照提示完成安装过程。 2. **激活软件**:启动软件后,进入激活界面输入上述提供的用户名“zendstudio.net”及序列号...

    基于python的车无忧网络平台设计与实现.pdf

    - Python是一种高级编程语言,具备面向对象编程特性,支持多种编程范式,如面向对象、命令式、函数式和过程式。 - 它具有高并发处理能力,适用于开发多任务和网络相关应用程序。 - Python是解释型语言,由Guido ...

    haproxy均衡负载

    例如,在大型Web集群环境中,Haproxy可以与Nginx配合使用,前者负责四层负载均衡,后者则专注于七层应用层面的处理。 #### 四、性能测试与监控 为了确保Haproxy的稳定运行,对其进行性能测试是非常必要的。常见的...

Global site tag (gtag.js) - Google Analytics