2011-12-21 6 views
12

È possibile recuperare il nome del test attualmente in esecuzione come in JUnit (utilizzando getName() o rules)?Recupera nome test su TestNG

@Test 
public void fooBar(){ 
    System.out.println(magic()); //should print "fooBar" 
} 

P.S. Non voglio usare uno strumento auto-scritto basato sulle tracce dello stack.

risposta

5

Secondo la documentazione di TestNG al numero: http://testng.org/doc/documentation-main.html è possibile implementare listener che potrebbero essere in grado di aiutarvi con il vostro problema.

Vedere la sezione 5.16 TestNG Listener e in particolare IInvokedMethodListener (javadoc: http://testng.org/javadocs/org/testng/IInvokedMethodListener.html). È possibile connettersi a beforeInvocation per prendere il nome del metodo, tenerlo da qualche parte e quindi utilizzarlo nel test. Potresti, naturalmente, utilizzare i dettagli immediatamente nell'implementazione del listener.

+0

Questa risposta è molto vecchia. La risposta di Dmitry è la più semplice e richiede meno sforzo. – gorbysbm

7

Dichiarare un parametro ITestContext nel metodo e ottenere qualsiasi informazione necessaria.

+0

In realtà non riesco a trovarlo in questo contesto di interfaccia/suite/currentXmlIl nome del test non contiene questa informazione. –

+1

La tua domanda è più una domanda Java che una versione TestNG, e dato che non vuoi usare l'unico modo che conosco per farlo (cammina nella traccia dello stack), non sono sicuro di cos'altro dire ... –

32

ho trovato una soluzione migliore con @BeforeMethod annotazione:

import java.lang.reflect.Method; 

public class Test 
{ 

    @BeforeMethod 
    public void handleTestMethodName(Method method) 
    { 
     String testName = method.getName(); 
     ... 
    } 

    ... 
} 

(sulla base di soluzione from this thread)

10

Quando si utilizza TestNG è possibile utilizzare @BeforeTest annotazione

Prova test set name in TestNG. tag di test del file xml:

<test name="Check name test" > 

e utilizzare questa metod:

@BeforeTest 
public void startTest(final ITestContext testContext) { 
    System.out.println(testContext.getName()); // it prints "Check name test" 
} 
1

È necessario prestare attenzione quando si tiene sui valori passati in ascoltatori come IInvokedMethodListener come implementazione ingenuo (compresi quelli nelle risposte già esistenti) non sarà thread-safe. Poiché TestNG può eseguire test contemporaneamente, è possibile vedere il valore memorizzato da un ascoltatore di un test diverso. Ecco un esempio con due test, testA() e testB():

  1. beforeInvocation(testA) negozi testA
  2. beforeInvocation(testB) negozi testB sovrascrivendo testA
  3. testA() recupera testB (!!)
  4. testB() recupera testB

La classe TestMethodCapture di seguito gestisce correttamente questa condizione di competizione associando l'ascoltatore e il relativo test tramite uno ThreadLocal, assicurando che i test in esecuzione simultanea non si sovrascrivano l'un l'altro.

Ancora meglio, non è limitato al solo recupero il nome del test, contiene un riferimento a entrambe le istanze ITestNGMethod e ITestResult associati alla corrente di prova, in modo da poter anche ispezionare il metodo di class, test groups e parameters.

Si può usare in questo modo:

@Listeners(TestMethodCapture.class) 
public class TestMethodCaptureTest { 
    @Test 
    public void fooBar() { 
    // will print "fooBar" 
    System.out.println(TestMethodCapture.getTestMethod().getMethodName()); 
    } 
} 

Ed ecco la classe stessa:

/** 
* Captures the currently executing test method so it can be accessed by the test, 
* e.g. to retrieve the test method's name. This class is thread-safe. 
* 
* <p>Register this class as a 
* <a href="http://testng.org/doc/documentation-main.html#testng-listeners">TestNG 
* listener</a>, then access the method and result from test code with the static 
* {@link #getTestMethod} and {@link #getTestResult} methods. 
* 
* <p>Annotating a test class with {@code @Listeners(TestMethodCapture.class)} is the 
* suggested way to enable capturing if your test's correctness will depend on this 
* listener being enabled. 
*/ 
public class TestMethodCapture implements IInvokedMethodListener { 
    private static ThreadLocal<ITestNGMethod> currentMethods = new ThreadLocal<>(); 
    private static ThreadLocal<ITestResult> currentResults = new ThreadLocal<>(); 

    @Override 
    public void beforeInvocation(IInvokedMethod method, ITestResult testResult) { 
    currentMethods.set(method.getTestMethod()); 
    currentResults.set(testResult); 
    } 

    @Override 
    public void afterInvocation(IInvokedMethod method, ITestResult testResult) { 
    currentMethods.remove(); 
    currentResults.remove(); 
    } 

    public static ITestNGMethod getTestMethod() { 
    return checkNotNull(currentMethods.get(), 
     "Did you forget to register the %s listener?", TestMethodCapture.class.getName()); 
    } 

    /** 
    * Parameters passed from a data provider are accessible in the test result. 
    */ 
    public static ITestResult getTestResult() { 
    return checkNotNull(currentResults.get(), 
     "Did you forget to register the %s listener?", TestMethodCapture.class.getName()); 
    } 
} 

Se non si utilizza Guava (perché no ??) è possibile aggiungere un metodo checkNotNUll() come questo per rendere questo compile:

private static <T> T checkNotNull(T o, String msg, Object param) { 
    if (o == null) { 
    throw new NullPointerException(String.format(msg, param)); 
    } 
    return o; 
} 
+0

Puoi spiegare il metodo checkNotNull che viene restituito? Dovremmo definire il metodo? Mostra un errore che questo metodo non è definito. – nivasan89

+1

@ nivasan89 scusa ho perso il tuo commento. ['checkNotNull()'] (https://github.com/google/guava/wiki/PreconditionsExplained) proviene da [Guava] (https://github.com/google/guava). Vorrei fortemente incoraggiare l'uso di questa libreria in qualsiasi progetto Java, ma questo metodo è essenzialmente un buon wrapper attorno a "if (foo == null) lancia NullPointerException();" così potresti semplicemente sostituire queste chiamate con un condizionale simile. – dimo414

+1

Questa dovrebbe essere la risposta accettata. – Bass