2022-01-29 00:03:43 +01:00
|
|
|
|
|
|
|
|
// Copyright Catch2 Authors
|
|
|
|
|
// Distributed under the Boost Software License, Version 1.0.
|
2022-10-28 11:22:53 +02:00
|
|
|
// (See accompanying file LICENSE.txt or copy at
|
2022-01-29 00:03:43 +01:00
|
|
|
// https://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
|
|
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
|
|
2020-01-20 23:24:04 +01:00
|
|
|
#include <catch2/catch_test_macros.hpp>
|
2026-04-08 10:26:57 +02:00
|
|
|
#include <catch2/internal/catch_enum_info.hpp>
|
|
|
|
|
#include <catch2/matchers/catch_matchers_vector.hpp>
|
2014-09-01 17:24:10 +01:00
|
|
|
|
2026-04-08 10:26:57 +02:00
|
|
|
namespace Bikeshed {
|
|
|
|
|
enum class Colours { Red, Green, Blue };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Important!: This macro must appear at top level scope - not inside a namespace
|
|
|
|
|
// You can fully qualify the names, or use a using if you prefer
|
|
|
|
|
CATCH_REGISTER_ENUM( Bikeshed::Colours,
|
|
|
|
|
Bikeshed::Colours::Red,
|
|
|
|
|
Bikeshed::Colours::Green,
|
|
|
|
|
Bikeshed::Colours::Blue )
|
|
|
|
|
|
|
|
|
|
TEST_CASE( "Enums in namespaces can quickly have stringification enabled using CATCH_REGISTER_ENUM" ) {
|
|
|
|
|
using Catch::Detail::stringify;
|
|
|
|
|
REQUIRE( stringify( Bikeshed::Colours::Red ) == "Red" );
|
|
|
|
|
REQUIRE( stringify( Bikeshed::Colours::Blue ) == "Blue" );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE( "parseEnums", "[Strings][enums]" ) {
|
|
|
|
|
using namespace Catch::Matchers;
|
|
|
|
|
using Catch::Detail::parseEnums;
|
|
|
|
|
|
|
|
|
|
SECTION( "No enums" )
|
|
|
|
|
CHECK_THAT( parseEnums( "" ), Equals( std::vector<Catch::StringRef>{} ) );
|
|
|
|
|
|
|
|
|
|
SECTION( "One enum value" ) {
|
|
|
|
|
CHECK_THAT( parseEnums( "ClassName::EnumName::Value1" ),
|
|
|
|
|
Equals(std::vector<Catch::StringRef>{"Value1"} ) );
|
|
|
|
|
CHECK_THAT( parseEnums( "Value1" ),
|
|
|
|
|
Equals( std::vector<Catch::StringRef>{"Value1"} ) );
|
|
|
|
|
CHECK_THAT( parseEnums( "EnumName::Value1" ),
|
|
|
|
|
Equals(std::vector<Catch::StringRef>{"Value1"} ) );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SECTION( "Multiple enum values" ) {
|
|
|
|
|
CHECK_THAT( parseEnums( "ClassName::EnumName::Value1, ClassName::EnumName::Value2" ),
|
|
|
|
|
Equals( std::vector<Catch::StringRef>{"Value1", "Value2"} ) );
|
|
|
|
|
CHECK_THAT( parseEnums( "ClassName::EnumName::Value1, ClassName::EnumName::Value2, ClassName::EnumName::Value3" ),
|
|
|
|
|
Equals( std::vector<Catch::StringRef>{"Value1", "Value2", "Value3"} ) );
|
|
|
|
|
CHECK_THAT( parseEnums( "ClassName::EnumName::Value1,ClassName::EnumName::Value2 , ClassName::EnumName::Value3" ),
|
|
|
|
|
Equals( std::vector<Catch::StringRef>{"Value1", "Value2", "Value3"} ) );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE( "Directly creating an EnumInfo" ) {
|
|
|
|
|
using namespace Catch::Detail;
|
|
|
|
|
auto enumInfo = makeEnumInfo( "EnumName", "EnumName::Value1, EnumName::Value2", {0, 1} );
|
|
|
|
|
|
|
|
|
|
CHECK( enumInfo.lookup(0) == "Value1" );
|
|
|
|
|
CHECK( enumInfo.lookup(1) == "Value2" );
|
|
|
|
|
CHECK( enumInfo.lookup(3) == "{** unexpected enum value **}" );
|
|
|
|
|
}
|
2014-09-01 17:24:10 +01:00
|
|
|
|
2018-07-02 11:13:07 +02:00
|
|
|
namespace {
|
2014-09-01 17:24:10 +01:00
|
|
|
// Enum without user-provided stream operator
|
|
|
|
|
enum Enum1 { Enum1Value0, Enum1Value1 };
|
|
|
|
|
|
|
|
|
|
// Enum with user-provided stream operator
|
|
|
|
|
enum Enum2 { Enum2Value0, Enum2Value1 };
|
|
|
|
|
|
2017-05-02 23:51:03 +02:00
|
|
|
std::ostream& operator<<( std::ostream& os, Enum2 v ) {
|
2014-09-01 17:24:10 +01:00
|
|
|
return os << "E2{" << static_cast<int>(v) << "}";
|
|
|
|
|
}
|
2018-07-02 11:13:07 +02:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
|
TEST_CASE( "toString(enum)", "[toString][enum]" ) {
|
|
|
|
|
Enum1 e0 = Enum1Value0;
|
|
|
|
|
CHECK( ::Catch::Detail::stringify(e0) == "0" );
|
|
|
|
|
Enum1 e1 = Enum1Value1;
|
|
|
|
|
CHECK( ::Catch::Detail::stringify(e1) == "1" );
|
|
|
|
|
}
|
2014-09-01 17:24:10 +01:00
|
|
|
|
2014-09-04 07:27:09 +01:00
|
|
|
TEST_CASE( "toString(enum w/operator<<)", "[toString][enum]" ) {
|
2014-09-01 17:24:10 +01:00
|
|
|
Enum2 e0 = Enum2Value0;
|
2017-05-02 23:51:03 +02:00
|
|
|
CHECK( ::Catch::Detail::stringify(e0) == "E2{0}" );
|
2014-09-01 17:24:10 +01:00
|
|
|
Enum2 e1 = Enum2Value1;
|
2017-05-02 23:51:03 +02:00
|
|
|
CHECK( ::Catch::Detail::stringify(e1) == "E2{1}" );
|
2014-09-01 17:24:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Enum class without user-provided stream operator
|
2018-07-02 11:13:07 +02:00
|
|
|
namespace {
|
2014-09-01 17:24:10 +01:00
|
|
|
enum class EnumClass1 { EnumClass1Value0, EnumClass1Value1 };
|
|
|
|
|
|
|
|
|
|
// Enum class with user-provided stream operator
|
2018-07-02 11:13:07 +02:00
|
|
|
enum class EnumClass2 { EnumClass2Value0, EnumClass2Value1 };
|
2014-09-01 17:24:10 +01:00
|
|
|
|
2017-05-02 23:51:03 +02:00
|
|
|
std::ostream& operator<<( std::ostream& os, EnumClass2 e2 ) {
|
2014-10-03 08:17:40 +01:00
|
|
|
switch( static_cast<int>( e2 ) ) {
|
|
|
|
|
case static_cast<int>( EnumClass2::EnumClass2Value0 ):
|
2014-09-04 07:27:09 +01:00
|
|
|
return os << "E2/V0";
|
2014-10-03 08:17:40 +01:00
|
|
|
case static_cast<int>( EnumClass2::EnumClass2Value1 ):
|
2014-09-04 07:27:09 +01:00
|
|
|
return os << "E2/V1";
|
|
|
|
|
default:
|
2014-10-03 08:17:40 +01:00
|
|
|
return os << "Unknown enum value " << static_cast<int>( e2 );
|
2014-09-01 17:24:10 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-02 11:13:07 +02:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
|
|
TEST_CASE( "toString(enum class)", "[toString][enum][enumClass]" ) {
|
|
|
|
|
EnumClass1 e0 = EnumClass1::EnumClass1Value0;
|
|
|
|
|
CHECK( ::Catch::Detail::stringify(e0) == "0" );
|
|
|
|
|
EnumClass1 e1 = EnumClass1::EnumClass1Value1;
|
|
|
|
|
CHECK( ::Catch::Detail::stringify(e1) == "1" );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2017-07-13 08:52:51 +01:00
|
|
|
TEST_CASE( "toString(enum class w/operator<<)", "[toString][enum][enumClass]" ) {
|
2014-09-01 17:24:10 +01:00
|
|
|
EnumClass2 e0 = EnumClass2::EnumClass2Value0;
|
2017-05-02 23:51:03 +02:00
|
|
|
CHECK( ::Catch::Detail::stringify(e0) == "E2/V0" );
|
2014-09-01 17:24:10 +01:00
|
|
|
EnumClass2 e1 = EnumClass2::EnumClass2Value1;
|
2017-05-02 23:51:03 +02:00
|
|
|
CHECK( ::Catch::Detail::stringify(e1) == "E2/V1" );
|
2014-09-04 07:27:09 +01:00
|
|
|
|
2019-04-25 10:13:11 +01:00
|
|
|
auto e3 = static_cast<EnumClass2>(10);
|
2017-05-02 23:51:03 +02:00
|
|
|
CHECK( ::Catch::Detail::stringify(e3) == "Unknown enum value 10" );
|
2014-09-01 17:24:10 +01:00
|
|
|
}
|
2019-04-04 15:55:46 +01:00
|
|
|
|
|
|
|
|
enum class EnumClass3 { Value1, Value2, Value3, Value4 };
|
|
|
|
|
|
2019-04-21 20:26:46 +03:00
|
|
|
CATCH_REGISTER_ENUM( EnumClass3, EnumClass3::Value1, EnumClass3::Value2, EnumClass3::Value3 )
|
2019-04-04 15:55:46 +01:00
|
|
|
|
|
|
|
|
|
2024-11-12 22:13:04 +01:00
|
|
|
TEST_CASE( "Enums can quickly have stringification enabled using CATCH_REGISTER_ENUM" ) {
|
2019-04-04 15:55:46 +01:00
|
|
|
using Catch::Detail::stringify;
|
|
|
|
|
REQUIRE( stringify( EnumClass3::Value1 ) == "Value1" );
|
|
|
|
|
REQUIRE( stringify( EnumClass3::Value2 ) == "Value2" );
|
|
|
|
|
REQUIRE( stringify( EnumClass3::Value3 ) == "Value3" );
|
2019-04-21 19:56:19 +03:00
|
|
|
REQUIRE( stringify( EnumClass3::Value4 ) == "{** unexpected enum value **}" );
|
2019-04-04 15:55:46 +01:00
|
|
|
|
2022-02-21 09:13:00 -05:00
|
|
|
EnumClass3 ec3 = EnumClass3::Value2;
|
2019-04-04 15:55:46 +01:00
|
|
|
REQUIRE( stringify( ec3 ) == "Value2" );
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-08 10:26:57 +02:00
|
|
|
enum class LargeEnum : std::int64_t {
|
|
|
|
|
Flag1 = 0x01,
|
|
|
|
|
Flag2 = 0x02,
|
|
|
|
|
Flag3 = 0x04,
|
|
|
|
|
Flag4 = 0x08,
|
|
|
|
|
Flag5 = 0x10,
|
|
|
|
|
};
|
2019-04-04 15:55:46 +01:00
|
|
|
|
2026-04-08 10:26:57 +02:00
|
|
|
CATCH_REGISTER_ENUM( LargeEnum,
|
|
|
|
|
LargeEnum::Flag1,
|
|
|
|
|
LargeEnum::Flag2,
|
|
|
|
|
LargeEnum::Flag3,
|
|
|
|
|
LargeEnum::Flag4,
|
|
|
|
|
LargeEnum::Flag5 )
|
2019-04-04 15:55:46 +01:00
|
|
|
|
2026-04-08 10:26:57 +02:00
|
|
|
TEST_CASE( "Enum backed by larger underlying type", "[enum][toString]" ) {
|
2019-04-25 10:13:11 +01:00
|
|
|
using Catch::Detail::stringify;
|
2026-04-08 10:26:57 +02:00
|
|
|
REQUIRE( stringify( LargeEnum::Flag1 ) == "Flag1" );
|
|
|
|
|
REQUIRE( stringify( LargeEnum::Flag2 ) == "Flag2" );
|
|
|
|
|
REQUIRE( stringify( LargeEnum::Flag5 ) == "Flag5" );
|
2019-04-04 15:55:46 +01:00
|
|
|
}
|