2011-01-07 20:57:32 +01:00
|
|
|
/*
|
2012-08-14 20:30:30 +02:00
|
|
|
* Created by Phil on 14/08/2012.
|
|
|
|
* Copyright 2012 Two Blue Cubes Ltd. All rights reserved.
|
2011-01-07 20:57:32 +01:00
|
|
|
*
|
|
|
|
* Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
* file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
*/
|
|
|
|
|
2012-08-14 20:30:30 +02:00
|
|
|
#include "catch_test_case_info.h"
|
2017-08-01 18:46:33 +02:00
|
|
|
#include "catch_enforce.h"
|
2017-09-07 12:24:33 +02:00
|
|
|
#include "catch_test_spec.h"
|
2011-01-07 20:57:32 +01:00
|
|
|
#include "catch_interfaces_testcase.h"
|
2017-07-25 21:57:35 +02:00
|
|
|
#include "catch_string_manip.h"
|
2011-01-07 20:57:32 +01:00
|
|
|
|
2017-02-12 12:17:07 +01:00
|
|
|
#include <cctype>
|
2017-06-04 21:39:27 +02:00
|
|
|
#include <exception>
|
2017-07-27 22:31:27 +02:00
|
|
|
#include <algorithm>
|
2017-11-07 19:01:10 +01:00
|
|
|
#include <sstream>
|
2017-02-12 12:17:07 +01:00
|
|
|
|
2012-05-16 00:58:23 +02:00
|
|
|
namespace Catch {
|
|
|
|
|
2018-06-12 15:09:30 +02:00
|
|
|
namespace {
|
|
|
|
TestCaseInfo::SpecialProperties parseSpecialTag( std::string const& tag ) {
|
|
|
|
if( startsWith( tag, '.' ) ||
|
|
|
|
tag == "!hide" )
|
|
|
|
return TestCaseInfo::IsHidden;
|
|
|
|
else if( tag == "!throws" )
|
|
|
|
return TestCaseInfo::Throws;
|
|
|
|
else if( tag == "!shouldfail" )
|
|
|
|
return TestCaseInfo::ShouldFail;
|
|
|
|
else if( tag == "!mayfail" )
|
|
|
|
return TestCaseInfo::MayFail;
|
|
|
|
else if( tag == "!nonportable" )
|
|
|
|
return TestCaseInfo::NonPortable;
|
|
|
|
else if( tag == "!benchmark" )
|
|
|
|
return static_cast<TestCaseInfo::SpecialProperties>( TestCaseInfo::Benchmark | TestCaseInfo::IsHidden );
|
|
|
|
else
|
|
|
|
return TestCaseInfo::None;
|
|
|
|
}
|
|
|
|
bool isReservedTag( std::string const& tag ) {
|
|
|
|
return parseSpecialTag( tag ) == TestCaseInfo::None && tag.size() > 0 && !std::isalnum( static_cast<unsigned char>(tag[0]) );
|
|
|
|
}
|
|
|
|
void enforceNotReservedTag( std::string const& tag, SourceLineInfo const& _lineInfo ) {
|
|
|
|
CATCH_ENFORCE( !isReservedTag(tag),
|
|
|
|
"Tag name: [" << tag << "] is not allowed.\n"
|
2019-04-08 23:30:28 +02:00
|
|
|
<< "Tag names starting with non alphanumeric characters are reserved\n"
|
2018-06-12 15:09:30 +02:00
|
|
|
<< _lineInfo );
|
|
|
|
}
|
2019-11-04 13:45:39 +01:00
|
|
|
|
|
|
|
std::string makeDefaultName() {
|
|
|
|
static size_t counter = 0;
|
|
|
|
return "Anonymous test case " + std::to_string(++counter);
|
|
|
|
}
|
2014-05-16 19:52:55 +02:00
|
|
|
}
|
2014-04-15 19:44:37 +02:00
|
|
|
|
2017-07-12 19:01:54 +02:00
|
|
|
TestCase makeTestCase( ITestInvoker* _testCase,
|
2013-04-23 19:58:56 +02:00
|
|
|
std::string const& _className,
|
2018-03-02 16:22:18 +01:00
|
|
|
NameAndTags const& nameAndTags,
|
2013-04-23 19:58:56 +02:00
|
|
|
SourceLineInfo const& _lineInfo )
|
2012-09-15 18:53:27 +02:00
|
|
|
{
|
2017-08-27 16:45:53 +02:00
|
|
|
bool isHidden = false;
|
2014-05-16 19:52:55 +02:00
|
|
|
|
|
|
|
// Parse out tags
|
2017-07-27 22:31:27 +02:00
|
|
|
std::vector<std::string> tags;
|
2019-06-22 20:11:14 +02:00
|
|
|
std::string tag;
|
2014-05-16 19:52:55 +02:00
|
|
|
bool inTag = false;
|
2019-09-08 14:49:40 +02:00
|
|
|
for (char c : nameAndTags.tags) {
|
2014-05-16 19:52:55 +02:00
|
|
|
if( !inTag ) {
|
2019-06-22 20:11:14 +02:00
|
|
|
if (c == '[') {
|
2014-05-16 19:52:55 +02:00
|
|
|
inTag = true;
|
2019-06-22 20:11:14 +02:00
|
|
|
}
|
|
|
|
} else {
|
2014-05-16 19:52:55 +02:00
|
|
|
if( c == ']' ) {
|
2014-12-15 08:25:34 +01:00
|
|
|
TestCaseInfo::SpecialProperties prop = parseSpecialTag( tag );
|
2017-08-05 23:24:17 +02:00
|
|
|
if( ( prop & TestCaseInfo::IsHidden ) != 0 )
|
2014-05-19 18:50:58 +02:00
|
|
|
isHidden = true;
|
2014-12-15 08:25:34 +01:00
|
|
|
else if( prop == TestCaseInfo::None )
|
|
|
|
enforceNotReservedTag( tag, _lineInfo );
|
|
|
|
|
2019-03-29 10:48:56 +01:00
|
|
|
// Merged hide tags like `[.approvals]` should be added as
|
|
|
|
// `[.][approvals]`. The `[.]` is added at later point, so
|
|
|
|
// we only strip the prefix
|
|
|
|
if (startsWith(tag, '.') && tag.size() > 1) {
|
|
|
|
tag.erase(0, 1);
|
|
|
|
}
|
2017-07-27 22:31:27 +02:00
|
|
|
tags.push_back( tag );
|
2014-05-16 19:52:55 +02:00
|
|
|
tag.clear();
|
2014-12-15 08:25:34 +01:00
|
|
|
inTag = false;
|
2014-04-15 19:44:37 +02:00
|
|
|
}
|
2014-05-16 19:52:55 +02:00
|
|
|
else
|
|
|
|
tag += c;
|
2014-04-15 19:44:37 +02:00
|
|
|
}
|
2013-12-04 08:58:39 +01:00
|
|
|
}
|
2014-05-20 19:11:23 +02:00
|
|
|
if( isHidden ) {
|
2017-07-27 22:31:27 +02:00
|
|
|
tags.push_back( "." );
|
2014-05-20 19:11:23 +02:00
|
|
|
}
|
2015-11-04 19:01:28 +01:00
|
|
|
|
2019-06-22 20:11:14 +02:00
|
|
|
TestCaseInfo info( static_cast<std::string>(nameAndTags.name), _className, tags, _lineInfo );
|
2018-03-02 16:22:18 +01:00
|
|
|
return TestCase( _testCase, std::move(info) );
|
2012-09-15 18:53:27 +02:00
|
|
|
}
|
2011-01-07 20:57:32 +01:00
|
|
|
|
2017-07-27 22:31:27 +02:00
|
|
|
void setTags( TestCaseInfo& testCaseInfo, std::vector<std::string> tags ) {
|
|
|
|
std::sort(begin(tags), end(tags));
|
|
|
|
tags.erase(std::unique(begin(tags), end(tags)), end(tags));
|
2015-07-02 09:20:18 +02:00
|
|
|
testCaseInfo.lcaseTags.clear();
|
2015-11-04 19:01:28 +01:00
|
|
|
|
2017-04-25 12:06:52 +02:00
|
|
|
for( auto const& tag : tags ) {
|
|
|
|
std::string lcaseTag = toLower( tag );
|
2015-07-02 09:20:18 +02:00
|
|
|
testCaseInfo.properties = static_cast<TestCaseInfo::SpecialProperties>( testCaseInfo.properties | parseSpecialTag( lcaseTag ) );
|
2017-07-27 22:31:27 +02:00
|
|
|
testCaseInfo.lcaseTags.push_back( lcaseTag );
|
2015-07-02 09:20:18 +02:00
|
|
|
}
|
2017-07-27 22:31:27 +02:00
|
|
|
testCaseInfo.tags = std::move(tags);
|
2015-07-02 09:20:18 +02:00
|
|
|
}
|
2015-11-04 19:01:28 +01:00
|
|
|
|
2013-04-23 19:58:56 +02:00
|
|
|
TestCaseInfo::TestCaseInfo( std::string const& _name,
|
|
|
|
std::string const& _className,
|
2017-07-27 22:31:27 +02:00
|
|
|
std::vector<std::string> const& _tags,
|
2013-04-23 19:58:56 +02:00
|
|
|
SourceLineInfo const& _lineInfo )
|
2019-11-04 13:45:39 +01:00
|
|
|
: name( _name.empty() ? makeDefaultName() : _name ),
|
2012-11-25 12:19:55 +01:00
|
|
|
className( _className ),
|
2012-12-05 09:40:53 +01:00
|
|
|
lineInfo( _lineInfo ),
|
2014-07-03 09:09:57 +02:00
|
|
|
properties( None )
|
2013-03-28 23:13:31 +01:00
|
|
|
{
|
2015-07-02 09:20:18 +02:00
|
|
|
setTags( *this, _tags );
|
2013-03-28 23:13:31 +01:00
|
|
|
}
|
2012-08-14 09:38:22 +02:00
|
|
|
|
2014-07-03 09:09:57 +02:00
|
|
|
bool TestCaseInfo::isHidden() const {
|
|
|
|
return ( properties & IsHidden ) != 0;
|
|
|
|
}
|
|
|
|
bool TestCaseInfo::throws() const {
|
|
|
|
return ( properties & Throws ) != 0;
|
|
|
|
}
|
|
|
|
bool TestCaseInfo::okToFail() const {
|
|
|
|
return ( properties & (ShouldFail | MayFail ) ) != 0;
|
|
|
|
}
|
|
|
|
bool TestCaseInfo::expectedToFail() const {
|
|
|
|
return ( properties & (ShouldFail ) ) != 0;
|
|
|
|
}
|
|
|
|
|
2017-07-27 22:31:27 +02:00
|
|
|
std::string TestCaseInfo::tagsAsString() const {
|
|
|
|
std::string ret;
|
|
|
|
// '[' and ']' per tag
|
2017-09-18 18:13:17 +02:00
|
|
|
std::size_t full_size = 2 * tags.size();
|
2017-07-27 22:31:27 +02:00
|
|
|
for (const auto& tag : tags) {
|
|
|
|
full_size += tag.size();
|
|
|
|
}
|
|
|
|
ret.reserve(full_size);
|
|
|
|
for (const auto& tag : tags) {
|
|
|
|
ret.push_back('[');
|
|
|
|
ret.append(tag);
|
|
|
|
ret.push_back(']');
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-07-03 09:09:57 +02:00
|
|
|
|
2018-03-02 16:22:18 +01:00
|
|
|
TestCase::TestCase( ITestInvoker* testCase, TestCaseInfo&& info ) : TestCaseInfo( std::move(info) ), test( testCase ) {}
|
2012-11-25 12:19:55 +01:00
|
|
|
|
2012-08-23 09:38:27 +02:00
|
|
|
|
2012-11-25 12:19:55 +01:00
|
|
|
void TestCase::invoke() const {
|
|
|
|
test->invoke();
|
2012-08-14 20:30:30 +02:00
|
|
|
}
|
|
|
|
|
2013-04-23 19:58:56 +02:00
|
|
|
bool TestCase::operator == ( TestCase const& other ) const {
|
2012-11-25 12:19:55 +01:00
|
|
|
return test.get() == other.test.get() &&
|
|
|
|
name == other.name &&
|
|
|
|
className == other.className;
|
2012-08-14 20:30:30 +02:00
|
|
|
}
|
|
|
|
|
2013-04-23 19:58:56 +02:00
|
|
|
bool TestCase::operator < ( TestCase const& other ) const {
|
2012-11-25 12:19:55 +01:00
|
|
|
return name < other.name;
|
2012-08-14 20:30:30 +02:00
|
|
|
}
|
2012-09-21 08:48:03 +02:00
|
|
|
|
2013-04-23 19:58:56 +02:00
|
|
|
TestCaseInfo const& TestCase::getTestCaseInfo() const
|
2012-11-25 12:19:55 +01:00
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2012-09-21 08:48:03 +02:00
|
|
|
} // end namespace Catch
|