Acme-ExtUtils-XSOne-Test-Calculator
view release on metacpan or search on metacpan
t/01-calculator.t view on Meta::CPAN
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::absolute(-7), 7, 'absolute(-7) = 7');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::absolute(4), 4, 'absolute(4) = 4');
# Test C helper functions (defined in basic.xs preamble)
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::safe_divide(10, 2), 5, 'safe_divide(10, 2) = 5');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::safe_divide(10, 0), 0, 'safe_divide(10, 0) = 0 (no croak)');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::clamp(5, 0, 10), 5, 'clamp(5, 0, 10) = 5');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::clamp(15, 0, 10), 10, 'clamp(15, 0, 10) = 10');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::clamp(-5, 0, 10), 0, 'clamp(-5, 0, 10) = 0');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::percent(200, 15), 30, 'percent(200, 15) = 30');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::percent(50, 50), 25, 'percent(50, 50) = 25');
is(Acme::ExtUtils::XSOne::Test::Calculator::Basic::percent(100, 100), 100, 'percent(100, 100) = 100');
};
subtest 'Division by zero' => sub {
plan tests => 2;
eval { Acme::ExtUtils::XSOne::Test::Calculator::Basic::divide(5, 0) };
like($@, qr/Division by zero/, 'divide by zero croaks');
eval { Acme::ExtUtils::XSOne::Test::Calculator::Basic::modulo(5, 0) };
like($@, qr/Modulo by zero/, 'modulo by zero croaks');
};
# =============================================================================
# Test Scientific operations
# =============================================================================
subtest 'Scientific functions' => sub {
plan tests => 14;
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::power(2, 10), 1024, 'power(2, 10) = 1024');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::power(5, 0), 1, 'power(5, 0) = 1');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::sqrt_val(16), 4, 'sqrt(16) = 4');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::sqrt_val(2), sqrt(2), 'sqrt(2) matches perl sqrt');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::cbrt_val(27), 3, 'cbrt(27) = 3');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::cbrt_val(-8), -2, 'cbrt(-8) = -2');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::exp_val(1) - exp(1)) < 0.0001, 'exp(1) = e');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::factorial(5), 120, 'factorial(5) = 120');
# Test C helper functions (defined in scientific.xs preamble)
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::ipow(2, 10), 1024, 'ipow(2, 10) = 1024');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::ipow(3, -2), 1/9, 'ipow(3, -2) = 1/9');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::safe_sqrt(16), 4, 'safe_sqrt(16) = 4');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::safe_sqrt(-1), 0, 'safe_sqrt(-1) = 0 (no croak)');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::combination(5, 2), 10, 'combination(5, 2) = 10');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::permutation(5, 2), 20, 'permutation(5, 2) = 20');
};
subtest 'Logarithms' => sub {
plan tests => 6;
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::log_natural(exp(1)) - 1) < 0.0001, 'ln(e) = 1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::log10_val(100) - 2) < 0.0001, 'log10(100) = 2');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::log_base(8, 2) - 3) < 0.0001, 'log2(8) = 3');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::nth_root(27, 3), 3, 'nth_root(27, 3) = 3');
# Test C helper function (defined in scientific.xs preamble)
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::safe_log(exp(1)) - 1) < 0.0001, 'safe_log(e) = 1');
is(Acme::ExtUtils::XSOne::Test::Calculator::Scientific::safe_log(-1), 0, 'safe_log(-1) = 0 (no croak)');
};
subtest 'Scientific error handling' => sub {
plan tests => 3;
eval { Acme::ExtUtils::XSOne::Test::Calculator::Scientific::sqrt_val(-1) };
like($@, qr/negative/, 'sqrt of negative croaks');
eval { Acme::ExtUtils::XSOne::Test::Calculator::Scientific::log_natural(-1) };
like($@, qr/non-positive/, 'log of negative croaks');
eval { Acme::ExtUtils::XSOne::Test::Calculator::Scientific::factorial(-1) };
like($@, qr/negative/, 'factorial of negative croaks');
};
# =============================================================================
# Test Trig operations
# =============================================================================
subtest 'Trigonometric functions' => sub {
plan tests => 14;
my $pi = Acme::ExtUtils::XSOne::Test::Calculator::pi();
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::sin_val(0)) < 0.0001, 'sin(0) = 0');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::sin_val($pi/2) - 1) < 0.0001, 'sin(pi/2) = 1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::cos_val(0) - 1) < 0.0001, 'cos(0) = 1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::cos_val($pi) - (-1)) < 0.0001, 'cos(pi) = -1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::tan_val(0)) < 0.0001, 'tan(0) = 0');
is(Acme::ExtUtils::XSOne::Test::Calculator::Trig::hypot_val(3, 4), 5, 'hypot(3, 4) = 5');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::deg_to_rad(180) - $pi) < 0.0001, 'deg_to_rad(180) = pi');
# Test C helper functions (defined in trig.xs preamble)
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::normalize_angle(3 * $pi) - $pi) < 0.0001, 'normalize_angle(3*pi) = pi');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::normalize_angle(-3 * $pi) - (-$pi)) < 0.0001, 'normalize_angle(-3*pi) = -pi');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::sec_val(0) - 1) < 0.0001, 'sec(0) = 1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::csc_val($pi/2) - 1) < 0.0001, 'csc(pi/2) = 1');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::cot_val($pi/4) - 1) < 0.0001, 'cot(pi/4) = 1');
is(Acme::ExtUtils::XSOne::Test::Calculator::Trig::is_valid_asin_arg(0.5), 1, 'is_valid_asin_arg(0.5) = 1');
is(Acme::ExtUtils::XSOne::Test::Calculator::Trig::is_valid_asin_arg(2.0), 0, 'is_valid_asin_arg(2.0) = 0');
};
subtest 'Inverse trig' => sub {
plan tests => 3;
my $pi = Acme::ExtUtils::XSOne::Test::Calculator::pi();
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::asin_val(1) - $pi/2) < 0.0001, 'asin(1) = pi/2');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::acos_val(0) - $pi/2) < 0.0001, 'acos(0) = pi/2');
ok(abs(Acme::ExtUtils::XSOne::Test::Calculator::Trig::atan_val(1) - $pi/4) < 0.0001, 'atan(1) = pi/4');
};
# =============================================================================
# Test Memory - THIS IS THE KEY TEST FOR SHARED STATE!
# =============================================================================
subtest 'Memory operations' => sub {
plan tests => 13;
Acme::ExtUtils::XSOne::Test::Calculator::Memory::clear();
ok(Acme::ExtUtils::XSOne::Test::Calculator::Memory::store(0, 42), 'store(0, 42) succeeds');
is(Acme::ExtUtils::XSOne::Test::Calculator::Memory::recall(0), 42, 'recall(0) = 42');
( run in 0.538 second using v1.01-cache-2.11-cpan-cdf2f3d4e48 )