Все зависит от того, как вы визуализируете свой TestCase ID в системе TCMS.
Если контрольный пример представляет собой тест, управляемый данными, то подход должен быть немного другим.
Если тестовый пример представляет собой обычный тест, то, я думаю, у вас уже есть работающее решение.
1120 Вот один из способов сделать это. Я использую TestNG 7.0.0-beta3
(последняя выпущенная на сегодня версия)
Предположение:
Шаги, которые необходимо выполнить:
@Test
, используя пользовательскую аннотацию, чтобы связать ее с конкретным тестовым набором TCMS. То же самое подробно описано в моем блоге здесь .
Вот пример, который показывает все это в действии:
Пользовательская аннотация выглядит следующим образом:
import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.ElementType.TYPE;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target({METHOD, TYPE})
public @interface Tcms {
String id() default "";
}
Слушатель выглядит следующим образом:
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import org.testng.IInvokedMethod;
import org.testng.IInvokedMethodListener;
import org.testng.ITestResult;
public class TestRailReporter implements IInvokedMethodListener {
private Map resultTracker = new ConcurrentHashMap<>();
@Override
public void beforeInvocation(IInvokedMethod method, ITestResult testResult) {
String key = testResult.getInstanceName() + "." + method.getTestMethod().getMethodName();
resultTracker.putIfAbsent(key, Boolean.TRUE);
}
@Override
public void afterInvocation(IInvokedMethod method, ITestResult testResult) {
Tcms tcms =
method.getTestMethod().getConstructorOrMethod().getMethod().getAnnotation(Tcms.class);
// Only report those tests to TestRail wherein our annotation is found.
if (tcms == null) {
return;
}
if (method.getTestMethod().isDataDriven()) {
// For data driven tests we need a different logic
String key = testResult.getInstanceName() + "." + method.getTestMethod().getMethodName();
if (method.getTestMethod().hasMoreInvocation()) {
Boolean result = resultTracker.get(key);
result = result && (testResult.getStatus() == ITestResult.SUCCESS);
resultTracker.put(key, result);
return;
}
postResultsToTestRail(tcms, resultTracker.get(key));
} else {
postResultsToTestRail(tcms, testResult.getStatus() == ITestResult.SUCCESS);
}
}
private void postResultsToTestRail(Tcms tcms, boolean pass) {
String testCaseId = tcms.id();
// Write logic here that takes care of posting results to the TCMS system
System.err.println("Test case Id [" + testCaseId + "] passed ? " + pass);
}
}
[1127 ] Пример тестового примера:
import org.testng.Assert;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Listeners;
import org.testng.annotations.Test;
@Listeners(TestRailReporter.class)
public class SampleTestCase {
@Test
@Tcms(id = "TESTRAIL-1")
public void testMethod() {
Assert.assertTrue(true);
}
@Test(dataProvider = "dp")
@Tcms(id = "TESTRAIL-2")
public void dataDrivenTestWithSomeFailures(int i) {
if (i % 2 == 0) {
Assert.fail("simulating a failure");
}
}
@Test(dataProvider = "dp")
@Tcms(id = "TESTRAIL-3")
public void dataDrivenTestWithNoFailures(int i) {
Assert.assertTrue(i >= 0);
}
@DataProvider(name = "dp")
public Object[][] getData() {
return new Object[][] {{1}, {2}, {3}};
}
}
Вывод:
Test case Id [TESTRAIL-3] passed ? true
Test case Id [TESTRAIL-2] passed ? false
java.lang.AssertionError: simulating a failure
at org.testng.Assert.fail(Assert.java:97)
at com.rationaleemotions.stackoverflow.qn54224337.SampleTestCase.dataDrivenTestWithSomeFailures(SampleTestCase.java:21)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.testng.internal.MethodInvocationHelper.invokeMethod(MethodInvocationHelper.java:131)
at org.testng.internal.TestInvoker.invokeMethod(TestInvoker.java:570)
at org.testng.internal.TestInvoker.invokeTestMethod(TestInvoker.java:170)
at org.testng.internal.MethodRunner.runInSequence(MethodRunner.java:46)
at org.testng.internal.TestInvoker$MethodInvocationAgent.invoke(TestInvoker.java:790)
at org.testng.internal.TestInvoker.invokeTestMethods(TestInvoker.java:143)
at org.testng.internal.TestMethodWorker.invokeTestMethods(TestMethodWorker.java:146)
at org.testng.internal.TestMethodWorker.run(TestMethodWorker.java:128)
at org.testng.TestRunner.privateRun(TestRunner.java:763)
at org.testng.TestRunner.run(TestRunner.java:594)
at org.testng.SuiteRunner.runTest(SuiteRunner.java:398)
at org.testng.SuiteRunner.runSequentially(SuiteRunner.java:392)
at org.testng.SuiteRunner.privateRun(SuiteRunner.java:355)
at org.testng.SuiteRunner.run(SuiteRunner.java:304)
at org.testng.SuiteRunnerWorker.runSuite(SuiteRunnerWorker.java:53)
at org.testng.SuiteRunnerWorker.run(SuiteRunnerWorker.java:96)
at org.testng.TestNG.runSuitesSequentially(TestNG.java:1146)
at org.testng.TestNG.runSuitesLocally(TestNG.java:1067)
at org.testng.TestNG.runSuites(TestNG.java:997)
at org.testng.TestNG.run(TestNG.java:965)
at org.testng.IDEARemoteTestNG.run(IDEARemoteTestNG.java:73)
at org.testng.RemoteTestNGStarter.main(RemoteTestNGStarter.java:123)
Test case Id [TESTRAIL-1] passed ? true
===============================================
Default Suite
Total tests run: 7, Passes: 6, Failures: 1, Skips: 0
===============================================
Редактировать: Основываясь на комментариях от OP, вот другой способ сделать это. [ 1129]
Подход 2
Используемая аннотация:
import static java.lang.annotation.ElementType.METHOD;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
@Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
@Target({METHOD})
public @interface Tcms {
String id() default "";
}
Аргумент, который будет передан методу тестирования поставщиком данных для управляемых данных тесты будут выглядеть следующим образом:
import java.lang.annotation.Annotation;
public class TestData implements Tcms {
private String tcmsId;
private String data;
public TestData(String tcmsId, String data) {
this.tcmsId = tcmsId;
this.data = data;
}
@Override
public String id() {
return tcmsId;
}
public String getData() {
return data;
}
@Override
public Class extends Annotation> annotationType() {
return Tcms.class;
}
@Override
public String toString() {
return getData();
}
}
Слушатель выглядит следующим образом:
import org.testng.IInvokedMethod;
import org.testng.IInvokedMethodListener;
import org.testng.ITestResult;
public class TestRailReporter2 implements IInvokedMethodListener {
@Override
public void afterInvocation(IInvokedMethod method, ITestResult testResult) {
if (method.getTestMethod().isDataDriven()) {
//Data driven tests need to be handled differently
Object[] parameters = testResult.getParameters();
if (parameters.length != 1) {
//If theres more than one parameter, then dont do anything.
return;
}
Object parameter = parameters[0];
if (!(parameter instanceof Tcms)) {
//If the parameter doesnt implement our interface dont do anything
return;
}
postResultsToTestRail(
(Tcms) parameter, testResult.getStatus() == ITestResult.SUCCESS, parameter.toString());
} else {
Tcms tcms =
method.getTestMethod().getConstructorOrMethod().getMethod().getAnnotation(Tcms.class);
if (tcms == null) {
return;
}
postResultsToTestRail(tcms, testResult.getStatus() == ITestResult.SUCCESS);
}
}
private void postResultsToTestRail(Tcms tcms, boolean pass) {
String testCaseId = tcms.id();
// Write logic here that takes care of posting results to the TCMS system
System.err.println("Test case Id [" + testCaseId + "] passed ? " + pass);
}
private void postResultsToTestRail(Tcms tcms, boolean pass, String param) {
String id = tcms.id();
// Write logic here that takes care of posting results to the TCMS system
System.err.println("Test case Id [" + id + "] with parameter [" + param + "] passed ? " + pass);
}
}
Класс теста выглядит следующим образом:
import org.testng.Assert;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Listeners;
import org.testng.annotations.Test;
@Listeners(TestRailReporter2.class)
public class AnotherSampleTestCase {
@Test
@Tcms(id = "TESTRAIL-1")
public void simpleTestMethod() {
Assert.assertTrue(true);
}
@Test(dataProvider = "dp")
public void dataDrivenTestMethod(TestData data) {
Assert.assertFalse(data.getData().trim().isEmpty());
}
@DataProvider(name = "dp")
public Object[][] getData() {
return new Object[][] {
{new TestData("TESTRAIL-2", "Jack")},
{new TestData("TESTRAIL-3", "")},
{new TestData("TESTRAIL-4", "Daniels")}
};
}
}
Вот результат выполнения :
Test case Id [TESTRAIL-2] with parameter [Jack] passed ? true
Test case Id [TESTRAIL-3] with parameter [] passed ? false
java.lang.AssertionError: did not expect to find [false] but found [true]
at org.testng.Assert.fail(Assert.java:97)
at org.testng.Assert.failNotEquals(Assert.java:969)
at org.testng.Assert.assertFalse(Assert.java:65)
at org.testng.Assert.assertFalse(Assert.java:75)
at com.rationaleemotions.stackoverflow.qn54224337.AnotherSampleTestCase.dataDrivenTestMethod(AnotherSampleTestCase.java:19)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.testng.internal.MethodInvocationHelper.invokeMethod(MethodInvocationHelper.java:131)
at org.testng.internal.TestInvoker.invokeMethod(TestInvoker.java:570)
at org.testng.internal.TestInvoker.invokeTestMethod(TestInvoker.java:170)
at org.testng.internal.MethodRunner.runInSequence(MethodRunner.java:46)
at org.testng.internal.TestInvoker$MethodInvocationAgent.invoke(TestInvoker.java:790)
at org.testng.internal.TestInvoker.invokeTestMethods(TestInvoker.java:143)
at org.testng.internal.TestMethodWorker.invokeTestMethods(TestMethodWorker.java:146)
at org.testng.internal.TestMethodWorker.run(TestMethodWorker.java:128)
at org.testng.TestRunner.privateRun(TestRunner.java:763)
at org.testng.TestRunner.run(TestRunner.java:594)
at org.testng.SuiteRunner.runTest(SuiteRunner.java:398)
at org.testng.SuiteRunner.runSequentially(SuiteRunner.java:392)
at org.testng.SuiteRunner.privateRun(SuiteRunner.java:355)
at org.testng.SuiteRunner.run(SuiteRunner.java:304)
at org.testng.SuiteRunnerWorker.runSuite(SuiteRunnerWorker.java:53)
at org.testng.SuiteRunnerWorker.run(SuiteRunnerWorker.java:96)
at org.testng.TestNG.runSuitesSequentially(TestNG.java:1146)
at org.testng.TestNG.runSuitesLocally(TestNG.java:1067)
at org.testng.TestNG.runSuites(TestNG.java:997)
at org.testng.TestNG.run(TestNG.java:965)
at org.testng.IDEARemoteTestNG.run(IDEARemoteTestNG.java:73)
at org.testng.RemoteTestNGStarter.main(RemoteTestNGStarter.java:123)
Test case Id [TESTRAIL-4] with parameter [Daniels] passed ? true
Test case Id [TESTRAIL-1] passed ? true
===============================================
Default Suite
Total tests run: 4, Passes: 3, Failures: 1, Skips: 0
===============================================
Process finished with exit code 0
Как уже было сказано, нам нужна точность.
Если вы используете debian, вы можете включить этот режим следующим образом.
//List all available mods
cd /etc/apache2/mods-avaliable
//Enable module - if you need to enable another module, just replace "deflate"
a2enmod deflate
//restart apache
service apache2 restart
//Check that module is effectively enabled
cd /etc/apache2/mods-enabled
ls -al
Это все для среды linux debian.
Вы можете проверить mod_deflate с помощью этого сайта:
Мой сайт получил хороший небольшой отчет, который показывает, что я экономлю 81% своей пропускной способности!
Убедитесь, что следующая строка присутствует (и не закомментирована) в вашей конфигурации apache (httpd.conf):
LoadModule deflate_module modules/mod_deflate.so
Закомментированная версия выглядит так:
# LoadModule deflate_module modules/mod_deflate.so
Затем, чтобы он сжимал файлы по мере их обслуживания, вам понадобится строка в вашем .htaccess, например:
AddOutputFilterByType DEFLATE text/html text/css text/plain text/xml application/x-javascript
Я согласен с комментариями « в каком контексте », но для охвата всех основ (в дополнение к другим ответам) вы можете также иметь возможность запускать;
a2enmod deflate
Это должно работать на многих серверах, на которых работает Apache (особенно на базе Debian), в случае, если модуль уже включен, команда сообщит вам, если модуль не включен, команда включит Это. Он также может сказать вам, что модуля не существует, и в этом случае вам нужно будет его установить.
Возможно, уже слишком поздно, но поехали.
mod_deflate включен по умолчанию. Чтобы быть уверенным, попробуйте
debian / ubuntu:
apache2ctl -t -D DUMP_MODULES
CentOS:
httpd -t -D DUMP_MODULES
и посмотрите, есть ли deflate_module.