一、使用Jmeter cookie的业务背景
业务背景:
我们在工作中经常会遇到如下压测场景:
登录后对查询、充值、购买...等业务进行压测;
这种场景很熟悉吧?很多业务操作是要依赖登录的,但是一个用户在每次进行登录时通常会生成cookie,而且在接下来的操作中都必须携带上它的cookie,服务器对后续的每次请求都会进行校验。
首先想到的是从登录响应中用正则表达式提取出cookies,然后在其它操作中调用不就行了吗?
不行!
因为cookies并不是在登录的响应结果中生成的,而是在response header中携带的,所以不能用正则表达式提取。
那用什么来获取呢?我们先来看一下jmeter官方文档中怎么说的吧:
二、解读Jmeter官方文档
Jmeter官方文档是这样写的:
下面我简单翻译一下:
cookie管理器元件有两大功能:
首先,它能够像web浏览器一样存储和发送cookies。
如果你有一个HTTP请求和响应包含一个cookie,cookie管理器会自动存储该cookie,并能够在该特定网站上接下来的所有请求中使用该 cookie。
JMeter的每个线程都有自己的“cookie存储区”。所以,如果你正在测试一个使用cookie来存储会话信息的网站,那么每个JMeter线程都有自己的会话。请注意,这种cookies不会在cookie管理器中展示出来,但可以在“查看结果树”中看到它们。
JMeter会检查收到的cookies是否对URL有效。这意味着跨域的cookie不能被存储。如果你有窃听行为,或者想要存储跨域cookies,需要在jmeter.poperties中设置:“ CookieManager.check.cookies = false”。
接收到的Cookies可以存储为JMeter的线程变量。如果要把 cookies保存为变量,需要设置:
“ CookieManager.save.cookies =true”。
此外,默认情况下jmeter中的cookies在存储时名字前缀为“ COOKIE_ ”,这是为了避免因局部变量重名引起的异常情况。如果不想用这个前缀,需要定义属性“CookieManager.name.prefix = ”(一个或多个空格)。如果使用了前缀,则一个名称为TEST的cookie的值可以通过$ {COOKIE_TEST}获取 。
其次,你可以手动添加一个cookie到cookie管理器。但是,如果你这样做,这个cookie将被所有JMeter的线程共享。注意,这样的cookies创建后生命周期会很长。
空值的cookies默认被忽略掉。这可以通过设置JMeter的属性进行更改:CookieManager.delete_null_cookies =false。请注意,这也适用手工定义的cookies-任何此类cookies在更新后会从cookie manager中移除。
还需要注意的是cookie名称必须是唯一的-如果一个cookie的名称与已有cookie同名,它将取代原有的cookie。
如果在一个sampler作用域中有多个cookie manager,目前没有办法去指定使用某一个。并且,一个cookie manager中保存的cookie不能被其它的cookie manager使用,所以要慎用多个cookie manger。
有没有觉得很难理解呢?
没关系,下面我们来总结一下:
Cookie Manager主要有两大作用:
1、自动管理cookie:
就像浏览器一样能够自动存储和发送Cookie,如果一个http请求的响应中包含Cookie,那么Cookie Manager就会自动地保存这些Cookie并在所有后来发送到该站点的请求中使用这些Cookie的值。每个线程都自己存储cookie的区域。
在cookie manager中看不到自动保存的cookie,我们可以在察看结果树View Results Tree中看到被发送请求的Cookie Data。
要把Cookies保存到线程变量中,要定义属性"CookieManager.save.cookies=true"。线程变量名为COOKIE_ + Cookie名。属性CookieManager.name.prefix= 可以用来修改默认的COOKIE_的值。
2、手动管理Cookie:
手动添加Cookie到Cookie Manager,这些Cookie的值会被所有线程共享。
比较简单的做法是使用firefox的firebug导出cookies ,然后再导入到jmeter的cookie manager中,当然,这里也可以通过Add按钮手动添加cookies。
Tips:
1、需要注意的是域、路径必须填上,尤其是域;因为Jmeter现版本默认不支持跨域的请求,不填的话设置的Cookie不会被带上。
2、在需要取Cookie的线程里添加一个Http Cookie管理器,可以默认为空,但是一定要添加,否则是不会存储cookie变量的
3、这样在同一个线程(组)内其它操作组件都是可以直接通过${COOKIE_xxxx}来获取
4、目前jmeter在一个sampler中不能同时有多个cookie manager
5、想要跨域存储cookies,需要设置 CookieManager.check.cookies =false
三、线程组内部传递cookies
了解了cookie manager,现在我们回到业务场景中来:
如何实现对查询接口(必须先登录,携带cookie)进行压测?
这是一个很简单的场景,但是如果直接按照上面这样的方式去执行,你会发现结果会报错,因为jmeter默认不会保存cookies,需要加上cookie manager,如下所示:
如上,加一个空的cookie manager即可以自动保存cookie,并且能够在同一个线程组之间传递cookie,此时再次运行脚本就通过了。
但是我们并不需要同时对登录操作进行压测,也就是说,我只需要用一个用户登录一次,然后循环进行其它业务操作,这时候需要把登录和查询操作分别放在两个线程组中,但是jmeter官方文档中说cookies不能跨域传递的,也就是不能在线程组中共享,那要怎么实现呢?
四、跨线程组传递cookies
官方文档中说cookie变量是线程内有效,线程外无效,但是实际测试同一个线程组的其它线程是可以直接访问的。
如果是跨线程组的情况,不能直接访问,可以用其它的方式来进行cookie变量的共享,试验了几种方式,下面这种方式最简单:
通过JMeterUtils.setProperty和JMeterUtils.getPropDefault方法实现
1、在生成cookie的请求中添加beanshell postprocessor后置处理器,并添加如下脚本,把cookie保存在Jmeter属性参数中:
import org.apache.jmeter.util.JMeterUtils;
JMeterUtils.setProperty(“cookie_name”, “cookie_value”);
其中cookie_name和cookie_value分别换成你对应的cookie名称(不带COOKIE_前缀)和cookie值
2、在需要使用cookie的另一个线程组的请求中添加beanshell preprocessor前置处理器,并添加如下脚本,取出cookie的值,并存入线程变量中:
import org.apache.jmeter.util.JMeterUtils;
String value = JMeterUtils.getPropDefault(“cookie_name”);
Vars.put(“cookie_name”,value);
这里的cookie_name和前面设置的名称一样。
然后在该线程组中就可以使用${cookie_name}来获取cookie的值,也就是实现了跨线程组传递cookie。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。