Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Racket test for Exception

I am taking my first steps with exception handling in Racket and would like to write a unit test, which checks, if a procedure raises an exception for a specific input.

The following is what I already have:

The procedure, which shall raise the exception:

(define (div-interval x y)
  ;; EXERCISE 2.10
  ; exception handling
  (with-handlers
    ([exn:fail:contract:divide-by-zero?
      (lambda (exn)
        (displayln (exn-message exn))
        #f)])

    (if (or
      (>= (upper-bound y) 0)
      (<= (lower-bound y) 0))

      (raise
        (make-exn:fail:contract:divide-by-zero
        "possible division by zero, interval spans zero"
        (current-continuation-marks)))

      (mul-interval
        x
        (make-interval
          (/ 1.0 (upper-bound y))
          (/ 1.0 (lower-bound y)))))))

The unit test:

(require rackunit)

(define exercise-test
  (test-suite
    "exercise test suite"

    #:before (lambda () (begin (display "before")(newline)))
    #:after (lambda () (begin (display "after")(newline)))

    (test-case
      "checking interval including zero"
      (check-exn
        exn:fail:contract:divide-by-zero?
        (div-interval
          (make-interval 1.0 2.0)
          (make-interval -3.0 2.0))
        "Exception: interval spans zero, possible division by zero")))

(run-test exercise-test)

There are some more tests in this test suite, but they are for other procedures, so I did not include them in this code. When I run my program, I get the following output:

before
possible division by zero, interval spans zero
after
'(#<test-success> #<test-success> #<test-success> #<test-success> #<test-error>)

Where the <test-error> is for the test case in this post.

It seems the procedure does not raise the exception.

Is this, because I have a handler in my procedure, which returns #f and thus already "ate" the exception?

How would I usually write a unit test for raised exceptions?

like image 984
Zelphir Kaltstahl Avatar asked Sep 30 '16 13:09

Zelphir Kaltstahl


1 Answers

You are absolutely right that you the code did not raise the exception because the with-handler caught it and returned #f. If you want the with-handler to re-raise the exception for you you need to use raise, making your code look something like:

(define (div-interval x y)
  (with-handlers
    ([exn:fail:contract:divide-by-zeor?
      (lambda (e)
        <do-stuff>
        (raise e))])
    <function-body>)

Now your handler will still run, but at the end it will re-raise the exception for you.

As for testing it, you are absolutely right that check-exn is the right way to go about it. Except because check-exn is a procedure, you need to wrap your code in a thunk making it look like:

(require rackunit)
(check-exn
  exn:fail:contract:divide-by-zero?
  (lambda ()
    (div-interval
      (make-interval 1.0 2.0)
      (make-interval -3.0 2.0))))
like image 52
Leif Andersen Avatar answered Oct 21 '22 09:10

Leif Andersen