Skip to content
This repository has been archived by the owner on Oct 28, 2021. It is now read-only.

Commit

Permalink
testeth print test suite suggestions
Browse files Browse the repository at this point in the history
  • Loading branch information
winsvega committed May 9, 2019
1 parent e80e197 commit ff01d20
Show file tree
Hide file tree
Showing 5 changed files with 86 additions and 0 deletions.
10 changes: 10 additions & 0 deletions test/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ list(REMOVE_ITEM sources ${unittest_sources})
# search for test names and create ctest tests
set(excludeSuites jsonrpc \"customTestSuite\" BlockQueueSuite)
set(allSuites jsonrpc)
set(allTests "")
foreach(file ${sources})
file(STRINGS ${file} test_list_raw REGEX "BOOST_.*TEST_(SUITE|CASE|SUITE_END)")
set(TestSuite "DEFAULT")
Expand Down Expand Up @@ -81,6 +82,7 @@ foreach(file ${sources})
set(TestEthArgs -t ${TestSuitePathFixed} -- ${TESTETH_ARGS})
add_test(NAME ${TestSuitePathFixed} WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/test COMMAND testeth ${TestEthArgs})
set_tests_properties(${TestSuitePathFixed} PROPERTIES TIMEOUT ${timeout})
set(allTests "${allTests} \"${TestSuitePathFixed}\",\n")
endif()
endif()
elseif(test MATCHES "^CASE .*")
Expand All @@ -92,6 +94,11 @@ foreach(file ${sources})
set(TestEthArgs -t ${TestSuitePathFixed}/${TestCase} -- ${TESTETH_ARGS})
add_test(NAME ${TestSuitePathFixed}/${TestCase} WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/test COMMAND testeth ${TestEthArgs})
set_tests_properties(${TestSuitePathFixed}/${TestCase} PROPERTIES TIMEOUT ${timeout})
set(allTests "${allTests} \"${TestSuitePathFixed}/${TestCase}\",\n")
if (NOT ";${allSuites};" MATCHES ";${TestSuitePathFixed};")
list(APPEND allSuites ${TestSuitePathFixed})
set(allTests "${allTests} \"${TestSuitePathFixed}\",\n")
endif()
endif()
endif()
elseif (";${test_raw};" MATCHES "BOOST_AUTO_TEST_SUITE_END()")
Expand All @@ -102,6 +109,9 @@ foreach(file ${sources})
endforeach(test_raw)
endforeach(file)

#Save allTests into a variable in include file
configure_file(tools/libtesteth/AllTestNames.h.in ${PROJECT_BINARY_DIR}/AllTestNames.h)

hunter_add_package(yaml-cpp)
find_package(yaml-cpp CONFIG REQUIRED)

Expand Down
6 changes: 6 additions & 0 deletions test/tools/libtesteth/AllTestNames.h.in
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
#pragma once

char const* c_allTestNames[] =
{
@allTests@
};
30 changes: 30 additions & 0 deletions test/tools/libtesteth/TestHelper.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -637,6 +637,36 @@ string prepareLLLCVersionString()
return "Error getting LLLC Version";
}

// A simple C++ implementation of the Levenshtein distance algorithm to measure the amount of
// difference between two strings. https://gist.github.com/TheRayTracer/2644387
size_t levenshteinDistance(char const* _s, size_t _n, char const* _t, size_t _m)
{
++_n;
++_m;
size_t* d = new size_t[_n * _m];

memset(d, 0, sizeof(size_t) * _n * _m);
for (size_t i = 1, im = 0; i < _m; ++i, ++im)
{
for (size_t j = 1, jn = 0; j < _n; ++j, ++jn)
{
if (_s[jn] == _t[im])
d[(i * _n) + j] = d[((i - 1) * _n) + (j - 1)];
else
{
d[(i * _n) + j] = min(d[(i - 1) * _n + j] + 1, /* A deletion. */
min(d[i * _n + (j - 1)] + 1, /* An insertion. */
d[(i - 1) * _n + (j - 1)] + 1)); /* A substitution. */
}
}
}

size_t r = d[_n * _m - 1];
delete[] d;
return r;
}


