(define-library (srfi 121 test)
  (export run-tests)
  (import (scheme base) (scheme read) (srfi 1) (srfi 121) (chibi test))
  (begin
    (define (run-tests)
      (test-begin "srfi-121: generators")

      (test-group "generators/constructors"
        (define (for-each-digit proc n)
          (when (> n 0)
            (let-values (((div rem) (truncate/ n 10)))
              (proc rem)
              (for-each-digit proc div))))
        (test '() (generator->list (generator)))
        (test '(1 2 3) (generator->list (generator 1 2 3)))
        (test '(8 9 10) (generator->list (make-iota-generator 3 8)))
        (test '(8 10 12) (generator->list (make-iota-generator 3 8 2)))
        (test '(3 4 5 6 7) (generator->list (make-range-generator 3 8)))
        (test '(3 4 5 6) (generator->list (make-range-generator 3) 4))
        (test '(3 5 7) (generator->list (make-range-generator 3 8 2)))
        (test '(0 1 2)
            (generator->list
             (make-coroutine-generator
              (lambda (yield)
                (let lp ((i 0))
                  (when (< i 3) (yield i) (lp (+ i 1))))))))
        (test '(1 2 3 4 5) (generator->list (list->generator '(1 2 3 4 5))))
        (test '(1 2 3 4 5) (generator->list (vector->generator '#(1 2 3 4 5))))
        (test '(5 4 3 2 1)
            (generator->list (reverse-vector->generator '#(1 2 3 4 5))))
        (test '(#\a #\b #\c #\d #\e)
            (generator->list (string->generator "abcde")))
        (test '(10 20 30)
            (generator->list (bytevector->generator (bytevector 10 20 30))))
        (test '(5 4 3 2 1)
             (generator->list
              (make-for-each-generator for-each-digit 12345)))
        (test '(0 2 4 6 8 10) (generator->list
                               (make-unfold-generator
                                (lambda (s) (> s 5))
                                (lambda (s) (* s 2))
                                (lambda (s) (+ s 1))
                                0)))
        )

      (test-group "generators/operators"
        (define (small? x) (< x 3))
        (test '(a b 0 1)
            (generator->list (gcons* 'a 'b (make-range-generator 0 2))))
        (test '(0 1 2 0 1)
            (generator->list (gappend (make-range-generator 0 3)
                                      (make-range-generator 0 2))))
        (test '() (generator->list (gappend)))
        (test '(15 22 31)
            (generator->list
             (gcombine (lambda args (values (apply + args) (apply + args)))
                       10
                       (generator 1 2 3)
                       (generator 4 5 6 7))))
        (test '(1 3 5 7 9) (generator->list (gfilter
                                             odd?
                                             (make-range-generator 1 11))))
        (test '(2 4 6 8 10) (generator->list (gremove
                                              odd?
                                              (make-range-generator 1 11))))
        (let ((g (make-range-generator 1 5)))
          (test '(1 2 3) (generator->list (gtake g 3)))
          (test '(4) (generator->list g)))
        (test '(1 2) (generator->list (gtake (make-range-generator 1 3) 3)))
        (test '(1 2 0) (generator->list (gtake (make-range-generator 1 3) 3 0)))
        (test '(3 4) (generator->list (gdrop (make-range-generator 1 5) 2)))
        (test '(1 2)
            (generator->list (gtake-while small? (make-range-generator 1 5))))
        (test '(3 4)
            (generator->list (gdrop-while small? (make-range-generator 1 5))))
        (test '() (generator->list (gdrop-while (lambda args #t) (generator 1 2 3))))
        (test '(0.0 1.0 0 2)
            (generator->list (gdelete 1 (generator 0.0 1.0 0 1 2))))
        (test '(0.0 0 2)
            (generator->list (gdelete 1 (generator 0.0 1.0 0 1 2) =)))
        (test '(a c e)
            (generator->list (gindex (list->generator '(a b c d e f))
                                     (list->generator '(0 2 4)))))
        (test '(a d e)
            (generator->list (gselect (list->generator '(a b c d e f))
                                      (list->generator '(#t #f #f #t #t #f)))))
        (test '(1 2 3)
            (generator->list (gdelete-neighbor-dups
                              (generator 1 1 2 3 3 3)
                              =)))
        (test '(1)
            (generator->list (gdelete-neighbor-dups
                              (generator 1 2 3)
                              (lambda args #t))))
        )

      (test-group "generators/consumers"
        (test '(1 2 3) (generator->list (generator 1 2 3 4 5) 3))
        (test '(5 4 3 2 1) (generator->reverse-list (generator 1 2 3 4 5)))
        (test '#(1 2 3 4 5) (generator->vector (generator 1 2 3 4 5)))
        (test '#(1 2 3) (generator->vector (generator 1 2 3 4 5) 3))
        (test "abc" (generator->string (generator #\a #\b #\c)))
        (test '(e d c b a . z)
            (let ((in (open-input-string "a b c d e")))
              (generator-fold cons 'z (lambda () (read in)))))
        (let ((n 0))
          (generator-for-each (lambda values (set! n (apply + values)))
                              (generator 1) (generator 2) (generator 3))
          (test 6 n))
        (test 3 (generator-find (lambda (x) (> x 2)) (make-range-generator 1 5)))
        (test 2 (generator-count odd? (make-range-generator 1 5)))
        (let ((g (make-range-generator 2 5)))
          (test #t (generator-any odd? g))
          (test '(4) (generator->list g)))
        (let ((g (make-range-generator 2 5)))
          (test #f (generator-every odd? g))
          (test '(3 4) (generator->list g)))
        (test '(#\a #\b #\c)
             (generator-unfold (make-for-each-generator string-for-each "abc")
                               unfold))
        )

      (test-end))))