;;; Excercise the `resolve' routines.                  -*- Scheme -*-
;;;
;;; Copyright (C) 2009  Ludovic Courtès <ludo@gnu.org>
;;;
;;; This file is part of Skribilo.
;;;
;;; Skribilo is free software; you can redistribute it and/or modify it
;;; under the terms of the GNU Lesser General Public License as published by
;;; the Free Software Foundation; either version 3 of the License, or (at
;;; your option) any later version.
;;;
;;; Skribilo is distributed in the hope that it will be useful, but
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser
;;; General Public License for more details.
;;;
;;; You should have received a copy of the GNU Lesser General Public License
;;; along with this program.  If not, see <http://www.gnu.org/licenses/>.

(define-module (tests resolve)
  #:use-module (skribilo ast)
  #:use-module (skribilo engine)
  #:use-module (skribilo resolve)
  #:use-module (skribilo package base)
  #:use-module (srfi srfi-64))

(define %engine
  (find-engine 'base))


(test-begin "resolve")

(test-assert "parents"
  (let* ((doc (document #:title "Doc"
                (chapter #:title "C"
                  (section #:title "S"
                    (p "par")))))
         (ch  (car (markup-body doc)))
         (sec (car (markup-body ch)))
         (par (car (markup-body sec))))
    (and (is-markup? ch 'chapter)
         (eq? (ast-parent ch) 'unspecified)
         (begin
           (resolve! doc %engine '())
           (and (eq? (ast-parent ch) doc)
                (eq? (ast-parent sec) ch)
                (eq? (ast-parent par) sec))))))

(test-assert "root document has no parent"
  (let ((doc (document #:title "Doc")))
    (resolve! doc %engine '())
    (and (not (ast-parent doc))
         (eq? doc (ast-document doc)))))

(test-assert "nested document has a parent"
  ;; Nested documents are sometimes used in the manual.
  (let* ((doc (document #:title "Doc"
                (document #:title "Nested Doc"
                  (chapter #:title "C"))))
         (sub (car (markup-body doc)))
         (ch  (car (markup-body sub))))
    (resolve! doc %engine '())
    (and (not (ast-parent doc))
         (document? sub)
         (eq? doc (ast-document doc))
         (eq? doc (ast-parent sub)))))

(test-assert "nested document is its own `ast-document'"
  (let* ((doc (document #:title "Doc"
                (document #:title "Nested Doc"
                  (chapter #:title "C"))))
         (sub (car (markup-body doc)))
         (ch  (car (markup-body sub))))
    (resolve! doc %engine '())
    (and (document? sub)
         (eq? sub (ast-document sub))
         (eq? sub (ast-document ch)))))

(test-assert "unresolved node in body"
  (let* ((resolved? #f)
         (doc (document #:title "Doc"
                (resolve (lambda (n e env)
                           (set! resolved? #t))))))
    (and (not resolved?)
         (begin
           (resolve! doc %engine '())
           resolved?))))

(test-assert "unresolved node in unresolved node in body"
  (let* ((resolved? #f)
         (doc (document #:title "Doc"
                (resolve (lambda (n e env)
                           (resolve (lambda (n e env)
                                      (set! resolved? #t))))))))
    (and (not resolved?)
         (begin
           (resolve! doc %engine '())
           resolved?))))

(test-assert "unresolved node in options"
  (let* ((resolved? #f)
         (doc (document #:title
                (resolve (lambda (n e env)
                           (set! resolved? #t)))
                "body...")))
    (and (not resolved?)
         (begin
           (resolve! doc %engine '())
           resolved?))))

(test-assert "unresolved node in processor body"
  (let* ((resolved? #f)
         (doc (document #:title "Doc"
                (processor #:combinator (lambda (e1 e2) e1)
                           (resolve (lambda (n e env)
                                      (set! resolved? #t)))))))
    (and (not resolved?)
         (begin
           (resolve! doc %engine '())
           resolved?))))

(test-assert "unresolved node in nested document"
  (let* ((resolved? #f)
         (doc (document #:title "Outer"
                (document #:title "Inner"
                  (resolve (lambda (n e env)
                             (set! resolved? #t)))))))
    (and (not resolved?)
         (begin
           (resolve! doc %engine '())
           resolved?))))

(test-assert "unresolved node product has a parent"
  (let* ((doc (document #:title "Doc"
                (resolve (lambda (n e env)
                           (chapter #:title "C"))))))
    (resolve! doc %engine '())

    (let ((ch (car (markup-body doc))))
      (and (is-markup? ch 'chapter)
           (eq? (ast-parent ch) doc)))))

(test-assert "node bindings"
  (let* ((doc (document #:title "Doc"
                (chapter #:title "C" #:ident "c"
                  (section #:title "S" #:ident "s"))))
         (ch  (car (markup-body doc)))
         (sec (car (markup-body ch))))
    (and (not (document-lookup-node doc "c"))
         (not (document-lookup-node doc "s"))
         (begin
           (resolve! doc %engine '())
           (and (eq? (document-lookup-node doc "c") ch)
                (eq? (document-lookup-node doc "s") sec))))))

(test-assert "unresolved node bindings"
  ;; Make sure nodes returned by `unresolved' nodes are eventually bound.
  (let* ((doc (document #:title "Doc"
                (resolve (lambda (n e env)
                           (chapter #:title "C" #:ident "c"
                             (section #:title "S" #:ident "s")))))))
    (and (not (document-lookup-node doc "c"))
         (not (document-lookup-node doc "s"))
         (begin
           (resolve! doc %engine '())
           (and (is-markup? (document-lookup-node doc "c") 'chapter)
                (is-markup? (document-lookup-node doc "s") 'section))))))

(test-assert "nested document bindings"
  ;; Bindings in nested documents are scoped.  This was not the case prior
  ;; to 0.9.2.
  (let* ((doc (document #:title "Doc"
                (chapter #:ident "outer")
                (document #:title "Nested Doc"
                  (chapter #:ident "inner"))))
         (out (car  (markup-body doc)))
         (sub (cadr (markup-body doc)))
         (in  (car  (markup-body sub))))
    (resolve! doc %engine '())
    (and (let ((x (document-lookup-node doc "outer")))
           (and (is-markup? x 'chapter)
                (eq? (ast-document x) doc)))
         (not (document-lookup-node doc "inner"))
         (let ((x (document-lookup-node sub "inner")))
           (and (is-markup? x 'chapter)
                (eq? (ast-document x) sub))))))

(test-assert "resolved nested document bindings"
  ;; Prior to 0.9.2, nodes of a sub-document returned by an `unresolved' node
  ;; would be bound in the root document.
  (let* ((doc (document #:title "Doc"
                (resolve (lambda (n e env)
                           (document #:title "Nested Doc"
                             (chapter #:ident "inner")))))))
    (resolve! doc %engine '())
    (and (not (document-lookup-node doc "inner"))
         (let* ((sub (car (markup-body doc)))
                (ch  (car (markup-body sub)))
                (x   (document-lookup-node sub "inner")))
           (and (document? sub)
                (is-markup? x 'chapter)
                (eq? x ch)
                (eq? (ast-parent x) sub)
                (eq? (ast-document x) sub))))))

(test-assert "node bindings in processor body"
  (let* ((doc (document #:title "Doc"
                (processor #:combinator (lambda (e1 e2) e1)
                  (chapter #:ident "c")))))
    (resolve! doc %engine '())
    (let* ((proc (car (markup-body doc)))
           (ch   (car (markup-body proc)))
           (ch*  (document-lookup-node doc "c")))
      (eq? ch ch*))))

(test-end "resolve")


(exit (= (test-runner-fail-count (test-runner-current)) 0))