void copyFile(fs::path const& _source, fs::path const& _destination)
{
fs::ifstream src(_source, ios::binary);
Expand Down
1 change: 1 addition & 0 deletions test/tools/libtesteth/TestHelper.h
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,7 @@ bytes importByteArray(std::string const& _str);
void requireJsonFields(json_spirit::mObject const& _o, std::string const& _section,
std::map<std::string, json_spirit::Value_type> const& _validationMap);
void checkHexHasEvenLength(std::string const&);
size_t levenshteinDistance(char const* _s, size_t _n, char const* _t, size_t _m);
void copyFile(boost::filesystem::path const& _source, boost::filesystem::path const& _destination);
eth::LogEntries importLog(json_spirit::mArray const& _o);
std::string exportLog(eth::LogEntries const& _logs);
Expand Down
39 changes: 39 additions & 0 deletions test/tools/libtesteth/boostTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -11,12 +11,14 @@
#define BOOST_TEST_MODULE EthereumTests
#define BOOST_TEST_NO_MAIN

#include <AllTestNames.h>
#include <test/tools/jsontests/BlockChainTests.h>
#include <test/tools/jsontests/StateTests.h>
#include <test/tools/jsontests/TransactionTests.h>
#include <test/tools/jsontests/vm.h>
#include <test/tools/libtesteth/TestHelper.h>
#include <boost/test/included/unit_test.hpp>
#include <boost/tokenizer.hpp>
#include <clocale>
#include <cstdlib>
#include <iostream>
Expand All @@ -27,6 +29,7 @@ using namespace boost::unit_test;
static std::ostringstream strCout;
std::streambuf* oldCoutStreamBuf;
std::streambuf* oldCerrStreamBuf;
void printTestSuiteSuggestions(string const& _sMinusTArg);

void customTestSuite()
{
Expand Down Expand Up @@ -133,10 +136,46 @@ int main(int argc, const char* argv[])
framework::master_test_suite().add(ts1);
}

string sMinusTArg;
// unit_test_main delete this option from _argv
for (int i = 0; i < argc; i++) // find -t boost arg
{
std::string const arg = std::string{argv[i]};
if (arg == "-t" && i + 1 < argc)
{
sMinusTArg = std::string{argv[i + 1]};
break;
}
}

std::cout << "Running tests using path: " << test::getTestPath() << std::endl;
int result = 0;
auto fakeInit = [](int, char* []) -> boost::unit_test::test_suite* { return nullptr; };
result = unit_test_main(fakeInit, argc, const_cast<char**>(argv));

// Print suggestions of a test case
if (result == boost::exit_exception_failure) // test suite not found
printTestSuiteSuggestions(sMinusTArg);
dev::test::TestOutputHelper::get().printTestExecStats();
return result;
}

void printTestSuiteSuggestions(string const& _sMinusTArg)
{
size_t allTestsElementIndex = 0;
// <index in availableTests, compared distance>
typedef std::pair<size_t, size_t> NameDistance;
// Use `vector` here because `set` does not work with sort
std::vector<NameDistance> distanceMap;
for (auto& it : c_allTestNames)
{
int const dist =
test::levenshteinDistance(_sMinusTArg.c_str(), _sMinusTArg.size(), it, strlen(it));
distanceMap.emplace_back(allTestsElementIndex++, dist);
}
std::sort(distanceMap.begin(), distanceMap.end(),
[](NameDistance const& _a, NameDistance const& _b) { return _a.second < _b.second; });
std::cerr << "Did you mean: \n";
for (size_t i = 0; i < 3 && i < distanceMap.size(); i++)
std::cerr << "-t " << c_allTestNames[distanceMap[i].first] << "\n";
}

0 comments on commit ff01d20

Please sign in to comment.