工作中接到一個任務,調研是否可以實作rpc服務自動diff功能。本篇是其中的一個子功能,測試類的testng.xml自動實作。
實作原理:
1.執行個體化XmlSuite類
2.掃描目錄下所有的classes檔案,然後周遊class的所有方法是否包含@Test注解
3.步驟2中的必要資訊放到XmlSuite中
4.儲存XmlSuite到指定位置
主要方法:
/**
* 生成testng.xml
* @return
*/
public Boolean createTestNGXML() {
String prefix = "Test";
String classesDir = "D:/opt/cases";
XmlSuite suite = new XmlSuite();
XmlTest xmlTest = new XmlTest(suite);
List<XmlClass> xmlClasses = new ArrayList<>();
// testng.xml全名
suite.setFileName(prefix + "_testng.xml");
// 不設定這個suite.toXml()會報NPT
suite.setJunit(false);
suite.setName(prefix);
xmlTest.setName(prefix);
try {
// 建立classloader執行個體
URLClassLoader loader = URLClassLoader.newInstance(new URL[] {}, getClass().getClassLoader());
// 測試用例class所在目錄加到classloader的classpath中
TestNGUtil.addPath(loader, new File(classesDir).toURI().toURL());
// 循環周遊所有的class檔案
Collection<File> classFiles = FileUtils.listFiles(new File(classesDir), new WildcardFileFilter("*.class"),
TrueFileFilter.INSTANCE);
for (File f : classFiles) {
// 包含$說明為内部類
if (f.getName().contains("$")) {
continue;
}
XmlClass xmlClass = new XmlClass();
List<XmlInclude> includes = new ArrayList<>();
// 隻要class名稱不要字尾
String clazzName = f.getName().replace(".class", "");
xmlClass.setName(clazzName);
Class<?> clazz = Class.forName(clazzName, false, loader);
// 周遊所有方法,找到包含@Test标注的方法
for (Method m : clazz.getMethods()) {
if (hasTestAnnotation(m)) {
XmlInclude xmlInclude = new XmlInclude(m.getName());
includes.add(xmlInclude);
}
}
xmlClass.setIncludedMethods(includes);
xmlClasses.add(xmlClass);
}
xmlTest.setXmlClasses(xmlClasses);
// 儲存到檔案系統中
FileUtils.write(new File(classesDir + "/" + suite.getFileName()), suite.toXml(), Charset.defaultCharset());
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
* 判斷目前方法是否包含@Test标注
*
* @param m
* @return
*/
private Boolean hasTestAnnotation(Method m) {
List<Annotation> annotations = Arrays.asList(m.getAnnotations());
for (Annotation annotation : annotations) {
if (annotation.annotationType().getName().equals(Test.class.getName())) {
return Boolean.TRUE;
}
}
return Boolean.FALSE;
}
額外的類TestNgUtil.java
/**
* addURL為protected方法,需要借助于reflect包才能實作動态添加class到classpath中
*
* @param urlClassLoader
* @param url
* @throws Exception
*/
public static void addPath(ClassLoader urlClassLoader, URL url) throws Exception {
Class<URLClassLoader> urlClass = URLClassLoader.class;
Method method = urlClass.getDeclaredMethod("addURL", new Class[] { URL.class });
method.setAccessible(true);
method.invoke(urlClassLoader, new Object[] { url });
}
import資訊
import java.io.File;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import org.apache.commons.io.FileUtils;
import org.apache.commons.io.filefilter.TrueFileFilter;
import org.apache.commons.io.filefilter.WildcardFileFilter;
import org.testng.annotations.Test;
import org.testng.xml.XmlClass;
import org.testng.xml.XmlInclude;
import org.testng.xml.XmlSuite;
import org.testng.xml.XmlTest;
import com.aaa.qa.utils.TestNGUtil;