• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright John Maddock 2008
2 // Copyright Paul A. Bristow
3 // Copyright Gautam Sewani
4 
5 // Use, modification and distribution are subject to the
6 // Boost Software License, Version 1.0.
7 // (See accompanying file LICENSE_1_0.txt
8 // or copy at http://www.boost.org/LICENSE_1_0.txt)
9 
10 #define BOOST_MATH_OVERFLOW_ERROR_POLICY throw_on_error
11 #include <boost/math/concepts/real_concept.hpp> // for real_concept
12 #include <boost/math/distributions/hypergeometric.hpp>
13 
14 #define BOOST_TEST_MAIN
15 #include <boost/test/unit_test.hpp> // Boost.Test
16 #include <boost/test/results_collector.hpp>
17 #include <boost/test/unit_test.hpp>
18 #include <boost/test/tools/floating_point_comparison.hpp>
19 
20 #include <iostream>
21    using std::cout;
22    using std::endl;
23    using std::setprecision;
24 
25 #include <boost/array.hpp>
26 #include "functor.hpp"
27 #include "handle_test_result.hpp"
28 #include "table_type.hpp"
29 
30 #define BOOST_CHECK_EX(a) \
31    {\
32       unsigned int failures = boost::unit_test::results_collector.results( boost::unit_test::framework::current_test_case().p_id ).p_assertions_failed;\
33       BOOST_CHECK(a); \
34       if(failures != boost::unit_test::results_collector.results( boost::unit_test::framework::current_test_case().p_id ).p_assertions_failed)\
35       {\
36          std::cerr << "Failure was with data ";\
37          std::cerr << std::setprecision(35); \
38          std::cerr << "x = " << x << ", r = " << r << ", n = " << n\
39          << ", N = " << N << ", p = " << cp << ", q = " << ccp << std::endl;\
40       }\
41    }
42 
expected_results()43 void expected_results()
44 {
45    //
46    // Define the max and mean errors expected for
47    // various compilers and platforms.
48    //
49    const char* largest_type;
50 #ifndef BOOST_MATH_NO_LONG_DOUBLE_MATH_FUNCTIONS
51    if(boost::math::policies::digits<double, boost::math::policies::policy<> >() == boost::math::policies::digits<long double, boost::math::policies::policy<> >())
52    {
53       largest_type = "(long\\s+)?double|real_concept";
54    }
55    else
56    {
57       largest_type = "long double|real_concept";
58    }
59 #else
60    largest_type = "(long\\s+)?double";
61 #endif
62 #ifndef BOOST_MATH_NO_LONG_DOUBLE_MATH_FUNCTIONS
63    if((boost::math::tools::digits<long double>() > boost::math::tools::digits<double>())
64       && (boost::math::tools::digits<long double>() < 100))
65    {
66       //
67       // Some split of errors from long double into double:
68       //
69       add_expected_result(
70          ".*",                          // compiler
71          ".*",                          // stdlib
72          ".*",                          // platform
73          "double",                      // test type(s)
74          "Random.*",                    // test data group
75          ".*", 1500, 1500);      // test function
76       add_expected_result(
77          ".*",                          // compiler
78          ".*",                          // stdlib
79          ".*",                          // platform
80          "double",                      // test type(s)
81          ".*",                    // test data group
82          ".*", 10, 10);      // test function
83    }
84 #endif
85 
86    add_expected_result(
87       ".*",                          // compiler
88       ".*",                          // stdlib
89       ".*",                          // platform
90       "real_concept",                // test type(s)
91       "Random.*",                    // test data group
92       ".*", 250000000, 25000000);    // test function
93    add_expected_result(
94       ".*",                          // compiler
95       ".*",                          // stdlib
96       ".*",                          // platform
97       largest_type,                  // test type(s)
98       "Random.*",                    // test data group
99       ".*", 10000000, 5000000);      // test function
100    add_expected_result(
101       ".*",                          // compiler
102       ".*",                          // stdlib
103       ".*",                          // platform
104       largest_type,                  // test type(s)
105       ".*",                          // test data group
106       ".*", 50, 20);                 // test function
107 }
108 
109 template <class T>
make_unsigned(T x)110 inline unsigned make_unsigned(T x)
111 {
112    return static_cast<unsigned>(x);
113 }
114 template<>
make_unsigned(boost::math::concepts::real_concept x)115 inline unsigned make_unsigned(boost::math::concepts::real_concept x)
116 {
117    return static_cast<unsigned>(x.value());
118 }
119 
120 template <class T>
pdf_tester(T r,T n,T N,T x)121 T pdf_tester(T r, T n, T N, T x)
122 {
123    boost::math::hypergeometric_distribution<T> d(make_unsigned(r), make_unsigned(n), make_unsigned(N));
124    return pdf(d, x);
125 }
126 
127 template <class T>
cdf_tester(T r,T n,T N,T x)128 T cdf_tester(T r, T n, T N, T x)
129 {
130    boost::math::hypergeometric_distribution<T> d(make_unsigned(r), make_unsigned(n), make_unsigned(N));
131    return cdf(d, x);
132 }
133 
134 template <class T>
ccdf_tester(T r,T n,T N,T x)135 T ccdf_tester(T r, T n, T N, T x)
136 {
137    boost::math::hypergeometric_distribution<T> d(make_unsigned(r), make_unsigned(n), make_unsigned(N));
138    return cdf(complement(d, x));
139 }
140 
141 template <class Real, class T>
do_test_hypergeometric(const T & data,const char * type_name,const char * test_name)142 void do_test_hypergeometric(const T& data, const char* type_name, const char* test_name)
143 {
144    // warning suppression:
145    (void)data;
146    (void)type_name;
147    (void)test_name;
148 
149 #if !defined(TEST_QUANT) || (TEST_QUANT == 0)
150    typedef Real                   value_type;
151 
152    typedef value_type (*pg)(value_type, value_type, value_type, value_type);
153 #if defined(BOOST_MATH_NO_DEDUCED_FUNCTION_POINTERS)
154    pg funcp = pdf_tester<value_type>;
155 #else
156    pg funcp = pdf_tester;
157 #endif
158 
159    boost::math::tools::test_result<value_type> result;
160 
161    std::cout << "Testing " << test_name << " with type " << type_name
162       << "\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n";
163 
164    //
165    // test hypergeometric against data:
166    //
167    result = boost::math::tools::test_hetero<Real>(
168       data,
169       bind_func<Real>(funcp, 0, 1, 2, 3),
170       extract_result<Real>(4));
171    handle_test_result(result, data[result.worst()], result.worst(), type_name, "hypergeometric PDF", test_name);
172 
173 #if defined(BOOST_MATH_NO_DEDUCED_FUNCTION_POINTERS)
174    funcp = cdf_tester<value_type>;
175 #else
176    funcp = cdf_tester;
177 #endif
178 
179    //
180    // test hypergeometric against data:
181    //
182    result = boost::math::tools::test_hetero<Real>(
183       data,
184       bind_func<Real>(funcp, 0, 1, 2, 3),
185       extract_result<Real>(5));
186    handle_test_result(result, data[result.worst()], result.worst(), type_name, "hypergeometric CDF", test_name);
187 
188 #if defined(BOOST_MATH_NO_DEDUCED_FUNCTION_POINTERS)
189    funcp = ccdf_tester<value_type>;
190 #else
191    funcp = ccdf_tester;
192 #endif
193 
194    //
195    // test hypergeometric against data:
196    //
197    result = boost::math::tools::test_hetero<Real>(
198       data,
199       bind_func<Real>(funcp, 0, 1, 2, 3),
200       extract_result<Real>(6));
201    handle_test_result(result, data[result.worst()], result.worst(), type_name, "hypergeometric CDF complement", test_name);
202    std::cout << std::endl;
203 #endif
204 }
205 
206 template <class Real, class T>
do_test_hypergeometric_quantile(const T & data,const char * type_name,const char * test_name)207 void do_test_hypergeometric_quantile(const T& data, const char* type_name, const char* test_name)
208 {
209    typedef Real                   value_type;
210 
211    std::cout << "Checking quantiles with " << test_name << " with type " << type_name
212       << "\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n";
213 
214    if(boost::math::tools::digits<value_type>() > 50)
215    {
216       for(unsigned i = 0; i < data.size(); ++i)
217       {
218          using namespace boost::math::policies;
219 
220          unsigned r = make_unsigned(data[i][0]);
221          unsigned n = make_unsigned(data[i][1]);
222          unsigned N = make_unsigned(data[i][2]);
223          unsigned x = make_unsigned(data[i][3]);
224          value_type cp = data[i][5];
225          value_type ccp = data[i][6];
226          //
227          // A bit of warning suppression:
228          //
229          (void)x;
230          (void)n;
231          (void)r;
232          (void)N;
233          (void)cp;
234          (void)ccp;
235 
236 #if !defined(TEST_QUANT) || (TEST_QUANT == 1)
237          boost::math::hypergeometric_distribution<value_type,
238             policy<discrete_quantile<integer_round_up> > > du(r, n, N);
239 
240          if((cp < 0.9) && (cp > boost::math::tools::min_value<value_type>()))
241          {
242             BOOST_CHECK_EX(quantile(du, cp) >= x);
243          }
244          if((ccp < 0.9) && (ccp > boost::math::tools::min_value<value_type>()))
245          {
246             BOOST_CHECK_EX(quantile(complement(du, ccp)) >= x);
247          }
248 #endif
249 #if !defined(TEST_QUANT) || (TEST_QUANT == 2)
250          boost::math::hypergeometric_distribution<value_type,
251             policy<discrete_quantile<integer_round_down> > > dl(r, n, N);
252          if((cp < 0.9) && (cp > boost::math::tools::min_value<value_type>()))
253          {
254             BOOST_CHECK_EX(quantile(dl, cp) <= x);
255          }
256          if((ccp < 0.9) && (ccp > boost::math::tools::min_value<value_type>()))
257          {
258             BOOST_CHECK_EX(quantile(complement(dl, ccp)) <= x);
259          }
260 #endif
261 #if !defined(TEST_QUANT) || (TEST_QUANT == 3)
262          boost::math::hypergeometric_distribution<value_type,
263             policy<discrete_quantile<integer_round_nearest> > > dn(r, n, N);
264 
265          if((cp < 0.9) && (cp > boost::math::tools::min_value<value_type>()))
266          {
267             BOOST_CHECK_EX(quantile(dn, cp) == x);
268          }
269          if((ccp < 0.9) && (ccp > boost::math::tools::min_value<value_type>()))
270          {
271             BOOST_CHECK_EX(quantile(complement(dn, ccp)) == x);
272          }
273 #endif
274 #if !defined(TEST_QUANT) || (TEST_QUANT == 4)
275          boost::math::hypergeometric_distribution<value_type,
276             policy<discrete_quantile<integer_round_outwards> > > dou(r, n, N);
277 
278          if((cp < 0.9) && (cp > boost::math::tools::min_value<value_type>()))
279          {
280             if(cp < 0.5)
281             {
282                BOOST_CHECK_EX(quantile(dou, cp) <= x);
283             }
284             else
285             {
286                BOOST_CHECK_EX(quantile(dou, cp) >= x);
287             }
288          }
289          if((ccp < 0.9) && (ccp > boost::math::tools::min_value<value_type>()))
290          {
291             if(ccp < 0.5)
292             {
293                BOOST_CHECK_EX(quantile(complement(dou, ccp)) >= x);
294             }
295             else
296             {
297                BOOST_CHECK_EX(quantile(complement(dou, ccp)) <= x);
298             }
299          }
300 #endif
301 #if !defined(TEST_QUANT) || (TEST_QUANT == 5)
302          boost::math::hypergeometric_distribution<value_type,
303             policy<discrete_quantile<integer_round_inwards> > > di(r, n, N);
304 
305          if((cp < 0.9) && (cp > boost::math::tools::min_value<value_type>()))
306          {
307             if(cp < 0.5)
308             {
309                BOOST_CHECK_EX(quantile(di, cp) >= x);
310             }
311             else
312             {
313                BOOST_CHECK_EX(quantile(di, cp) <= x);
314             }
315          }
316          if((ccp < 0.9) && (ccp > boost::math::tools::min_value<value_type>()))
317          {
318             if(ccp < 0.5)
319             {
320                BOOST_CHECK_EX(quantile(complement(di, ccp)) <= x);
321             }
322             else
323             {
324                BOOST_CHECK_EX(quantile(complement(di, ccp)) >= x);
325             }
326          }
327 #endif
328       }
329    }
330 }
331 
332 
333 template <class RealType>
test_spot(unsigned x,unsigned n,unsigned r,unsigned N,RealType p,RealType cp,RealType ccp,RealType tol)334 void test_spot(unsigned x, unsigned n, unsigned r, unsigned N,
335                RealType p, RealType cp, RealType ccp, RealType tol)
336 {
337    //
338    // A bit of warning suppression:
339    //
340    (void)x;
341    (void)n;
342    (void)r;
343    (void)N;
344    (void)p;
345    (void)cp;
346    (void)ccp;
347    (void)tol;
348 #if !defined(TEST_QUANT) || (TEST_QUANT == 0)
349    boost::math::hypergeometric_distribution<RealType> d(r, n, N);
350 
351    std::pair<unsigned, unsigned> extent = range(d);
352    // CDF's:
353    BOOST_CHECK_CLOSE(pdf(d, x), p, tol);
354    BOOST_CHECK_CLOSE(cdf(d, x), cp, tol);
355    BOOST_CHECK_CLOSE(cdf(complement(d, x)), ccp, tol);
356    // Again with real-value arguments:
357    BOOST_CHECK_CLOSE(pdf(d, static_cast<RealType>(x)), p, tol);
358    BOOST_CHECK_CLOSE(cdf(d, static_cast<RealType>(x)), cp, tol);
359    BOOST_CHECK_CLOSE(cdf(complement(d, static_cast<RealType>(x))), ccp, tol);
360    //
361    // Quantiles, don't bother checking these for type float
362    // as there's not enough precision in the p and q values
363    // to get back to where we started:
364    //
365    if(boost::math::tools::digits<RealType>() > 50)
366    {
367       using namespace boost::math::policies;
368 
369       boost::math::hypergeometric_distribution<RealType,
370          policy<discrete_quantile<integer_round_up> > > du(r, n, N);
371 
372       BOOST_CHECK_EX(quantile(du, cp) >= x);
373       BOOST_CHECK_EX(quantile(complement(du, ccp)) >= x);
374 
375       boost::math::hypergeometric_distribution<RealType,
376          policy<discrete_quantile<integer_round_down> > > dl(r, n, N);
377       BOOST_CHECK_EX(quantile(dl, cp) <= x);
378       BOOST_CHECK_EX(quantile(complement(dl, ccp)) <= x);
379 
380       boost::math::hypergeometric_distribution<RealType,
381          policy<discrete_quantile<integer_round_nearest> > > dn(r, n, N);
382 
383       BOOST_CHECK_EX(quantile(dn, cp) == x);
384       BOOST_CHECK_EX(quantile(complement(dn, ccp)) == x);
385    }
386    //
387    // Error checking of out of bounds arguments:
388    //
389    BOOST_MATH_CHECK_THROW(pdf(d, extent.second + 1), std::domain_error);
390    BOOST_MATH_CHECK_THROW(cdf(d, extent.second + 1), std::domain_error);
391    BOOST_MATH_CHECK_THROW(cdf(complement(d, extent.second + 1)), std::domain_error);
392    if(extent.first > 0)
393    {
394       BOOST_MATH_CHECK_THROW(pdf(d, extent.first - 1), std::domain_error);
395       BOOST_MATH_CHECK_THROW(cdf(d, extent.first - 1), std::domain_error);
396       BOOST_MATH_CHECK_THROW(cdf(complement(d, extent.first - 1)), std::domain_error);
397    }
398    BOOST_MATH_CHECK_THROW(quantile(d, 1.1f), std::domain_error);
399    BOOST_MATH_CHECK_THROW(quantile(complement(d, 1.1f)), std::domain_error);
400    BOOST_MATH_CHECK_THROW(quantile(d, -0.001f), std::domain_error);
401    BOOST_MATH_CHECK_THROW(quantile(complement(d, -0.001f)), std::domain_error);
402    //
403    // Checking of extreme values:
404    //
405    BOOST_CHECK_EQUAL(quantile(d, 0), extent.first);
406    BOOST_CHECK_EQUAL(quantile(d, 1), extent.second);
407    BOOST_CHECK_EQUAL(quantile(complement(d, 0)), extent.second);
408    BOOST_CHECK_EQUAL(quantile(complement(d, 1)), extent.first);
409    BOOST_CHECK_EQUAL(cdf(d, extent.second), 1);
410    BOOST_CHECK_EQUAL(cdf(complement(d, extent.second)), 0);
411 #endif
412 }
413 
414 template <class RealType>
test_spots(RealType,const char * type_name)415 void test_spots(RealType /*T*/, const char* type_name)
416 {
417    // Basic sanity checks.
418    // 50 eps as a percentage, up to a maximum of double precision
419    // Test data taken from Mathematica 6
420 #define T RealType
421 #include "hypergeometric_test_data.ipp"
422    do_test_hypergeometric<T>(hypergeometric_test_data, type_name, "Mathematica data");
423 
424 #include "hypergeometric_dist_data2.ipp"
425    if(boost::is_floating_point<RealType>::value)
426    {
427       //
428       // Don't test this for real_concept: it's too slow!!!
429       //
430       do_test_hypergeometric<T>(hypergeometric_dist_data2, type_name, "Random large data");
431    }
432 
433    do_test_hypergeometric_quantile<T>(hypergeometric_test_data, type_name, "Mathematica data");
434    if(boost::is_floating_point<RealType>::value)
435    {
436       //
437       // Don't test this for real_concept: it's too slow!!!
438       //
439       do_test_hypergeometric_quantile<T>(hypergeometric_dist_data2, type_name, "Random large data");
440    }
441 
442    RealType tolerance = (std::max)(
443       static_cast<RealType>(2e-16L),    // limit of test data
444       boost::math::tools::epsilon<RealType>());
445    cout<<"Absolute tolerance:"<<tolerance<<endl;
446 
447    tolerance *= 50 * 100; // 50eps as a percentage
448    cout << "Tolerance for type " << typeid(RealType).name()  << " is " << tolerance << " %" << endl;
449 
450    //
451    // These sanity check values were calculated using the online
452    // calculator at http://stattrek.com/Tables/Hypergeometric.aspx
453    // It's assumed that the test values are accurate to no more than
454    // double precision.
455    //
456    test_spot(20, 200, 50, 500, static_cast<T>(0.120748236361163), static_cast<T>(0.563532430195156), static_cast<T>(1 - 0.563532430195156), tolerance);
457    test_spot(53, 452, 64, 500, static_cast<T>(0.0184749573044286), static_cast<T>(0.0299118078796907), static_cast<T>(1 - 0.0299118078796907), tolerance);
458    test_spot(32, 1287, 128, 5000, static_cast<T>(0.0807012167418264), static_cast<T>(0.469768774237742), static_cast<T>(1 - 0.469768774237742), tolerance);
459    test_spot(1, 13, 4, 26, static_cast<T>(0.248695652173913), static_cast<T>(0.296521739130435), static_cast<T>(1 - 0.296521739130435), tolerance);
460    test_spot(2, 13, 4, 26, static_cast<T>(0.40695652173913), static_cast<T>(0.703478260869565), static_cast<T>(1 - 0.703478260869565), tolerance);
461    test_spot(3, 13, 4, 26, static_cast<T>(0.248695652173913), static_cast<T>(0.952173913043478), static_cast<T>(1 - 0.952173913043478), tolerance);
462    test_spot(40, 70, 89, 170, static_cast<T>(0.0721901023798991), static_cast<T>(0.885447799131944), static_cast<T>(1 - 0.885447799131944), tolerance);
463 
464    boost::math::hypergeometric_distribution<RealType> d(50, 200, 500);
465    BOOST_CHECK_EQUAL(range(d).first, 0u);
466    BOOST_CHECK_EQUAL(range(d).second, 50u);
467    BOOST_CHECK_CLOSE(mean(d), static_cast<RealType>(20), tolerance);
468    BOOST_CHECK_CLOSE(mode(d), static_cast<RealType>(20), tolerance);
469    BOOST_CHECK_CLOSE(variance(d), static_cast<RealType>(10.821643286573146292585170340681L), tolerance);
470    BOOST_CHECK_CLOSE(skewness(d), static_cast<RealType>(0.048833071022952084732902910189366L), tolerance);
471    BOOST_CHECK_CLOSE(kurtosis_excess(d), static_cast<RealType>(2.5155486690782804816404001878293L), tolerance);
472    BOOST_CHECK_CLOSE(kurtosis(d), kurtosis_excess(d) + 3, tolerance);
473    BOOST_CHECK_EQUAL(quantile(d, 0.5f), median(d));
474 
475    BOOST_MATH_CHECK_THROW(d = boost::math::hypergeometric_distribution<RealType>(501, 40, 500), std::domain_error);
476    BOOST_MATH_CHECK_THROW(d = boost::math::hypergeometric_distribution<RealType>(40, 501, 500), std::domain_error);
477 }
478 
479 
BOOST_AUTO_TEST_CASE(test_main)480 BOOST_AUTO_TEST_CASE( test_main )
481 {
482    expected_results();
483    // Basic sanity-check spot values.
484    // (Parameter value, arbitrarily zero, only communicates the floating point type).
485    test_spots(0.0F, "float"); // Test float. OK at decdigits = 0 tolerance = 0.0001 %
486    test_spots(0.0, "double"); // Test double. OK at decdigits 7, tolerance = 1e07 %
487 #ifndef BOOST_MATH_NO_LONG_DOUBLE_MATH_FUNCTIONS
488    test_spots(0.0L, "long double"); // Test long double.
489 #ifndef BOOST_MATH_NO_REAL_CONCEPT_TESTS
490    test_spots(boost::math::concepts::real_concept(0), "real_concept"); // Test real_concept.
491 #endif
492 #else
493    std::cout << "<note>The long double tests have been disabled on this platform "
494       "either because the long double overloads of the usual math functions are "
495       "not available at all, or because they are too inaccurate for these tests "
496       "to pass.</note>" << std::endl;
497 #endif
498 
499 
500 } // BOOST_AUTO_TEST_CASE( test_main )
501 
502