ProteoWizard
unit.hpp
Go to the documentation of this file.
1 //
2 // $Id: unit.hpp 4138 2012-11-21 22:54:56Z chambm $
3 //
4 //
5 // Original author: Darren Kessner <darren@proteowizard.org>
6 //
7 // Copyright 2006 Louis Warschaw Prostate Cancer Center
8 // Cedars Sinai Medical Center, Los Angeles, California 90048
9 //
10 // Licensed under the Apache License, Version 2.0 (the "License");
11 // you may not use this file except in compliance with the License.
12 // You may obtain a copy of the License at
13 //
14 // http://www.apache.org/licenses/LICENSE-2.0
15 //
16 // Unless required by applicable law or agreed to in writing, software
17 // distributed under the License is distributed on an "AS IS" BASIS,
18 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19 // See the License for the specific language governing permissions and
20 // limitations under the License.
21 //
22 
23 
24 #ifndef _UNIT_HPP_
25 #define _UNIT_HPP_
26 
27 
28 #include "Exception.hpp"
29 #include "DateTime.hpp"
30 #include "Filesystem.hpp"
32 #include <string>
33 #include <sstream>
34 #include <cmath>
35 
36 
37 namespace pwiz {
38 namespace util {
39 
40 
41 //
42 // These are assertion macros for unit testing. They throw a runtime_error
43 // exception on failure, instead of calling abort(), allowing the application
44 // to recover and return an appropriate error value to the shell.
45 //
46 // unit_assert(x): asserts x is true
47 // unit_assert_equal(x, y, epsilon): asserts x==y, within epsilon
48 // unit_assert_matrices_equal(A, B, epsilon): asserts A==B, within epsilon
49 //
50 
51 
52 inline std::string unit_assert_message(const char* filename, int line, const char* expression)
53 {
54  std::ostringstream oss;
55  oss << "[" << filename << ":" << line << "] Assertion failed: " << expression;
56  return oss.str();
57 }
58 
59 inline std::string unit_assert_equal_message(const char* filename, int line, const std::string& x, const std::string& y, const char* expression)
60 {
61  std::ostringstream oss;
62  oss << "[" << filename << ":" << line << "] Assertion failed: expected \"" << x << "\" but got \"" << y << "\" (" << expression << ")";
63  return oss.str();
64 }
65 
66 inline std::string unit_assert_numeric_equal_message(const char* filename, int line, double x, double y, double epsilon)
67 {
68  std::ostringstream oss;
69  oss.precision(10);
70  oss << "[" << filename << ":" << line << "] Assertion failed: |" << x << " - " << y << "| < " << epsilon;
71  return oss.str();
72 }
73 
74 inline std::string unit_assert_exception_message(const char* filename, int line, const char* expression, const std::string& exception)
75 {
76  std::ostringstream oss;
77  oss << "[" << filename << ":" << line << "] Assertion failed to throw \"" << exception << "\": " << expression;
78  return oss.str();
79 }
80 
81 
82 #define unit_assert(x) \
83  (!(x) ? throw std::runtime_error(unit_assert_message(__FILE__, __LINE__, #x)) : 0)
84 
85 
86 #define unit_assert_operator_equal(expected, actual) \
87  (!(expected == actual) ? throw std::runtime_error(unit_assert_equal_message(__FILE__, __LINE__, lexical_cast<string>(expected), lexical_cast<string>(actual), #actual)) : 0)
88 
89 
90 #define unit_assert_equal(x, y, epsilon) \
91  (!(fabs((x)-(y)) <= (epsilon)) ? throw std::runtime_error(unit_assert_numeric_equal_message(__FILE__, __LINE__, (x), (y), (epsilon))) : 0)
92 
93 
94 #define unit_assert_throws(x, exception) \
95  { \
96  bool threw = false; \
97  try { (x); } \
98  catch (exception&) \
99  { \
100  threw = true; \
101  } \
102  if (!threw) \
103  throw std::runtime_error(unit_assert_exception_message(__FILE__, __LINE__, #x, #exception)); \
104  }
105 
106 
107 #define unit_assert_throws_what(x, exception, whatStr) \
108  { \
109  bool threw = false; \
110  try { (x); } \
111  catch (exception& e) \
112  { \
113  if (e.what() == std::string(whatStr)) \
114  threw = true; \
115  else \
116  throw std::runtime_error(unit_assert_exception_message(__FILE__, __LINE__, #x, std::string(#exception)+" "+(whatStr)+"\nBut a different exception was thrown: ")+(e.what())); \
117  } \
118  if (!threw) \
119  throw std::runtime_error(unit_assert_exception_message(__FILE__, __LINE__, #x, std::string(#exception)+" "+(whatStr))); \
120  }
121 
122 
123 #define unit_assert_matrices_equal(A, B, epsilon) \
124  unit_assert(boost::numeric::ublas::norm_frobenius((A)-(B)) < (epsilon))
125 
126 
127 #define unit_assert_vectors_equal(A, B, epsilon) \
128  unit_assert(boost::numeric::ublas::norm_2((A)-(B)) < (epsilon))
129 
130 
131 // the following macros are used by the ProteoWizard tests to report test status and duration to TeamCity
132 
133 inline std::string escape_teamcity_string(const std::string& str)
134 {
135  string result = str;
136  bal::replace_all(result, "'", "|'");
137  bal::replace_all(result, "\n", "|n");
138  bal::replace_all(result, "\r", "|r");
139  bal::replace_all(result, "|", "||");
140  bal::replace_all(result, "[", "|[");
141  bal::replace_all(result, "]", "|]");
142  return result;
143 }
144 
145 #define TEST_PROLOG_EX(argc, argv, suffix) \
146  bfs::path testName = bfs::change_extension(bfs::basename(argv[0]), (suffix)); \
147  string teamcityTestName = pwiz::util::escape_teamcity_string(testName.string()); \
148  bpt::ptime testStartTime; \
149  vector<string> testArgs(argv, argv+argc); \
150  bool teamcityTestDecoration = find(testArgs.begin(), testArgs.end(), "--teamcity-test-decoration") != testArgs.end(); \
151  if (teamcityTestDecoration) \
152  { \
153  testStartTime = bpt::microsec_clock::local_time(); \
154  cout << "##teamcity[testStarted name='" << teamcityTestName << "']" << endl; \
155  } \
156  int testExitStatus = 0;
157 
158 #define TEST_PROLOG(argc, argv) TEST_PROLOG_EX(argc, argv, "")
159 
160 #define TEST_FAILED(x) \
161  if (teamcityTestDecoration) \
162  cout << "##teamcity[testFailed name='" << teamcityTestName << "' message='" << pwiz::util::escape_teamcity_string((x)) << "']\n"; \
163  cerr << (x) << endl; \
164  testExitStatus = 1;
165 
166 #define TEST_EPILOG \
167  if (teamcityTestDecoration) \
168  cout << "##teamcity[testFinished name='" << teamcityTestName << \
169  "' duration='" << round((bpt::microsec_clock::local_time() - testStartTime).total_microseconds() / 1000.0) << "']" << endl; \
170  return testExitStatus;
171 
172 
173 } // namespace util
174 } // namespace pwiz
175 
176 
177 #endif // _UNIT_HPP_
178