使用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,一经查实,立即删除!

相关文章

带有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头文件中的一…

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

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

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

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

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

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

[转载]Buffon投针实验:究竟为什么是pi?

Buffon投针实验&#xff1a;究竟为什么是pi&#xff1f; Brain Storm | 2009-11-06 20:16| 57 Comments | 本文内容遵从CC版权协议 转载请注明出自matrix67.com数学学习真正悲哀的就是&#xff0c;记住了某个神奇而伟大的定理&#xff0c;看懂了其最严密的推导过程&#xff0c;…

初始化懒惰关系以及何时使用它们的5种方法

实体之间关系的延迟加载是JPA中公认的最佳实践。 它的主要目标是仅从数据库中检索请求的实体&#xff0c;并仅在需要时加载相关实体。 如果我们只需要请求的实体&#xff0c;那是一个很好的方法。 但是&#xff0c;如果我们还需要一些相关实体&#xff0c;它会增加工作量&#…

fieldset ----- 不常用的HTML标签

fieldset 元素可将表单内的相关元素分组。 <fieldset> 标签将表单内容的一部分打包&#xff0c;生成一组相关表单的字段。 当一组表单元素放到 <fieldset> 标签内时&#xff0c;浏览器会以特殊方式来显示它们&#xff0c;它们可能有特殊的边界、3D 效果&#xff…

使用入站适配器公开HTTP Restful API。 第1部分(XML)

1.简介 这篇文章的目的是使用Spring Integration HTTP入站适配器实现HTTP Restful API。 本教程分为两个部分&#xff1a; XML配置示例&#xff08;同一篇文章&#xff09;。 Java DSL示例。 这将在本教程的下一部分中进行说明&#xff0c;展示如何使用Spring Integration Ja…

使用jOOQ和JavaFX将SQL数据转换为图表

最近&#xff0c;我们已经展示了Java 8和函数式编程将如何为使用jOOQ和Java 8 lambda和Streams进行SQL数据的函数数据转换为Java开发人员带来新的视角。 今天&#xff0c;我们将这一步骤更进一步&#xff0c;将数据转换为JavaFX XYChart.Series以根据数据生成美观的条形图。 设…

node.js学习笔记(1)

一&#xff0e; 安装以及环境配置 安装路径 http://nodejs.cn/download/ 多种环境选择 环境变量的配置 Step1 先检查环境变量中的系统变量里面的path,查看是否加入了node.js 例如我的node.js安装路径是C:\Program Files\nodejs 那么&#xff0c;这个path里面就应该加…

主要版本发布后Java开发人员应使用的15种工具

新部署的生存工具包&#xff1a;适用于Java开发人员的工具&#xff0c;这些工具经常将代码部署到生产中&#xff01; Takipi会检测生产中的所有错误&#xff0c;并像发生错误时一样显示变量值 立即部署并获得免费的T恤 适用于新部署的终极生存套件 与在僵尸末日场景下玩弄&…

Java EE 7批处理和魔兽世界–第2部分

今天&#xff0c;我将把第二部分带到我以前关于Java EE 7批处理和《魔兽世界–第1部分》的帖子中。 在本文中&#xff0c;我们将了解如何从第1部分中获得的数据中汇总和提取指标。 概括 批处理目的是下载魔兽世界拍卖行的数据&#xff0c;处理拍卖并提取指标。 这些指标将建立…

js导航条 二级滑动 模仿块级作用域

for(var i 1;i<7;i){    //因为首级标题有6个&#xff0c;对每个首级标题添加mouseover和mouseout事件。    //这里用到块级作用域(function(k){document.getElementById("p_"k).addEventListener(mouseover,function(event){document.getElementById(p_…

struts+swfupload实现批量图片上传(上):swfupload

custom_settings : {progressTarget : "fsUploadProgress",cancelButtonId : "btnCancel",uploadButtonId : "btnUpload",myFileListTarget : "idFileList" },custom_settings调用方法 this.customSettings.cancelButtonId 缩略图js …

40行中的持久性KeyValue Server和一个可悲的事实

再次出现。. 回顾 Peters关于Unsafe用法的书面概述 &#xff0c;我将简要介绍一下Java中的低级技术如何通过启用更高级别的抽象或允许Java性能级别来节省开发工作可能很多人都不知道。 我的主要观点是表明&#xff0c;将对象转换为字节&#xff0c;反之亦然是一个重要的基础&a…

TreeMap源码分析——深入分析(基于JDK1.6)

TreeMap有Values、EntrySet、KeySet、PrivateEntryIterator、EntryIterator、ValueIterator、KeyIterator、DescendingKeyIterator、NavigableSubMap、AscendingSubMap、DescendingSubMap、SubMap、Entry共十三个内部类。Entry是在TreeMap中用于表示树的节点的内部类&#xff0…

Python2.6 Cx_Oracle Linux下编译安装

分类&#xff1a; python Oracle 2012-06-07 00:04 239人阅读 评论(0) 收藏 举报(一) Python 2.6 安装 1.下载Python2.6.X 版本的源码包&#xff0c;这里采用平台编译安装。 Python-2.6.4.tar.bz2 2.解压缩 ,使用J参数解压bigz2类型的压缩文件 tar -jxvf Python-2.6.4.tar.bz2…

Apache TomEE(和Tomcat)的自签名证书

可能在大多数Java EE项目中&#xff0c;您将拥有具有SSL支持&#xff08; https &#xff09;的部分或整个系统&#xff0c;因此浏览器和服务器可以通过安全连接进行通信。 这意味着在处理数据之前&#xff0c;已发送的数据已加密&#xff0c;传输并最终解密。 问题在于&…