changeset 0:d098192f01d9

Initial commit to the repository.
author Brian Neal <bgneal@gmail.com>
date Sat, 19 Mar 2011 19:53:12 -0500 (2011-03-20)
parents
children 62a54c46da31
files .hgignore README.markdown fructose_gen.py tests/g.cpp tests/g1.h tests/g2.h tests/g3.h tests/g_expected.cpp tests/test.bash tests/x.cpp tests/x1.h tests/x2.h tests/x3.h tests/x_expected.cpp
diffstat 14 files changed, 828 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.hgignore	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,3 @@
+syntax: glob
+*.pyc
+*.swp
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/README.markdown	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,93 @@
+fructose_gen.py
+===============
+
+Quick Start
+-----------
+
+`fructose_gen.py` is a Python script for auto-generatng the `main()` function
+for the C++ testing framework [Fructose][1].
+
+Sample usage:
+
+    $ python fructose_gen.py [options] test1.h test2.h ... testN.h > main.cpp
+
+In this example, `fructose_gen.py` will read the Fructose test files `test1.h`
+through `testN.h` and produce on standard output C++ code with a generated
+`main()` function. This auto-generated code will instantiate all the test
+instances, register the appropriate tests with each instance, and then execute
+each test in turn.
+
+To see usage information and a list of options:
+
+    $ python fructose_gen.py --help
+
+
+Code Generation Styles
+----------------------
+
+`fructose_gen.py` supports two styles of code generation, described below.
+
+### xUnit Style ###
+
+The default style is xUnit style. In this form, `fructose_gen.py` will scan
+C++ code looking for classes or structs that inherit from `fructose::test_base<>`.
+Inside those classes or structs, member functions that match the following
+pattern are assumed to be test functions:
+
+      void testXXXX(const std::string&)
+
+Upon finding such a function, `fructose_gen.py` will register that member
+function as a test with the name "testXXXX". 
+
+
+### Generator Style ###
+
+To remain backward compatible with the `generator` program that ships with
+Fructose, invoke `fructose_gen.py` with a `-g` or `--generator` option flag.
+
+In this style, `fructose_gen.py` will scan files for the `FRUCTOSE_CLASS`,
+`FRUCTOSE_STRUCT` and `FRUCTOSE_TEST` macros. See the Fructose documentation
+for more information.
+
+
+Caveats
+-------
+
+`fructose_gen.py` is not a true C++ code parser, and in fact is quite simple
+in how it operates. This is sufficient for most cases, but please be aware of
+the following limitations.
+
+1. Ensure your class (or struct) definition is all on one line:
+
+         class my_unit_test : public fructose::test_base<my_unit_test>
+
+   If you split the above across multiple lines `fructose_gen.py` will not
+   recognize your class and will not generate a test instance for it.
+
+2. `fructose_gen.py` does not understand C-style comments or the preprocessor.
+   To comment out a test, you can either use C++ comments, or change the function
+   name slightly to ensure it won't be recognized. Examples:
+
+          /*
+          ** void test_is_sorted(const std::string& name)  // this won't work
+          */
+
+          #if 0
+          void test_is_sorted(const std::string& name)   // this won't work
+          #endif
+
+          void not_a_test_is_sorted(const std::string& name) // this works
+          // void test_is_sorted(const std::string& name)    // this works
+          // FRUCTOSE_TEST(is_sorted)                        // this works
+
+   The above also applies to commenting out test classes.
+
+
+Support
+-------
+See the [fructose_gen support site][2] hosted at [bitbucket.org][3].
+
+
+[1]: http://fructose.sourceforge.net/
+[2]: https://bitbucket.org/bgneal/fructose_gen
+[3]: https://bitbucket.org
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/fructose_gen.py	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,222 @@
+#!/usr/bin/env python
+"""
+Copyright (C) 2011 by Brian Neal <bgneal@gmail.com>
+
+fructose_gen.py - A program to auto-generate the main() routine for the C++
+testing framework Fructose.
+
+"""
+from __future__ import with_statement
+import re
+import optparse
+import sys
+
+
+USAGE = "usage: %prog [options] test1.h test2.h ... > main.cpp"
+DESCRIPTION = "Generates the main() routine for the Fructose C++ testing framework"
+VERSION = "%prog 0.1"
+INDENT_CNT = 4
+INDENT = " " * INDENT_CNT
+
+
+def strip_comments(s):
+    """
+    A simple function to strip out C++ style // comments from a string.
+    This function is really dumb; it doesn't know about string literals, etc.,
+    but it should suit our purposes for finding commented out test classes and
+    cases.
+
+    """
+    i = s.find('//')
+    return s if i == -1 else s[:i]
+
+
+class TestClass(object):
+    """
+    This class represents a Fructose test class.
+    Each test class has a name attribute and a list of test case names
+    (strings).
+
+    """
+    def __init__(self, name):
+        self.name = name
+        self.test_cases = []
+
+
+class TestFile(object):
+    """
+    A class to represent a Fructose unit test file.
+    Each test file has a filename attribute and a list of test classes.
+
+    """
+    def __init__(self, filename):
+        self.filename = filename
+        self.test_classes = []
+
+
+class TestFileParser(object):
+    """
+    Base class for parsing Fructose unit test files.
+    """
+    def __init__(self, filename):
+        self.test_file = TestFile(filename)
+
+    def parse(self):
+        """
+        Parse the file by reading it line by line.
+        Returns a TestFile object that contains the test classes found within.
+
+        """
+        with open(self.test_file.filename, 'r') as f:
+            for line in f:
+                s = strip_comments(line)
+                if s:
+                    self._parse_line(s)
+
+        return self.test_file
+
+    def _parse_line(self, line):
+        """
+        Parses each line of the test file, calling into derived classes to
+        find test classes and test cases.
+
+        """
+        test_class = self._parse_test_class(line)
+        if test_class:
+            self.test_file.test_classes.append(test_class)
+        else:
+            test_case = self._parse_test_case(line)
+            if len(self.test_file.test_classes) and test_case:
+                self.test_file.test_classes[-1].test_cases.append(test_case)
+
+    def _parse_test_class(self, line):
+        """Derived classes override this"""
+        raise NotImplementedError
+
+    def _parse_test_case(self, line):
+        """Derived classes override this"""
+        raise NotImplementedError
+
+
+class GeneratorFileParser(TestFileParser):
+    """
+    This class parses Fructose test files using the generator style of code
+    generation.
+
+    """
+    CLASS_RE = re.compile(r'\bFRUCTOSE_(?:CLASS|STRUCT)\s*\(\s*([a-zA-Z_]\w*)\s*\)')
+    CASE_RE = re.compile(r'\bFRUCTOSE_TEST\s*\(\s*([a-zA-Z_]\w*)\s*\)')
+
+    def _parse_test_class(self, line):
+        m = self.CLASS_RE.search(line)
+        return TestClass(m.group(1)) if m else None
+
+    def _parse_test_case(self, line):
+        m = self.CASE_RE.search(line)
+        return m.group(1) if m else None
+
+
+class XunitFileParser(TestFileParser):
+    """
+    This class parses Fructose test files using the xUnit style of code
+    generation.
+
+    """
+    CLASS_RE = re.compile(r'\b(?:struct|class)\s+([a-zA-Z_]\w*)\s+:\s+public'
+                            r'\s+(?:fructose\s*::\s*)?test_base\s*<\s*\1\s*>')
+
+    CASE_RE = re.compile(r'\bvoid\s+(test\w+)\s*\(const\s+(?:std::)?string\s*&'
+                            r'(?:\s+[a-zA-Z_]\w+)?\s*\)')
+
+    def _parse_test_class(self, line):
+        m = self.CLASS_RE.search(line)
+        return TestClass(m.group(1)) if m else None
+
+    def _parse_test_case(self, line):
+        m = self.CASE_RE.search(line)
+        return m.group(1) if m else None
+
+
+def generate_test_instance(test_class):
+    """
+    Generates the code to instantiate a test instance, register and run the
+    tests.
+
+    """
+    type_name = test_class.name
+    instance = type_name + '_instance'
+
+    print "%s{" % INDENT
+    block_indent = INDENT * 2
+    print "%s%s %s;" % (block_indent, type_name, instance)
+    for test_case in test_class.test_cases:
+        print '%s%s.add_test("%s", &%s::%s);' % (
+                block_indent,
+                instance,
+                test_case,
+                type_name,
+                test_case,
+                )
+    print "%sconst int r = %s.run(argc, argv);" % (block_indent, instance)
+    print "%sretval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;" % block_indent
+    print "%s}" % INDENT
+
+
+
+def generate_main(test_files):
+    """
+    Generates the main() file given a list of TestFile objects.
+
+    """
+    for test_file in test_files:
+        print '#include "%s"' % test_file.filename
+
+    print '\n#include <stdlib.h>\n'
+    print 'int main(int argc, char* argv[])\n{'
+    print '%sint retval = EXIT_SUCCESS;\n' % INDENT
+
+    for test_file in test_files:
+        for test_class in test_file.test_classes:
+            generate_test_instance(test_class)
+
+    print '\n%sreturn retval;\n}' % INDENT
+
+
+def main(argv=None):
+
+    parser = optparse.OptionParser(usage=USAGE, description=DESCRIPTION,
+                                   version=VERSION)
+    parser.set_defaults(
+        generator=False,
+    )
+    parser.add_option("-g", "--generator", action="store_true",
+            help="use generator style code generation [default: %default]")
+
+    opts, args = parser.parse_args(args=argv)
+
+    xunit = not opts.generator
+
+    parser_class = XunitFileParser if xunit else GeneratorFileParser
+
+    if len(args) == 0:
+        sys.exit("No input files")
+
+    test_files = []
+    for test_file in args:
+
+        test_parser = parser_class(test_file)
+        try:
+            test_files.append(test_parser.parse())
+        except IOError, ex:
+            sys.stderr.write("Error parsing %s: %s, skipping" % (test_file, ex))
+
+    generate_main(test_files)
+
+
+if __name__ == '__main__':
+    try:
+        main()
+    except IOError, ex:
+        sys.exit("IO Error: %s" % ex)
+    except KeyboardInterrupt:
+        sys.exit("Control-C interrupt")
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/g.cpp	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,41 @@
+#include "g1.h"
+#include "g2.h"
+#include "g3.h"
+
+#include <stdlib.h>
+
+int main(int argc, char* argv[])
+{
+    int retval = EXIT_SUCCESS;
+
+    {
+        sample_test sample_test_instance;
+        sample_test_instance.add_test("test42", &sample_test::test42);
+        sample_test_instance.add_test("beast", &sample_test::beast);
+        sample_test_instance.add_test("fivealive", &sample_test::fivealive);
+        const int r = sample_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        misc_tests misc_tests_instance;
+        misc_tests_instance.add_test("exceptions", &misc_tests::exceptions);
+        misc_tests_instance.add_test("loopdata", &misc_tests::loopdata);
+        misc_tests_instance.add_test("floatingpoint", &misc_tests::floatingpoint);
+        const int r = misc_tests_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        exception_test exception_test_instance;
+        exception_test_instance.add_test("array_bounds", &exception_test::array_bounds);
+        exception_test_instance.add_test("should_catch_std_exceptions", &exception_test::should_catch_std_exceptions);
+        const int r = exception_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        MyTest MyTest_instance;
+        const int r = MyTest_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+
+    return retval;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/g1.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,29 @@
+#ifndef FRUCTOSE_MAIN_TEST_X1_H
+#define FRUCTOSE_MAIN_TEST_X1_H
+
+#include "fructose/fructose.h"
+
+const int life_the_available_tests_and_everything = 42;
+const int the_neighbour_of_the_beast = 668;
+const int is_alive = 6;
+
+FRUCTOSE_CLASS(sample_test)
+{
+public:
+    FRUCTOSE_TEST(test42)
+    {
+        fructose_assert(life_the_available_tests_and_everything == 6*7);
+    }
+
+    FRUCTOSE_TEST(beast)
+    {
+        fructose_assert(the_neighbour_of_the_beast == 668);
+    }
+
+    FRUCTOSE_TEST(fivealive)
+    {
+        const int five = 5;
+        fructose_assert_eq(five, is_alive);
+    }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/g2.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,68 @@
+#ifndef FRUCTOSE_MAIN_TEST_X2_H
+#define FRUCTOSE_MAIN_TEST_X2_H
+
+#include <stdexcept>
+#include <cmath>
+
+#include "fructose/fructose.h"
+
+// These tests are rigged so that some of them fail.
+// The tests include exercising floating point comparisons
+// and exception handling.
+
+namespace {
+    void throw_a_runtime_error()
+    {
+        throw std::runtime_error("this is a runtime error");
+    }
+}
+
+FRUCTOSE_STRUCT(misc_tests)
+{
+    FRUCTOSE_TEST(exceptions)
+    {
+        fructose_assert_exception(throw_a_runtime_error(), std::logic_error);
+        fructose_assert_no_exception(throw_a_runtime_error());
+    }
+
+    FRUCTOSE_TEST(loopdata)
+    {
+        static const struct {
+            int line_number;
+            int a;
+            int b;
+            int expected_result;
+        } data[] = {
+            {__LINE__, 3, 4, 12}
+          , {__LINE__, 1, 50, 50}
+          , {__LINE__, 5, 12, 60}
+          , {__LINE__, 6, 6, 37}
+          , {__LINE__, 7, 10, 70}
+        };
+
+        for (std::size_t i = 0; i < sizeof(data)/sizeof(data[0]); ++i)
+        {
+            if (verbose())
+            {
+                std::cout << "Testing to see if "
+                          << data[i].a << " * " << data[i].b 
+                          << " = " << data[i].expected_result
+                          << std::endl;
+            }
+            int result = data[i].a * data[i].b;
+            fructose_loop1_assert(data[i].line_number, i,
+                                 result == data[i].expected_result);
+        }
+    }
+
+    FRUCTOSE_TEST(floatingpoint)
+    {
+        double my_pi = 3.141592654;
+        double calc_pi = 4.0 * atan(1.0);
+        fructose_assert_double_eq_rel_abs(my_pi, calc_pi, 0.01, 0.01);
+        fructose_assert_double_eq(my_pi, calc_pi);
+        fructose_assert_double_ne(my_pi, calc_pi);
+        fructose_assert_double_ne_rel_abs(my_pi, calc_pi, 0.01, 0.01);
+    }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/g3.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,64 @@
+#ifndef FRUCTOSE_MAIN_TEST_X3_H
+#define FRUCTOSE_MAIN_TEST_X3_H
+
+#include <stdexcept>
+#include <vector>
+
+#include "fructose/fructose.h"
+
+struct my_logic_error : public std::logic_error
+{
+    my_logic_error(const char* message) : std::logic_error(message) {}
+};
+
+struct my_runtime_error : public std::runtime_error
+{
+    my_runtime_error(const char* message) : std::runtime_error(message) {}
+};
+
+FRUCTOSE_CLASS(exception_test)
+{
+public:
+    FRUCTOSE_TEST(array_bounds)
+    {
+      std::vector<int> v;
+      v.push_back(1234);
+      fructose_assert_exception(v.at(2), std::out_of_range);
+    }
+
+    FRUCTOSE_TEST(should_catch_std_exceptions)
+    {
+        fructose_assert_exception(throw my_logic_error("a coding error has been detected"), 
+                                  std::logic_error);
+        fructose_assert_exception(throw my_runtime_error("my runtime error"),
+                                  std::runtime_error);
+        fructose_assert_exception(throw my_logic_error("another coding error has been detected"), 
+                                  std::exception);
+        fructose_assert_exception(throw my_runtime_error("my runtime error"),
+                                  std::exception);
+    }
+
+    //FRUCTOSE_TEST(commented_out)
+    //{
+    //   fructose_assert(false);
+    //}
+};
+
+// FRUCTOSE_CLASS(MyTest)
+// {
+// public:
+//    void testIt(const std::string&)
+//    {
+//       fructose_assert(true);
+//    }
+// };
+
+FRUCTOSE_CLASS(MyTest)
+{
+public:
+   void testIt(const std::string&)
+   {
+      fructose_assert(true);
+   }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/g_expected.cpp	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,41 @@
+#include "g1.h"
+#include "g2.h"
+#include "g3.h"
+
+#include <stdlib.h>
+
+int main(int argc, char* argv[])
+{
+    int retval = EXIT_SUCCESS;
+
+    {
+        sample_test sample_test_instance;
+        sample_test_instance.add_test("test42", &sample_test::test42);
+        sample_test_instance.add_test("beast", &sample_test::beast);
+        sample_test_instance.add_test("fivealive", &sample_test::fivealive);
+        const int r = sample_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        misc_tests misc_tests_instance;
+        misc_tests_instance.add_test("exceptions", &misc_tests::exceptions);
+        misc_tests_instance.add_test("loopdata", &misc_tests::loopdata);
+        misc_tests_instance.add_test("floatingpoint", &misc_tests::floatingpoint);
+        const int r = misc_tests_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        exception_test exception_test_instance;
+        exception_test_instance.add_test("array_bounds", &exception_test::array_bounds);
+        exception_test_instance.add_test("should_catch_std_exceptions", &exception_test::should_catch_std_exceptions);
+        const int r = exception_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        MyTest MyTest_instance;
+        const int r = MyTest_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+
+    return retval;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/test.bash	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,23 @@
+#!/bin/bash
+
+INCLUDES="-I../../fructose/include -I../../tclap-1.2.0/include"
+
+../fructose_gen.py x1.h x2.h x3.h > x.cpp
+#g++ -o x.out $INCLUDES x.cpp
+
+../fructose_gen.py --generator g1.h g2.h g3.h > g.cpp
+#g++ -o g.out $INCLUDES g.cpp
+
+diff x_expected.cpp x.cpp >/dev/null
+if [ $? -eq 0 ]; then
+   echo "xunit style passed"
+else
+   echo "xunit style failed"
+fi
+
+diff g_expected.cpp g.cpp >/dev/null
+if [ $? -eq 0 ]; then
+   echo "generator style passed"
+else
+   echo "generator style failed"
+fi
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/x.cpp	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,42 @@
+#include "x1.h"
+#include "x2.h"
+#include "x3.h"
+
+#include <stdlib.h>
+
+int main(int argc, char* argv[])
+{
+    int retval = EXIT_SUCCESS;
+
+    {
+        sample_test sample_test_instance;
+        sample_test_instance.add_test("test42", &sample_test::test42);
+        sample_test_instance.add_test("testbeast", &sample_test::testbeast);
+        sample_test_instance.add_test("testfivealive", &sample_test::testfivealive);
+        const int r = sample_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        misc_tests misc_tests_instance;
+        misc_tests_instance.add_test("testexceptions", &misc_tests::testexceptions);
+        misc_tests_instance.add_test("testloopdata", &misc_tests::testloopdata);
+        misc_tests_instance.add_test("testfloatingpoint", &misc_tests::testfloatingpoint);
+        const int r = misc_tests_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        exception_test exception_test_instance;
+        exception_test_instance.add_test("test_array_bounds", &exception_test::test_array_bounds);
+        exception_test_instance.add_test("test_should_catch_std_exceptions", &exception_test::test_should_catch_std_exceptions);
+        const int r = exception_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        MyTest MyTest_instance;
+        MyTest_instance.add_test("testIt", &MyTest::testIt);
+        const int r = MyTest_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+
+    return retval;
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/x1.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,28 @@
+#ifndef FRUCTOSE_MAIN_TEST_X1_H
+#define FRUCTOSE_MAIN_TEST_X1_H
+
+#include "fructose/fructose.h"
+
+const int life_the_available_tests_and_everything = 42;
+const int the_neighbour_of_the_beast = 668;
+const int is_alive = 6;
+
+struct sample_test : public fructose::test_base<sample_test>
+{
+    void test42(const std::string& test_name)
+    {
+        fructose_assert(life_the_available_tests_and_everything == 6*7);
+    }
+
+    void testbeast(const std::string& test_name)
+    {
+        fructose_assert(the_neighbour_of_the_beast == 668);
+    }
+
+    void testfivealive(const std::string& test_name)
+    {
+        const int five = 5;
+        fructose_assert_eq(five, is_alive);
+    }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/x2.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,69 @@
+#ifndef FRUCTOSE_MAIN_TEST_X2_H
+#define FRUCTOSE_MAIN_TEST_X2_H
+
+#include <stdexcept>
+#include <cmath>
+
+#include "fructose/fructose.h"
+
+// These tests are rigged so that some of them fail.
+// The tests include exercising floating point comparisons
+// and exception handling.
+
+namespace {
+    void throw_a_runtime_error()
+    {
+        throw std::runtime_error("this is a runtime error");
+    }
+}
+
+class misc_tests : public fructose::test_base<misc_tests>
+{
+public:
+    void testexceptions(const std::string& test_name)
+    {
+        fructose_assert_exception(throw_a_runtime_error(), std::logic_error);
+        fructose_assert_no_exception(throw_a_runtime_error());
+    }
+
+    void testloopdata(const std::string& test_name)
+    {
+        static const struct {
+            int line_number;
+            int a;
+            int b;
+            int expected_result;
+        } data[] = {
+            {__LINE__, 3, 4, 12}
+          , {__LINE__, 1, 50, 50}
+          , {__LINE__, 5, 12, 60}
+          , {__LINE__, 6, 6, 37}
+          , {__LINE__, 7, 10, 70}
+        };
+
+        for (std::size_t i = 0; i < sizeof(data)/sizeof(data[0]); ++i)
+        {
+            if (verbose())
+            {
+                std::cout << "Testing to see if "
+                          << data[i].a << " * " << data[i].b 
+                          << " = " << data[i].expected_result
+                          << std::endl;
+            }
+            int result = data[i].a * data[i].b;
+            fructose_loop1_assert(data[i].line_number, i,
+                                 result == data[i].expected_result);
+        }
+    }
+
+    void testfloatingpoint(const std::string& test_name)
+    {
+        double my_pi = 3.141592654;
+        double calc_pi = 4.0 * atan(1.0);
+        fructose_assert_double_eq_rel_abs(my_pi, calc_pi, 0.01, 0.01);
+        fructose_assert_double_eq(my_pi, calc_pi);
+        fructose_assert_double_ne(my_pi, calc_pi);
+        fructose_assert_double_ne_rel_abs(my_pi, calc_pi, 0.01, 0.01);
+    }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/x3.h	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,63 @@
+#ifndef FRUCTOSE_MAIN_TEST_X3_H
+#define FRUCTOSE_MAIN_TEST_X3_H
+
+#include <stdexcept>
+#include <vector>
+
+#include "fructose/fructose.h"
+
+struct my_logic_error : public std::logic_error
+{
+    my_logic_error(const char* message) : std::logic_error(message) {}
+};
+
+struct my_runtime_error : public std::runtime_error
+{
+    my_runtime_error(const char* message) : std::runtime_error(message) {}
+};
+
+struct exception_test : public fructose::test_base<exception_test> 
+{
+    void test_array_bounds(const std::string&) 
+    {
+      std::vector<int> v;
+      v.push_back(1234);
+      fructose_assert_exception(v.at(2), std::out_of_range);
+    }
+
+    void test_should_catch_std_exceptions(const std::string& )
+    {
+        fructose_assert_exception(throw my_logic_error("a coding error has been detected"), 
+                                  std::logic_error);
+        fructose_assert_exception(throw my_runtime_error("my runtime error"),
+                                  std::runtime_error);
+        fructose_assert_exception(throw my_logic_error("another coding error has been detected"), 
+                                  std::exception);
+        fructose_assert_exception(throw my_runtime_error("my runtime error"),
+                                  std::exception);
+    }
+
+    //void test_commented_out(const std::string&)
+    //{
+    //   fructose_assert(false);
+    //}
+};
+
+// class MyTest : public fructose::test_base<MyTest>
+// {
+// public:
+//    void testIt(const std::string&)
+//    {
+//       fructose_assert(true);
+//    }
+//  };
+
+class MyTest : public fructose::test_base<MyTest>
+{
+public:
+   void testIt(const std::string&)
+   {
+      fructose_assert(true);
+   }
+};
+#endif
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/x_expected.cpp	Sat Mar 19 19:53:12 2011 -0500
@@ -0,0 +1,42 @@
+#include "x1.h"
+#include "x2.h"
+#include "x3.h"
+
+#include <stdlib.h>
+
+int main(int argc, char* argv[])
+{
+    int retval = EXIT_SUCCESS;
+
+    {
+        sample_test sample_test_instance;
+        sample_test_instance.add_test("test42", &sample_test::test42);
+        sample_test_instance.add_test("testbeast", &sample_test::testbeast);
+        sample_test_instance.add_test("testfivealive", &sample_test::testfivealive);
+        const int r = sample_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        misc_tests misc_tests_instance;
+        misc_tests_instance.add_test("testexceptions", &misc_tests::testexceptions);
+        misc_tests_instance.add_test("testloopdata", &misc_tests::testloopdata);
+        misc_tests_instance.add_test("testfloatingpoint", &misc_tests::testfloatingpoint);
+        const int r = misc_tests_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        exception_test exception_test_instance;
+        exception_test_instance.add_test("test_array_bounds", &exception_test::test_array_bounds);
+        exception_test_instance.add_test("test_should_catch_std_exceptions", &exception_test::test_should_catch_std_exceptions);
+        const int r = exception_test_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+    {
+        MyTest MyTest_instance;
+        MyTest_instance.add_test("testIt", &MyTest::testIt);
+        const int r = MyTest_instance.run(argc, argv);
+        retval = retval == EXIT_SUCCESS ? r : EXIT_FAILURE;
+    }
+
+    return retval;
+}