使用Jasmine,Spock和Nashorn测试JVM服务器端JavaScript

JavaScript使用不仅限于浏览器中的客户端代码或NodeJS支持的服务器端代码。 许多基于JVM的项目都将其用作内部脚本语言。 测试这种功能既不简单也不标准。 在本文中,我打算演示一种使用成熟的工具(例如JasmineSpockNashorn在服务器端JVM环境中测试JavaScript的方法。

与客户端编码相比,在JVM应用程序内部使用JavaScript作为脚本引擎有很大的不同。 不幸的是,如今没有用于测试它的工业标准工具。


关于Internet中现有的方法,我想强调以下缺点:

  • 缺乏与构建和持续集成工具(Maven,Gradle,Jenkins等)的集成
  • 与IDE的合作不足
    • 无法运行单个套件或通过IDE进行测试
  • 与浏览器环境紧密耦合
  • 无法使用自定义的JavaScript执行程序

据我所知,大多数项目通过调用JS引擎运行器,将被测脚本传递给它并通过检查脚本执行后对引擎或模拟的副作用进行断言来测试其嵌入式业务脚本。

这些方法通常具有类似的缺点:

  • 难以对JS代码进行存根或模拟,通常会导致对JS prototype黑客攻击
  • 脚本的模拟环境需要过多的编排
  • 难以将测试组织到套件中并报告测试执行错误
  • 以前的原因为特定项目创建了自定义测试套件框架
  • 不利用现有JavaScript测试工具和框架

因此,在JVM项目中需要舒适的嵌入式JavaScript测试的推动下,我创建了此示例设置。 为了实现我们的目标,将使用下一个工具。

  • Jasmine是最著名JavaScript TDD / BDD工具之一
  • Spock是由Junit和Groovy支持的JVM的出色测试框架
  • Nashorn是JDK8中引入的现代脚本引擎

定制JavaScript运行器(基于Nashorn)

在非浏览器JS环境中不需要遵循标准,因此开发人员通常使用自定义函数,内置变量等扩展脚本引擎。对于生产和测试目的,使用完全相同的运行程序极为重要。

让我们考虑一下,我们有这样的自定义运行器,接受脚本名称和预定义变量的映射作为参数并返回执行脚本的结果值。

JavaScriptRunner.java

public class JavaScriptRunner {public static Object run(String script, Map<String, Object> params) throws Exception {ScriptEngineManager factory = new ScriptEngineManager();ScriptEngine engine = factory.getEngineByName("nashorn");engine.getBindings(ScriptContext.ENGINE_SCOPE).putAll(params);return engine.eval(new InputStreamReader(JavaScriptRunner.class.getResourceAsStream(script))); (1)}
}
1个 在类路径中搜索脚本源。

茉莉花的设置

要开始使用Jasmine框架,我们需要:

  • 下载Jasmine并将其解压缩到项目资源目录中的/jasmine/jasmine-2.1.2文件夹中
  • 自定义引导脚本,因为Jasmine不支持基于JVM的平台

jasmine2-bootstrap.js

var loadFromClassPath = function(path) { (1)load(Java.type("ua.eshepelyuk.blog.nashorn.Jasmine2Specification").class.getResource(path).toExternalForm());
};var window = this;loadFromClassPath("/jasmine/jasmine-2.1.2/jasmine.js");
loadFromClassPath("/jasmine/jasmine2-html-stub.js"); (2)
loadFromClassPath("/jasmine/jasmine-2.1.2/boot.js");
load({script: __jasmineSpec__, name: __jasmineSpecName__}); (3)onload(); (4)jsApiReporter.specs(); (5)
1个 helper函数从类路径位置解析脚本路径。
2 Nashorn专用代码可在非浏览器环境中调整Jasmine 。 不属于Jasmine发行。
3 加载测试套件源代码,有关详细信息,请参见下一部分。
4 伪造浏览器load事件,应触发测试套件执行。
5 该值将作为脚本结果返回。

将Jasmine报告转换为Spock测试

使用Jasmine JS执行程序和引导脚本,我们可以创建JUnit测试以遍历套件结果并检查是否全部成功。 但是,了解哪个特定测试失败以及失败的原因将成为一场噩梦。 我们真正想要拥有的是将每个Jasmine规范表示为JUnit测试的功能,因此任何Java工具都可以拾取并检查结果。 这就是为什么Spock可以解决问题的原因,它的数据驱动测试允许开发人员声明输入数据列表,并针对该数据集的每个项目创建并执行新测试。 这与Junit 参数化测试非常相似,但功能更强大。

因此,想法是将运行引导脚本后获得的Jasmine测试套件结果视为输入数据数组,其每一项都将传递给Spock测试。 然后测试本身将提供断言以正确报告成功和失败的测试,即断言应检查Jasmine规范的状态。

  • 如果状态为pendingpassed ,则表示规范被忽略或成功
  • 否则, Spock测试应该抛出断言错误,并用Jasmine报告的失败消息填充断言异常

Jasmine2Specification.groovy

abstract class Jasmine2Specification extends Specification {@Shared def jasmineResultsdef setupSpec() {def scriptParams = ["__jasmineSpec__"    : getMetaClass().getMetaProperty("SPEC").getProperty(null), (1)"__jasmineSpecName__": "${this.class.simpleName}.groovy"]jasmineResults = JavaScriptRunner.run("/jasmine/jasmine2-bootstrap.js", scriptParams) (2)}def isPassed(def specRes) {specRes.status == "passed" || specRes.status == "pending"}def specErrorMsg(def specResult) {specResult.failedExpectations.collect {it.value}.collect {it.stack}.join("\n\n\n")}@Unroll def '#specName'() {expect:assert isPassed(item), specErrorMsg(item) (3)where:item << jasmineResults.collect { it.value }specName = (item.status != "pending" ? item.fullName : "IGNORED: $item.fullName") (4)}
}
1个 Jasmine套件的源代码公开为jasmineSpec变量,可让JS执行器访问。
2 Jasmine套件的实际执行。
3 对于每个套件结果,我们assert要么成功,要么在失败时使用Jasmine发起的消息引发断言错误。
4 附加的数据提供程序变量以突出显示被忽略的测试。

完整的例子

让我们为简单JavaScript函数创建测试套件。

mathUtils.js

var add = function add(a, b) {return a + b;
};

使用上一步中的基类,我们可以创建包含JavaScript测试的Spock套件。 为了演示我们解决方案的所有可能性,我们将创建成功,失败和被忽略的测试。

MathUtilsTest.groovy

class MathUtilsTest extends Jasmine2Specification {static def SPEC = """ (1)
loadFromClassPath("/js/mathUtils.js"); (2)
describe("suite 1", function() {it("should pass", function() {expect(add(1, 2)).toBe(3);});it("should fail", function() {expect(add(1, 2)).toBe(3);expect(add(1, 2)).toBe(0);});xit("should be ignored", function() {expect(add(1, 2)).toBe(3);});
})
"""
}
1个 Jasmine套件的实际代码表示为String变量。
2 使用从jasmine-bootstrap.js继承的功能加载受测模块。


图1. IntelliJ IDEA的测试结果


图1. IntelliJ IDEA的测试结果

IntelliJ Idea语言注入

尽管此微框架可以在所有IDE中使用,但由于其语言注入 ,它的最方便用法将在IntelliJ IDEA中 。 该功能允许将任意语言嵌入到以其他编程语言创建的文件中。 因此,我们可以将JavaScript代码块嵌入到用Groovy编写的Spock规范中。

图2.语言注入

图2.语言注入

解决方案的优缺点

优点

  • 使用两种语言的行业标准测试工具
  • 与构建工具和持续集成工具的无缝集成
  • 从IDE运行单个套件的能力
  • 借助Jasmine的突出功能,可以从特定套件运行单个测试

缺点

  • 在测试异常的情况下,没有干净的方法来检测特定的源代码行
  • 一点面向IntelliJ IDEA的设置

聚苯乙烯

在此示例项目中,我使用了JDK8的现代Nashorn引擎。 但实际上对此没有限制。 同样的方法已成功应用于使用较旧Rhino引擎的项目。 再说一次, Jasmine只是我的个人喜好。 随着其他工作代码的调整,可以利用MochaQUnit等。

  • 完整的项目代码可在My GitHub上获得 。

翻译自: https://www.javacodegeeks.com/2014/12/testing-jvm-server-side-javascript-with-jasmine-spock-and-nashorn.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/361096.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#中的多态

封装、继承、多态&#xff0c;面向对象的三大特性&#xff0c;前两项理解相对容易&#xff0c;但要理解多态&#xff0c;特别是深入的了解&#xff0c;对于初学者而言可能就会有一定困难了。我一直认为学习OO的最好方法就是结合实践&#xff0c;封装、继承在实际工作中的应用随…

AJAX JSON

1、AJAX [1] AJAX简介 > 全称&#xff1a; Asynchronous JavaScript And XML > 异步的JavaScript和XML > AJAX就是通过JavaScript向服务器发送请求&#xff0c;并接收响应&#xff0c;然后我们在通过DOM来修改页面。 > XML指的是服务器响应的…

在WildFly 8.2中修补焊接3 – Java EE 8的第一个实验RI

Java EE 8一直在发展&#xff0c;并且已经提出了几个新的组件JSR。 JSR 365将定义CDI 2.0的规范。 红帽公司已经开始研究Weld 3的实现原型&#xff0c;并且Alpha3最近发布了 。 Red Hat的Java EE 8兼容应用服务器将是WildFly&#xff0c;将在其中实现所有不同的技术。 同时&am…

mat-form-field must contain a MatFormFieldControl错误的解决方法

下面的代码竟然出错了&#xff1a; <mat-form-field><input matInput placeholder"输入名称"></mat-form-field> 错误提示的莫名其妙&#xff0c;其实只要导入以下模块就可了&#xff1a; imports: [MatFormFieldModule,MatInputModule,] 更多专业…

lua# lua5.1.4 源码文件作用一览

写了个脚本列出lua源码C文件头部的注释&#xff0c;作为我有一搭没一搭以Lua为对象学习编译原理的开端。 lua5.1.4全部的源码有35个C文件&#xff0c;17216行代码。每个文件基本的功能如下 ./output_lua_sources_comments.sh ~/resources/sources/lua/src …

带有Hibernate OGM的NoSQL –第一部分:持久化您的第一个实体

Hibernate OGM的第一个最终版本已经发布 &#xff0c;团队从发布狂潮中恢复了一些。 因此&#xff0c;他们考虑建立一系列教程式博客&#xff0c;使您有机会轻松地从Hibernate OGM重新开始。 感谢Gunnar Morling&#xff08; gunnarmorling &#xff09;创建了本教程。 介绍 不…

为自己写程序之JavsScript代码段测试器

JavaScript的测试&#xff0c;通常是在Firefox的firebug插件中测试的。不过有时只是测试几行代码都要写一个html&#xff0c;再打开浏览器测试运行结果&#xff0c;感觉并不是很方便。 今天花了点时间做了一个简易的JS片段测试器。其实这主要是看了IronJs开源项目以后&#xff…

sizeof和strlen的区别(其中涉及NUL的讲解)

本文是自己结合平时所学的知识&#xff0c;对sizeof和strlen的区别进行了总结&#xff0c;如有不对的地方还请批评指证&#xff0c;共同进步&#xff01;&#xff01;&#xff01; 一、从C语言的定义上来讲 1、sizeof是关键字&#xff0c;而strlen是包含在string.h头文件中的一…

我的Wiki:使用JConsole对WildFly(或JBoss AS7)进行远程JMX访问

与以前的版本相比&#xff0c;JBoss AS7的目标之一是使其在默认情况下更加安全。 受此目标直接影响的领域之一是&#xff0c;您不再期望服务器在端口上公开某些服务&#xff0c;而无需任何身份验证/授权就可以访问它。 请记住&#xff0c;在以前的JBoss AS版本中&#xff0c;只…

js判断对象类型

1.typeof typeof只能判断区分基本类型&#xff0c;number、string、boolean、undefined和object,function&#xff1b; typeof 0; //number; typeof true; //boolean; typeof undefined; //undefined; typeof "hello world" //string; typeof function(){}; …

我喜欢的类型

http://v.qq.com/cover/h/hfd581s2y9unvy8.html?vidp0011ocge8q 转载于:https://www.cnblogs.com/sliz/archive/2012/12/09/2809742.html

使用WildFly和Java EE 7映像与Docker提供者一起流浪

什么是无业游民&#xff1f; Vagrant是创建虚拟开发环境的简化且可移植的方式。 它可与多种虚拟化软件一起使用&#xff0c;例如VirtualBox&#xff0c;VMWare&#xff0c;AWS等。 它还可以与多种配置软件一起使用&#xff0c;例如Ansible&#xff0c;Chef&#xff0c;Puppet或…

Apache Nutch 1.6 发布

Apache Nutch 1.6 发布&#xff0c;该版本修复了超过 20 个 bug&#xff0c;新功能包括&#xff1a;新的 HostNormalizer&#xff0c;可通过 MIME-type 和 Indexer API 的功能增强来动态设置 fetchInterval &#xff0c;更新 Tika 到 1.2 版本&#xff0c;更新 Autimaton 到 1.…

EE Servlet 3:如何在Web应用程序中设置后端服务

在Web应用程序中&#xff0c;提供用户界面&#xff08;UI&#xff09;通常只是工作的一半。 许多应用程序都有后端服务支持的要求。 后端服务的一些示例是调度程序进程&#xff08;批处理&#xff09;&#xff0c;侦听队列并在消息进入时作出响应&#xff0c;或者是简单的事情&…

es6解构赋值

解构赋值语法是一个 Javascript 表达式,这使得可以将值从数组或属性从对象提取到不同的变量中。 数组解构赋值&#xff1a; {let a,b,rest;[a,b][1,2];console.log(a,b); //1 2 } {let a,b,rest;[a,b,...rest][1,2,3,4,5]; console.log(a,b,rest); //1 2 [ 3, 4, 5 ]…

Jquery插件之ajaxForm

如今ajax满天飞&#xff0c;作为重点的form自然也受到照顾。 其实&#xff0c;我们在平常使用Jquery异步提交表单&#xff0c;一般是在submit()中&#xff0c;使用$.ajax进行。比如&#xff1a; $(function(){$(#myForm).submit(function(){$.ajax({url:"/WebTest/test/te…

休眠锁定模式–乐观锁定模式如何工作

显式乐观锁定 在上一篇文章中 &#xff0c;我介绍了Java持久性锁定的基本概念。 隐式锁定机制可防止丢失更新 &#xff0c;它适用于我们可以主动修改的实体。 虽然隐式乐观锁定是一种广泛使用的技术&#xff0c;但是很少有人了解显式乐观锁定模式的内部工作原理。 当锁定的实…

CSS中可继承的属性

不可继承的属性太多了不要背&#xff0c;记住可以继承的属性有哪些就行了。可以继承的属性很少&#xff0c;只有颜色&#xff0c;文字&#xff0c;字体间距行高对齐方式&#xff0c;和列表的样式可以继承。这么来记很轻松的呀&#xff01;不要被下边的吓到了哦~所有元素可继承&…

如何在JMeter中执行客户端Web性能测试?

在本文中&#xff0c;我们将看到如何使用Jmeter插件进行客户端性能测试。 我将使用jmeter webdriver插件。 在开始本主题之前&#xff0c;请从我以前的文章中获得有关客户端性能测试的一些基本信息。 因此&#xff0c;让我们开始吧&#xff1a; 安装 通过这篇文章之后的链接&…

inline-block的兼容性问题

我们都知道在IE6 7 中用*display&#xff1a;block&#xff1b;*zoom&#xff1a;1&#xff1b;可以解决 inline-block 的兼容问题 很多人认为IE6 7 是不支持inline-block的&#xff0c;严格来说应该是&#xff1a;IE6 7 对 inline-block 支持的不够完全 这个要分两种情况来说&…