1 /* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3 * Data Differential YATL (i.e. libtest) library
5 * Copyright (C) 2012 Data Differential, http://datadifferential.com/
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are
11 * * Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
14 * * Redistributions in binary form must reproduce the above
15 * copyright notice, this list of conditions and the following disclaimer
16 * in the documentation and/or other materials provided with the
19 * * The names of its contributors may not be used to endorse or
20 * promote products derived from this software without specific prior
23 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
26 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
27 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
28 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
29 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
30 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
31 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
32 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
33 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
37 #include "libtest/yatlcon.h"
39 #include <libtest/common.h>
47 std::string
& escape4XML(std::string
const& arg
, std::string
& escaped_string
)
49 escaped_string
.clear();
52 for (std::string::const_iterator x
= arg
.begin(), end
= arg
.end(); x
!= end
; ++x
)
55 if (' ' <= c
and c
<= '~' and c
!= '\\' and c
!= '"' and c
!= '>' and c
!= '<')
75 escaped_string
+= '\\';
77 case '"': escaped_string
+= '"'; break;
78 case '\\': escaped_string
+= '\\'; break;
79 case '\t': escaped_string
+='t'; break;
80 case '\r': escaped_string
+='r'; break;
81 case '\n': escaped_string
+='n'; break;
83 char const* const hexdig
= "0123456789ABCDEF";
85 escaped_string
+= hexdig
[c
>> 4];
86 escaped_string
+= hexdig
[c
& 0xF];
92 return escaped_string
;
97 TestCase(const std::string
& arg
):
103 const std::string
& name() const
108 test_return_t
result() const
113 void result(test_return_t arg
)
118 void result(test_return_t arg
, const libtest::Timer
& timer_
)
124 const libtest::Timer
& timer() const
129 void timer(libtest::Timer
& arg
)
136 test_return_t _result
;
137 libtest::Timer _timer
;
140 Formatter::Formatter(const std::string
& frame_name
, const std::string
& arg
)
142 _suite_name
= frame_name
;
147 Formatter::~Formatter()
149 std::for_each(_testcases
.begin(), _testcases
.end(), DeleteFromVector());
153 TestCase
* Formatter::current()
155 return _testcases
.back();
158 void Formatter::skipped()
160 current()->result(TEST_SKIPPED
);
164 << "[ " << test_strerror(current()->result()) << " ]";
169 void Formatter::failed()
172 current()->result(TEST_FAILURE
);
175 << "." << current()->name() << "\t\t\t\t\t"
176 << "[ " << test_strerror(current()->result()) << " ]";
181 void Formatter::success(const libtest::Timer
& timer_
)
184 current()->result(TEST_SUCCESS
, timer_
);
185 std::string escaped_string
;
190 << current()->timer()
191 << " [ " << test_strerror(current()->result()) << " ]";
196 void Formatter::xml(libtest::Framework
& framework_
, std::ofstream
& output
)
198 std::string escaped_string
;
200 output
<< "<testsuites name="
201 << escape4XML(framework_
.name(), escaped_string
) << ">" << std::endl
;
203 for (Suites::iterator framework_iter
= framework_
.suites().begin();
204 framework_iter
!= framework_
.suites().end();
207 output
<< "\t<testsuite name="
208 << escape4XML((*framework_iter
)->name(), escaped_string
)
209 << " classname=\"\" package=\"\">"
212 for (TestCases::iterator case_iter
= (*framework_iter
)->formatter()->testcases().begin();
213 case_iter
!= (*framework_iter
)->formatter()->testcases().end();
216 output
<< "\t\t<testcase name="
217 << escape4XML((*case_iter
)->name(), escaped_string
)
219 << (*case_iter
)->timer().elapsed_milliseconds()
223 switch ((*case_iter
)->result())
226 output
<< "\t\t <skipped/>" << std::endl
;
230 output
<< "\t\t <failure message=\"\" type=\"\"/>"<< std::endl
;
236 output
<< "\t\t</testcase>" << std::endl
;
238 output
<< "\t</testsuite>" << std::endl
;
240 output
<< "</testsuites>" << std::endl
;
243 void Formatter::push_testcase(const std::string
& arg
)
245 assert(_suite_name
.empty() == false);
246 TestCase
* _current_testcase
= new TestCase(arg
);
247 _testcases
.push_back(_current_testcase
);
250 void Formatter::reset()
253 } // namespace libtest