Как запустить одиночный тест с помощью UnitTest ++?

Как запустить одиночный тест с помощью UnitTest ++?

Я запускаю UnitTest ++ из коробки как есть. Моя функция main выглядит так:

int main()
{
   printf("diamond test v0.1 %s\n\n",TIMESTAMP);
   diamond::startup();
   UnitTest::RunAllTests();
   diamond::shutdown();
   printf("press any key to continue...");
   getc(stdin);
}

Для отладки я хотел бы написать что-то вроде UnitTest::RunSingleTests("MyNewUnitTest"); вместо UnitTest::RunAllTests();. Обеспечивает ли UnitTest ++ такую ​​функцию, и если да, то какой синтаксис?

Ответ 1

попробуйте сделать это как ваш основной() для unittest (я фактически поместил это в файл и добавил, что в библиотеку unittest, так что при обращении к библиотеке исполняемый файл автоматически использует этот main(). очень удобно.)

int main( int argc, char** argv )
{
  if( argc > 1 )
  {
      //if first arg is "suite", we search for suite names instead of test names
    const bool bSuite = strcmp( "suite", argv[ 1 ] ) == 0;

      //walk list of all tests, add those with a name that
      //matches one of the arguments  to a new TestList
    const TestList& allTests( Test::GetTestList() );
    TestList selectedTests;
    Test* p = allTests.GetHead();
    while( p )
    {
      for( int i = 1 ; i < argc ; ++i )
        if( strcmp( bSuite ? p->m_details.suiteName
                           : p->m_details.testName, argv[ i ] ) == 0 )
          selectedTests.Add( p );
      p = p->next;
    }

      //run selected test(s) only
    TestReporterStdout reporter;
    TestRunner runner( reporter );
    return runner.RunTestsIf( selectedTests, 0, True(), 0 );
  }
  else
  {
    return RunAllTests();
  }
}

вызывать с аргументами для запуска одного теста:

> myexe MyTestName

или отдельный пакет

> myexe suite MySuite

Ответ 2

Это почти правильно. "Тест" на самом деле обозначается как node в связанном списке, поэтому, когда вы добавляете его в новый список, вы должны исправить указатель, чтобы не включать больше тестов, чем вы планировали.

Итак, вам нужно заменить

  p = p->next;

с

  Test* q = p;
  p = p->next;
  q->next = NULL;

Джеффри

Ответ 3

RunTestsIf может запускать только один пакет, если вы укажете ему имя.

class MyTrue
{
    public:
        MyTrue(const std::string & suiteName, const std::string & testName)
                : suite(suiteName), test(testName) {}

        bool operator()(const UnitTest::Test* const testCase) const
        {
            return suite.compare(testCase->m_details.suiteName) == 0 && 
                         test.compare(testCase->m_details.testName) == 0;
        }
    private:
        std::string suite;
        std::string test;
};

int main (...) {
    bool isSuite = false;
    std::string suiteName = "suite01";
    std::string testName  = "test01";

    UnitTest::TestReporterStdout reporter;
    UnitTest::TestRunner runner(reporter);
    if (isSuite) {
        runner.RunTestsIf(UnitTest::Test::GetTestList(),
            NULL, MyTrue(suiteName, testName), 0);
    } else {
        runner.RunTestsIf(UnitTest::Test::GetTestList(),
            NULL, UnitTest::True(), 0);
    }
}

Ответ 4

Вы можете сделать это, используя параметр predicate RunTestsIf:

TestReporterStdout reporter;
TestRunner runner(reporter);
return runner.RunTestsIf(Test::GetTestList(), "MySuite",
    [](Test* t) { 
        return strcmp(t->m_details.testName, "MyTest") == 0; 
    }, 0);

Если у вас нет наборов или вы хотите найти их все, вы можете заменить "MySuite" на NULL.

Ответ 5

Ответ, заданный @stijn, имеет ошибку в обработке списка тестов, и поэтому он обычно запускает дополнительные тесты, которые вы не запрашивали.

В этом примере используется предикатный функтор, а также используется встроенное имя набора имен, предоставляемое RunTestsIf. Это правильно и намного проще.

#include "UnitTest++.h"
#include "TestReporterStdout.h"
#include <string.h>
using namespace UnitTest;

/// Predicate that is true for tests with matching name,
/// or all tests if no names were given.
class Predicate
{
public:

Predicate(const char **tests, int nTests)
    : _tests(tests),
      _nTests(nTests)
{
}

bool operator()(Test *test) const
{
    bool match = (_nTests == 0);
    for (int i = 0; !match && i < _nTests; ++i) {
        if (!strcmp(test->m_details.testName, _tests[i])) {
            match = true;
        }
    }
    return match;
}

private:
    const char **_tests;
    int _nTests;
};

int main(int argc, const char** argv)
{
    const char *suiteName = 0;
    int arg = 1;

    // Optional "suite" arg must be followed by a suite name.
    if (argc >=3 && strcmp("suite", argv[arg]) == 0) {
        suiteName = argv[++arg];
        ++arg;
    } 

    // Construct predicate that matches any tests given on command line.
    Predicate pred(argv + arg, argc - arg);

    // Run tests that match any given suite and tests.
    TestReporterStdout reporter;
    TestRunner runner(reporter);
    return runner.RunTestsIf(Test::GetTestList(), suiteName, pred, 0);
}

Ответ 6

Решение в принятом ответе не работает для меня. Когда первый тест пакета загружается в p, он просто не перейдет к следующему тесту (не знаю точно, почему).

Я использую Xcode и UnitTest ++ v1.4

#include "UnitTest++.h"
#include "TestReporterStdout.h"

#define SUITE_NAME "ActionFeedback"

using namespace UnitTest;

int main( int argc, char** argv )
{
#ifdef SUITE_NAME
    TestReporterStdout reporter;
    TestRunner runner( reporter );
    return runner.RunTestsIf( Test::GetTestList() ,  SUITE_NAME , True(), 0 );
#else
    return RunAllTests();
#endif

}