ANTLR和StringTemplate实例:自动生成单元测试类
ANTLR和StringTemplate实例:自动生成单元测试类
1. ANTLR语法
要想自动生成单元测试,首先第一步就是分析被测试类。这里以Java代码为例,用ANTLR对Java代码进行分析。要想靠自己完全手写出一门语言的ANTLR语法文件的复杂程度难以想象,很贴心的是在ANTLR的GitHub网站上列出了很多常见语言的语法文件,例如Java,Sqlite和MySQL的SQL语法等。
有了.g4语法文件,按照Antlr v4入门教程和实例中的步骤,就能自动生成出解析器的代码,这里就不再详述了。
2. StringTemplate基础
StringTemplate(简称ST)也是ANTLR提供的一个非常好用的工具。它的功能类似于Velocity、FreeMaker等模板引擎,可以根据事先定义好的模板,在运行时根据不同的传值渲染出不同的网页、邮件、代码等。但从名称中的String就能看出,它是比较轻量级的。试用了一下的确如此,支持在Java中硬编码简单的模板。
但试用过程中还是碰到了有不少问题,总感觉它的模板语法有些复杂啊!而且最新的ST4的API与之前版本发生了很大变化,网上找的很多例子都不好用了。具体还是参考官网的教程吧,以及这个CheatSheet表格。当然,还有花了不少时间调试好的本文的代码示例!
3. 单元测试生成器
首先来看Main方法。输入文本就是code变量表示的一个Java类,里面有两个方法。然后使用自动生成出的ANTLR代码,构建起语义分析器和解析器的处理链,并传入UnitTestGenerator监听器对输入文本进行遍历。
public class JavaCodeParseTest {
public static void main(String[] args) {
String code =
"package com.jcache.store;" +
"public class CacheStore {" +
"Object getCache(int a) {" +
"if (a == 1)" +
"return 1;" +
"else " +
"return 2;" +
"}" +
"void setCache(int a) {" +
"return;" +
"}" +
"}";
// 1.Lexical analysis
JavaLexer lexer = new JavaLexer(new ANTLRInputStream(code));
CommonTokenStream tokens = new CommonTokenStream(lexer);
// 2.Syntax analysis
JavaParser parser = new JavaParser(tokens);
ParseTree tree = parser.compilationUnit();
// 3.Application based on Syntax Tree
ParseTreeWalker walker = new ParseTreeWalker();
walker.walk(new UnitTestGenerator(), tree);
}
}
下面就来看一个核心代码UnitTestGenerator的实现。
这里简单说一下代码中的几个关键点:
- ST模板组:在模板文件中定义比较简单,如果要在Java中定义模板的话,一定要参照本例static初始化块的写法。
- 嵌套子模板和Multi-valued:要想根据方法名列表,自动为每个方法名都生成一个方法,就需要按照
/**
* Simple unit test generator.
*/
public class UnitTestGenerator extends JavaBaseListener {
/** Constants: template name, placeholder name, generated code name */
private static final String CLASS_ST_NAME = "classST";
private static final String METHOD_ST_NAME = "methodST";
private static final String TEST_PKG_NAME = "testPkgName";
private static final String TEST_CLASS_NAME = "testClassName";
private static final String TEST_METHOD_NAME = "testMethodName";
private static final String CLASS_NAME_SUFFIX = "Test";
private static final String METHOD_NAME_PREFIX = "test";
/** Template for Java */
private static final String METHOD_ST =
t("@Test") +
t("public void " + $(TEST_METHOD_NAME) + "() throws Exception {") +
tt("//body...") +
t("}") +
n("");
private static final String CLASS_ST =
n("package " + $(TEST_PKG_NAME) + ";") +
n("") +
n("import org.junit.*;") +
n("") +
n("public class " + $(TEST_CLASS_NAME) + " {") +
n("") +
/**
* Apply nested template 'methodST' to multi-valued attributes 'testMethodName'.
* NOTE: <attribute:template(argument-list)>
* Apply template to attribute with optional argument-list.
* Example: <name:bold()> applies bold() to name's value.
* The first argument of the template gets the iterated value.
*/
n($(TEST_METHOD_NAME + ":" + METHOD_ST_NAME +"();separator=\"\n\"")) +
n("}");
/** ST group to nest template */
private static STGroup group;
static {
group = new STGroup('$', '$');
CompiledST classST = group.defineTemplate(CLASS_ST_NAME, CLASS_ST);
classST.addArg(new FormalArgument(TEST_PKG_NAME));
classST.addArg(new FormalArgument(TEST_CLASS_NAME));
classST.addArg(new FormalArgument(TEST_METHOD_NAME));
CompiledST methodST = group.defineTemplate(METHOD_ST_NAME, METHOD_ST);
methodST.addArg(new FormalArgument(TEST_METHOD_NAME));
}
/** Attributes. NOTE: group.getInstanceOf() return new ST */
private Map<String, Object> attributeMap = new HashMap<>();
@Override
public void enterPackageDeclaration(@NotNull JavaParser.PackageDeclarationContext ctx) {
attributeMap.put(TEST_PKG_NAME, ctx.qualifiedName().getText());
}
@Override
public void enterClassDeclaration(@NotNull ClassDeclarationContext ctx) {
String orgClassName = ctx.Identifier().getText();
String testClassName = orgClassName + CLASS_NAME_SUFFIX;
attributeMap.put(TEST_CLASS_NAME, testClassName);
}
@Override
public void enterMethodDeclaration(@NotNull MethodDeclarationContext ctx) {
String orgMethodName = ctx.Identifier().getText();
String testMethodName = METHOD_NAME_PREFIX + orgMethodName.substring(0, 1).toUpperCase()
+ orgMethodName.substring(1);
// Multi-valued attribute
List<String> methodNames = (List<String>) attributeMap.get(TEST_METHOD_NAME);
if (methodNames == null) {
methodNames = new ArrayList<>();
attributeMap.put(TEST_METHOD_NAME, methodNames);
}
methodNames.add(testMethodName);
}
@Override
public void enterStatement(@NotNull StatementContext ctx) {
// If/else/switch, for/while, try-catch
switch (ctx.getStart().getText()) {
case "if":
break;
case "for":
break;
case "try":
break;
default:
break;
}
}
@Override
public void exitCompilationUnit(@NotNull JavaParser.CompilationUnitContext ctx) {
ST template = group.getInstanceOf(CLASS_ST_NAME);
attributeMap.entrySet().forEach(e -> template.add(e.getKey(), e.getValue()));
System.out.println(template.render());
// Cleanup
attributeMap.clear();
}
// =======================================
// String Utility
// =======================================
private static String $(String attrName) {
return "$" + attrName + "$";
}
private static String n(String str) {
return str + "\n";
}
private static String t(String str) {
return "\t" + n(str);
}
private static String tt(String str) {
return "\t\t" + n(str);
}
}
ANTLR和StringTemplate实例:自动生成单元测试类的更多相关文章
- 使用T4为数据库自动生成实体类
T4 (Text Template Transformation Toolkit) 是一个基于模板的代码生成器.使用T4你可以通过写一些ASP.NET-like模板,来生成C#, T-SQL, XML ...
- .net core 中简单封装Dapper.Extensions 并使用sqlsuger自动生成实体类
引言 由公司需要使用dapper 同时支持多数据库 又需要支持实体类 又需要支持sql 还需要支持事务 所以采用了 dapper + dapperExtensions 并配套 生成实体类小工具的方 ...
- Mybatis自动生成实体类
Maven自动生成实体类需要的jar包 一.pom.xml中 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns ...
- Springboot mybatis generate 自动生成实体类和Mapper
https://github.com/JasmineQian/SpringDemo_2019/tree/master/mybatis Springboot让java开发变得方便,Springboot中 ...
- 【原创】有关Silverlight中自动生成的类中 没有WCF层edmx模型新加入的对象 原因分析。
前端页面层: 编译老是不通过,报如下如所示错误: -- 然后下意识的查了下 生成的cs文件,没有搜到根据edmx 生成的 对应的类. 结果整理: 1.尽管在 edmx 模 ...
- Mybatis自动生成实体类、dao接口和mapping映射文件
由于Mybatis是一种半自动的ORM框架,它的工作主要是配置mapping映射文件,为了减少手动书写映射文件,可以利用mybatis生成器,自动生成实体类.dao接口以及它的映射文件,然后直接拷贝到 ...
- mybatis怎样自动生成java类,配置文件?
其实没有什么东西是可以自动生成的,只不过是别人已经写好了,你调用罢了. 所以想要mybatis自动生成java类,配置文件等,就必须要一些配置和一些jar包.当然这些配置也很简单. 为了有个初步的认识 ...
- Asp.Net Core如何根据数据库自动生成实体类
通过引用Nuget包添加实体类 运行 Install-Package Microsoft.EntityFrameworkCore.SqlServer 运行 Install-Package Micros ...
- .NET SourceGenerators 根据 HTTPAPI 接口自动生成实现类
目录 摘要 元数据分析 使用 Source generators 实现 使用 Source generators 实现程序集分析 使用方法 SourceCode && Nuget pa ...
随机推荐
- php+MySql实现登录系统与输出浏览者信息功能
这篇文章主要介绍了php+MySql实现登录系统与输出浏览者信息功能 的相关资料,需要的朋友可以参考下 本系统,与之前在<ASP 连接Access数据库的登陆系统>(点击打开链接)一文 ...
- poj1182-食物链-带权并查集-种类并查集
(这应该是我写的第一个和带权并查集相关的题,还不是很了解,所以这篇博客我以后还会做修改,力求更号理解! 题意和思路: 中文题意,我简单提一下: A->B,B->C,C->A.A吃B, ...
- .Net中集合排序还可以这么玩
背景: public class StockQuantity { public StockQuantity(string status, DateTime dateTime, int quantity ...
- 解析spring中的BeanFactory
我们常把spring看作一个bean工厂或者ioc容器,它帮助我们负责对象的创建管理,以及对象间依赖关系的建立,还有其他的功能. 关于工厂的实现,一般来说与我们接触最多的就是BeanFactory和A ...
- 一日一练-CSS CSS中percentage百分值的使用
子曰:学好百分值,考试考百分 首先是确定CSS 中的percentage 都可以应用在CSS 中的哪些属性,以及这些属性的值如何进行计算的,参考CSS 参考手册进行统计. 定位(Positioning ...
- 深入解析 SQL Server 高可用镜像实现原理
作者:郭忆 本文由 网易云 发布. SQL Server 是 windows 平台 .NET 架构下标配数据库解决方案,与 Oracle.MySQL 共同构成了 DB-Engines Ranking ...
- 极客验证官方demo构建使用及代码分析
#什么是极客验证? 官方定义:极验验证是一种在计算机领域用于区分自然人和机器人的,通过简单集成的方式,为开发者提供安全.便捷的云端验证服务. #使用命令从github上获取: git clone ht ...
- [bzoj4405][wc2016]挑战NPC
来自FallDream的博客,未经允许,请勿转载,谢谢. 小N最近在研究NP完全问题,小O看小N研究得热火朝天,便给他出了一道这样的题目: 有n个球,用整数1到n编号.还有m个筐子,用整数1到m编号. ...
- 2393Cirno的完美算数教室 容斥
2393: Cirno的完美算数教室 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 652 Solved: 389[Submit][Status][ ...
- 【vijos1943】上学路上
题目戳这里 描述 小雪与小可可吵架了,他们决定以后互相再也不理对方了.尤其是,他们希望以后上学的路上不会再相遇. 我们将他们所在城市的道路网视作无限大的正交网格图,每一个整数点 (x,y) 对应了一个 ...