diff options
| author | zhanyong.wan <zhanyong.wan@861a406c-534a-0410-8894-cb66d6ee9925> | 2012-06-07 20:34:34 +0000 | 
|---|---|---|
| committer | zhanyong.wan <zhanyong.wan@861a406c-534a-0410-8894-cb66d6ee9925> | 2012-06-07 20:34:34 +0000 | 
| commit | a88c9a88e49e90ec414175543b2b7ff2f70866a7 (patch) | |
| tree | 617807532cbbd68ce8e684a570791231b1b9cd92 /include/gtest | |
| parent | a3b859162dd7a4a1798cf8753a03098f2cbdb62e (diff) | |
| download | googletest-a88c9a88e49e90ec414175543b2b7ff2f70866a7.tar.gz googletest-a88c9a88e49e90ec414175543b2b7ff2f70866a7.tar.bz2 googletest-a88c9a88e49e90ec414175543b2b7ff2f70866a7.zip | |
Improves gtest's failure messages.  In particulars, char pointers and
char arrays are not escapped properly.
Diffstat (limited to 'include/gtest')
| -rw-r--r-- | include/gtest/gtest-printers.h | 86 | ||||
| -rw-r--r-- | include/gtest/gtest.h | 97 | ||||
| -rw-r--r-- | include/gtest/internal/gtest-internal.h | 61 | ||||
| -rw-r--r-- | include/gtest/internal/gtest-port.h | 4 | ||||
| -rw-r--r-- | include/gtest/internal/gtest-string.h | 13 | 
5 files changed, 162 insertions, 99 deletions
| diff --git a/include/gtest/gtest-printers.h b/include/gtest/gtest-printers.h index 55d44fa1..0639d9f5 100644 --- a/include/gtest/gtest-printers.h +++ b/include/gtest/gtest-printers.h @@ -630,9 +630,12 @@ void UniversalPrintArray(const T* begin, size_t len, ::std::ostream* os) {    }  }  // This overload prints a (const) char array compactly. -GTEST_API_ void UniversalPrintArray(const char* begin, -                                    size_t len, -                                    ::std::ostream* os); +GTEST_API_ void UniversalPrintArray( +    const char* begin, size_t len, ::std::ostream* os); + +// This overload prints a (const) wchar_t array compactly. +GTEST_API_ void UniversalPrintArray( +    const wchar_t* begin, size_t len, ::std::ostream* os);  // Implements printing an array type T[N].  template <typename T, size_t N> @@ -673,19 +676,72 @@ class UniversalPrinter<T&> {  // Prints a value tersely: for a reference type, the referenced value  // (but not the address) is printed; for a (const) char pointer, the  // NUL-terminated string (but not the pointer) is printed. +  template <typename T> -void UniversalTersePrint(const T& value, ::std::ostream* os) { -  UniversalPrint(value, os); -} -inline void UniversalTersePrint(const char* str, ::std::ostream* os) { -  if (str == NULL) { -    *os << "NULL"; -  } else { -    UniversalPrint(string(str), os); +class UniversalTersePrinter { + public: +  static void Print(const T& value, ::std::ostream* os) { +    UniversalPrint(value, os);    } -} -inline void UniversalTersePrint(char* str, ::std::ostream* os) { -  UniversalTersePrint(static_cast<const char*>(str), os); +}; +template <typename T> +class UniversalTersePrinter<T&> { + public: +  static void Print(const T& value, ::std::ostream* os) { +    UniversalPrint(value, os); +  } +}; +template <typename T, size_t N> +class UniversalTersePrinter<T[N]> { + public: +  static void Print(const T (&value)[N], ::std::ostream* os) { +    UniversalPrinter<T[N]>::Print(value, os); +  } +}; +template <> +class UniversalTersePrinter<const char*> { + public: +  static void Print(const char* str, ::std::ostream* os) { +    if (str == NULL) { +      *os << "NULL"; +    } else { +      UniversalPrint(string(str), os); +    } +  } +}; +template <> +class UniversalTersePrinter<char*> { + public: +  static void Print(char* str, ::std::ostream* os) { +    UniversalTersePrinter<const char*>::Print(str, os); +  } +}; + +#if GTEST_HAS_STD_WSTRING +template <> +class UniversalTersePrinter<const wchar_t*> { + public: +  static void Print(const wchar_t* str, ::std::ostream* os) { +    if (str == NULL) { +      *os << "NULL"; +    } else { +      UniversalPrint(::std::wstring(str), os); +    } +  } +}; +#endif + +template <> +class UniversalTersePrinter<wchar_t*> { + public: +  static void Print(wchar_t* str, ::std::ostream* os) { +    UniversalTersePrinter<const wchar_t*>::Print(str, os); +  } +}; + +template <typename T> +void UniversalTersePrint(const T& value, ::std::ostream* os) { +  UniversalTersePrinter<T>::Print(value, os);  }  // Prints a value using the type inferred by the compiler.  The @@ -790,7 +846,7 @@ Strings UniversalTersePrintTupleFieldsToStrings(const Tuple& value) {  template <typename T>  ::std::string PrintToString(const T& value) {    ::std::stringstream ss; -  internal::UniversalTersePrint(value, &ss); +  internal::UniversalTersePrinter<T>::Print(value, &ss);    return ss.str();  } diff --git a/include/gtest/gtest.h b/include/gtest/gtest.h index 226307ec..a13cfeb8 100644 --- a/include/gtest/gtest.h +++ b/include/gtest/gtest.h @@ -1291,24 +1291,101 @@ GTEST_API_ void InitGoogleTest(int* argc, wchar_t** argv);  namespace internal { +// FormatForComparison<ToPrint, OtherOperand>::Format(value) formats a +// value of type ToPrint that is an operand of a comparison assertion +// (e.g. ASSERT_EQ).  OtherOperand is the type of the other operand in +// the comparison, and is used to help determine the best way to +// format the value.  In particular, when the value is a C string +// (char pointer) and the other operand is an STL string object, we +// want to format the C string as a string, since we know it is +// compared by value with the string object.  If the value is a char +// pointer but the other operand is not an STL string object, we don't +// know whether the pointer is supposed to point to a NUL-terminated +// string, and thus want to print it as a pointer to be safe. +// +// INTERNAL IMPLEMENTATION - DO NOT USE IN A USER PROGRAM. + +// The default case. +template <typename ToPrint, typename OtherOperand> +class FormatForComparison { + public: +  static ::std::string Format(const ToPrint& value) { +    return ::testing::PrintToString(value); +  } +}; + +// Array. +template <typename ToPrint, size_t N, typename OtherOperand> +class FormatForComparison<ToPrint[N], OtherOperand> { + public: +  static ::std::string Format(const ToPrint* value) { +    return FormatForComparison<const ToPrint*, OtherOperand>::Format(value); +  } +}; + +// By default, print C string as pointers to be safe, as we don't know +// whether they actually point to a NUL-terminated string. + +#define GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_(CharType)                \ +  template <typename OtherOperand>                                      \ +  class FormatForComparison<CharType*, OtherOperand> {                  \ +   public:                                                              \ +    static ::std::string Format(CharType* value) {                      \ +      return ::testing::PrintToString(static_cast<const void*>(value)); \ +    }                                                                   \ +  } + +GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_(char); +GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_(const char); +GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_(wchar_t); +GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_(const wchar_t); + +#undef GTEST_IMPL_FORMAT_C_STRING_AS_POINTER_ + +// If a C string is compared with an STL string object, we know it's meant +// to point to a NUL-terminated string, and thus can print it as a string. + +#define GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(CharType, OtherStringType) \ +  template <>                                                           \ +  class FormatForComparison<CharType*, OtherStringType> {               \ +   public:                                                              \ +    static ::std::string Format(CharType* value) {                      \ +      return ::testing::PrintToString(value);                           \ +    }                                                                   \ +  } + +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(char, ::std::string); +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(const char, ::std::string); + +#if GTEST_HAS_GLOBAL_STRING +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(char, ::string); +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(const char, ::string); +#endif + +#if GTEST_HAS_GLOBAL_WSTRING +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(wchar_t, ::wstring); +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(const wchar_t, ::wstring); +#endif + +#if GTEST_HAS_STD_WSTRING +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(wchar_t, ::std::wstring); +GTEST_IMPL_FORMAT_C_STRING_AS_STRING_(const wchar_t, ::std::wstring); +#endif + +#undef GTEST_IMPL_FORMAT_C_STRING_AS_STRING_ +  // Formats a comparison assertion (e.g. ASSERT_EQ, EXPECT_LT, and etc)  // operand to be used in a failure message.  The type (but not value)  // of the other operand may affect the format.  This allows us to  // print a char* as a raw pointer when it is compared against another -// char*, and print it as a C string when it is compared against an -// std::string object, for example. -// -// The default implementation ignores the type of the other operand. -// Some specialized versions are used to handle formatting wide or -// narrow C strings. +// char* or void*, and print it as a C string when it is compared +// against an std::string object, for example.  //  // INTERNAL IMPLEMENTATION - DO NOT USE IN A USER PROGRAM.  template <typename T1, typename T2>  String FormatForComparisonFailureMessage(const T1& value,                                           const T2& /* other_operand */) { -  // C++Builder compiles this incorrectly if the namespace isn't explicitly -  // given. -  return ::testing::PrintToString(value); +  return FormatForComparison<T1, T2>::Format(value);  }  // The helper function for {ASSERT|EXPECT}_EQ. @@ -1320,7 +1397,7 @@ AssertionResult CmpHelperEQ(const char* expected_expression,  #ifdef _MSC_VER  # pragma warning(push)          // Saves the current warning state.  # pragma warning(disable:4389)  // Temporarily disables warning on -                               // signed/unsigned mismatch. +                                // signed/unsigned mismatch.  #endif    if (expected == actual) { diff --git a/include/gtest/internal/gtest-internal.h b/include/gtest/internal/gtest-internal.h index d8834500..a4d18396 100644 --- a/include/gtest/internal/gtest-internal.h +++ b/include/gtest/internal/gtest-internal.h @@ -195,67 +195,6 @@ class GTEST_API_ ScopedTrace {  template <typename T>  String StreamableToString(const T& streamable); -// The Symbian compiler has a bug that prevents it from selecting the -// correct overload of FormatForComparisonFailureMessage (see below) -// unless we pass the first argument by reference.  If we do that, -// however, Visual Age C++ 10.1 generates a compiler error.  Therefore -// we only apply the work-around for Symbian. -#if defined(__SYMBIAN32__) -# define GTEST_CREF_WORKAROUND_ const& -#else -# define GTEST_CREF_WORKAROUND_ -#endif - -// When this operand is a const char* or char*, if the other operand -// is a ::std::string or ::string, we print this operand as a C string -// rather than a pointer (we do the same for wide strings); otherwise -// we print it as a pointer to be safe. - -// This internal macro is used to avoid duplicated code. -#define GTEST_FORMAT_IMPL_(operand2_type, operand1_printer)\ -inline String FormatForComparisonFailureMessage(\ -    operand2_type::value_type* GTEST_CREF_WORKAROUND_ str, \ -    const operand2_type& /*operand2*/) {\ -  return operand1_printer(str);\ -}\ -inline String FormatForComparisonFailureMessage(\ -    const operand2_type::value_type* GTEST_CREF_WORKAROUND_ str, \ -    const operand2_type& /*operand2*/) {\ -  return operand1_printer(str);\ -} - -GTEST_FORMAT_IMPL_(::std::string, String::ShowCStringQuoted) -#if GTEST_HAS_STD_WSTRING -GTEST_FORMAT_IMPL_(::std::wstring, String::ShowWideCStringQuoted) -#endif  // GTEST_HAS_STD_WSTRING - -#if GTEST_HAS_GLOBAL_STRING -GTEST_FORMAT_IMPL_(::string, String::ShowCStringQuoted) -#endif  // GTEST_HAS_GLOBAL_STRING -#if GTEST_HAS_GLOBAL_WSTRING -GTEST_FORMAT_IMPL_(::wstring, String::ShowWideCStringQuoted) -#endif  // GTEST_HAS_GLOBAL_WSTRING - -#undef GTEST_FORMAT_IMPL_ - -// The next four overloads handle the case where the operand being -// printed is a char/wchar_t pointer and the other operand is not a -// string/wstring object.  In such cases, we just print the operand as -// a pointer to be safe. -#define GTEST_FORMAT_CHAR_PTR_IMPL_(CharType)                       \ -  template <typename T>                                             \ -  String FormatForComparisonFailureMessage(CharType* GTEST_CREF_WORKAROUND_ p, \ -                                           const T&) { \ -    return PrintToString(static_cast<const void*>(p));              \ -  } - -GTEST_FORMAT_CHAR_PTR_IMPL_(char) -GTEST_FORMAT_CHAR_PTR_IMPL_(const char) -GTEST_FORMAT_CHAR_PTR_IMPL_(wchar_t) -GTEST_FORMAT_CHAR_PTR_IMPL_(const wchar_t) - -#undef GTEST_FORMAT_CHAR_PTR_IMPL_ -  // Constructs and returns the message for an equality assertion  // (e.g. ASSERT_EQ, EXPECT_STREQ, etc) failure.  // diff --git a/include/gtest/internal/gtest-port.h b/include/gtest/internal/gtest-port.h index 08703e31..d4b69ce6 100644 --- a/include/gtest/internal/gtest-port.h +++ b/include/gtest/internal/gtest-port.h @@ -1585,6 +1585,10 @@ inline bool IsUpper(char ch) {  inline bool IsXDigit(char ch) {    return isxdigit(static_cast<unsigned char>(ch)) != 0;  } +inline bool IsXDigit(wchar_t ch) { +  const unsigned char low_byte = static_cast<unsigned char>(ch); +  return ch == low_byte && isxdigit(low_byte) != 0; +}  inline char ToLower(char ch) {    return static_cast<char>(tolower(static_cast<unsigned char>(ch))); diff --git a/include/gtest/internal/gtest-string.h b/include/gtest/internal/gtest-string.h index a9024508..967b1173 100644 --- a/include/gtest/internal/gtest-string.h +++ b/include/gtest/internal/gtest-string.h @@ -82,15 +82,6 @@ class GTEST_API_ String {   public:    // Static utility methods -  // Returns the input enclosed in double quotes if it's not NULL; -  // otherwise returns "(null)".  For example, "\"Hello\"" is returned -  // for input "Hello". -  // -  // This is useful for printing a C string in the syntax of a literal. -  // -  // Known issue: escape sequences are not handled yet. -  static String ShowCStringQuoted(const char* c_str); -    // Clones a 0-terminated C string, allocating memory using new.  The    // caller is responsible for deleting the return value using    // delete[].  Returns the cloned string, or NULL if the input is @@ -139,10 +130,6 @@ class GTEST_API_ String {    // returned.    static String ShowWideCString(const wchar_t* wide_c_str); -  // Similar to ShowWideCString(), except that this function encloses -  // the converted string in double quotes. -  static String ShowWideCStringQuoted(const wchar_t* wide_c_str); -    // Compares two wide C strings.  Returns true iff they have the same    // content.    // | 
