port of SWI-Prolog's plunit.
This commit is contained in:
		| @@ -471,6 +471,7 @@ all: startup.yss | ||||
| 	@INSTALL_DLLS@ (cd library/lammpi; $(MAKE)) | ||||
| 	@INSTALL_DLLS@ (cd library/matrix; $(MAKE)) | ||||
| 	@INSTALL_DLLS@ (cd packages/sgml; $(MAKE)) | ||||
| 	@INSTALL_DLLS@ (cd packages/plunit; $(MAKE)) | ||||
| 	@INSTALL_DLLS@ (cd packages/swi-minisat2/C; $(MAKE)) | ||||
| 	@INSTALL_MATLAB@ (cd library/matlab; $(MAKE)) | ||||
| 	@ENABLE_JPL@ @INSTALL_DLLS@ (cd packages/jpl; $(MAKE)) | ||||
| @@ -506,6 +507,7 @@ install_unix: startup.yss libYap.a | ||||
| 	mkdir -p $(DESTDIR)$(SHAREDIR)/Yap/pl | ||||
| 	for f in $(PL_SOURCES); do $(INSTALL) $$f $(DESTDIR)$(SHAREDIR)/Yap/pl; done | ||||
| 	@INSTALL_DLLS@ (cd packages/PLStream; $(MAKE) install) | ||||
| 	@INSTALL_DLLS@ (cd packages/plunit; $(MAKE) install) | ||||
| 	@INSTALL_DLLS@ (cd packages/sgml; $(MAKE) install) | ||||
| 	@INSTALL_DLLS@ (cd packages/swi-minisat2/C; $(MAKE) install) | ||||
| 	@INSTALL_DLLS@ (cd library/random; $(MAKE) install) | ||||
| @@ -544,6 +546,7 @@ install_win32: startup.yss | ||||
| 	$(INSTALL) config.h $(DESTDIR)$(INCLUDEDIR)/config.h | ||||
| 	$(INSTALL) parms.h $(DESTDIR)$(INCLUDEDIR)/parms.h | ||||
| 	(cd packages/PLStream; $(MAKE) install) | ||||
| 	(cd packages/plunit; $(MAKE) install) | ||||
| 	(cd packages/sgml; $(MAKE) install) | ||||
| 	(cd packages/swi-minisat2/C; $(MAKE) install) | ||||
| 	(cd library/random; $(MAKE) install) | ||||
| @@ -594,6 +597,7 @@ depend: $(HEADERS) $(C_SOURCES) | ||||
| clean: clean_docs | ||||
| 	rm -f *.o *~ *.BAK *.a | ||||
| 	@INSTALL_DLLS@ (cd packages/PLStream; $(MAKE) clean) | ||||
| 	@INSTALL_DLLS@ (cd packages/plunit; $(MAKE) clean) | ||||
| 	@INSTALL_DLLS@ (cd library/matrix; $(MAKE) clean) | ||||
| 	@INSTALL_DLLS@ (cd library/random; $(MAKE) clean) | ||||
| 	@INSTALL_DLLS@ (cd library/regex; $(MAKE) clean) | ||||
|   | ||||
							
								
								
									
										14
									
								
								configure
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								configure
									
									
									
									
										vendored
									
									
								
							| @@ -17660,6 +17660,7 @@ mkdir -p  packages/jpl/src/java/jpl/test | ||||
| mkdir -p  packages/jpl/examples | ||||
| mkdir -p  packages/jpl/examples/java | ||||
| mkdir -p  packages/PLStream | ||||
| mkdir -p  packages/plunit | ||||
| mkdir -p  packages/ProbLog | ||||
| mkdir -p  packages/ProbLog/simplecudd | ||||
| mkdir -p  packages/sgml | ||||
| @@ -17700,15 +17701,17 @@ ac_config_files="$ac_config_files packages/CLPBN/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/cplint/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/sgml/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/PLStream/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/plunit/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/ProbLog/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/swi-minisat2/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/swi-minisat2/C/Makefile" | ||||
|  | ||||
| ac_config_files="$ac_config_files packages/ProbLog/Makefile" | ||||
| ac_config_files="$ac_config_files packages/sgml/Makefile" | ||||
|  | ||||
|  | ||||
| if test "$ENABLE_CHR" = ""; then | ||||
| @@ -18345,11 +18348,12 @@ do | ||||
|     "LGPL/swi_console/Makefile") CONFIG_FILES="$CONFIG_FILES LGPL/swi_console/Makefile" ;; | ||||
|     "packages/CLPBN/Makefile") CONFIG_FILES="$CONFIG_FILES packages/CLPBN/Makefile" ;; | ||||
|     "packages/cplint/Makefile") CONFIG_FILES="$CONFIG_FILES packages/cplint/Makefile" ;; | ||||
|     "packages/sgml/Makefile") CONFIG_FILES="$CONFIG_FILES packages/sgml/Makefile" ;; | ||||
|     "packages/PLStream/Makefile") CONFIG_FILES="$CONFIG_FILES packages/PLStream/Makefile" ;; | ||||
|     "packages/plunit/Makefile") CONFIG_FILES="$CONFIG_FILES packages/plunit/Makefile" ;; | ||||
|     "packages/ProbLog/Makefile") CONFIG_FILES="$CONFIG_FILES packages/ProbLog/Makefile" ;; | ||||
|     "packages/swi-minisat2/Makefile") CONFIG_FILES="$CONFIG_FILES packages/swi-minisat2/Makefile" ;; | ||||
|     "packages/swi-minisat2/C/Makefile") CONFIG_FILES="$CONFIG_FILES packages/swi-minisat2/C/Makefile" ;; | ||||
|     "packages/ProbLog/Makefile") CONFIG_FILES="$CONFIG_FILES packages/ProbLog/Makefile" ;; | ||||
|     "packages/sgml/Makefile") CONFIG_FILES="$CONFIG_FILES packages/sgml/Makefile" ;; | ||||
|     "packages/chr/Makefile") CONFIG_FILES="$CONFIG_FILES packages/chr/Makefile" ;; | ||||
|     "packages/clpqr/Makefile") CONFIG_FILES="$CONFIG_FILES packages/clpqr/Makefile" ;; | ||||
|     "packages/jpl/Makefile") CONFIG_FILES="$CONFIG_FILES packages/jpl/Makefile" ;; | ||||
|   | ||||
| @@ -1632,6 +1632,7 @@ mkdir -p  packages/jpl/src/java/jpl/test | ||||
| mkdir -p  packages/jpl/examples | ||||
| mkdir -p  packages/jpl/examples/java | ||||
| mkdir -p  packages/PLStream | ||||
| mkdir -p  packages/plunit | ||||
| mkdir -p  packages/ProbLog | ||||
| mkdir -p  packages/ProbLog/simplecudd | ||||
| mkdir -p  packages/sgml | ||||
| @@ -1654,11 +1655,12 @@ AC_CONFIG_FILES([LGPL/clp/Makefile]) | ||||
| AC_CONFIG_FILES([LGPL/swi_console/Makefile]) | ||||
| AC_CONFIG_FILES([packages/CLPBN/Makefile]) | ||||
| AC_CONFIG_FILES([packages/cplint/Makefile]) | ||||
| AC_CONFIG_FILES([packages/sgml/Makefile]) | ||||
| AC_CONFIG_FILES([packages/PLStream/Makefile]) | ||||
| AC_CONFIG_FILES([packages/plunit/Makefile]) | ||||
| AC_CONFIG_FILES([packages/ProbLog/Makefile ]) | ||||
| AC_CONFIG_FILES([packages/swi-minisat2/Makefile]) | ||||
| AC_CONFIG_FILES([packages/swi-minisat2/C/Makefile]) | ||||
| AC_CONFIG_FILES([packages/ProbLog/Makefile ]) | ||||
| AC_CONFIG_FILES([packages/sgml/Makefile]) | ||||
|  | ||||
| if test "$ENABLE_CHR" = ""; then | ||||
| AC_CONFIG_FILES([packages/chr/Makefile]) | ||||
|   | ||||
							
								
								
									
										172
									
								
								packages/plunit/ChangeLog
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								packages/plunit/ChangeLog
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | ||||
| [Jul 29 2009] | ||||
|  | ||||
|  * ENHANCED: System and library modules now use =system= as their parent module rather than =user=. | ||||
| [Jun 15 2009] | ||||
|  | ||||
|  * DOC: Item#483: Broken link in PlUnit.  Ulrich Neumerkel. | ||||
|     | ||||
| [May 14 2009] | ||||
|  | ||||
|  * DOC: module specifier, typos | ||||
|     | ||||
| [Mar 19 2009] | ||||
|  | ||||
|  * CLEANUP: Removed all trailing whitespace from all source-files.  This avoids many GIT (merge) warnings. | ||||
| [Mar 12 2009] | ||||
|  | ||||
|  * MODIFIED: Renamed concat_atom/2,3 into atomic_list_concat/2,3 for compatibility.  Moved the old predicates into library(backcomp).  It is adviced to update your code, but 99% of the code will remain working because the old definitions are autoloaded. | ||||
| [Feb 11 2009] | ||||
|  | ||||
|  * CLEANUP: Module declaration code | ||||
| [Sep 11 2008] | ||||
|  | ||||
|  * PORT: Add AC_PREREQ to configure.h for systems that auto-select autoconf | ||||
|    versions.  After tip by Ulrich Neumerkel. | ||||
|     | ||||
| [Aug 11 2008] | ||||
|  | ||||
|  * INSTALL: Remove all configure files from the git repository | ||||
|     | ||||
| [Jul 10 2008] | ||||
|  | ||||
|  * FIXED: PlUnit: Perform goal_expansion/2 on setup, condition and cleanup options. | ||||
|    Ivo Swartjes. | ||||
|     | ||||
| [Jun 26 2008] | ||||
|  | ||||
|  * ADDED: forall option to unit tests that allows running the same test on | ||||
|    different data sets. | ||||
|     | ||||
|  * FIXED: wrong handling of gloabl option sto(false) | ||||
| [May 31 2008] | ||||
|  | ||||
|  * FIXED: Properly support tests with condition and setup | ||||
|  | ||||
| [May 14 2008] | ||||
|  | ||||
|  * ENHANCED: PlUnit: Test options in set_test_options/1; fix documentation. | ||||
|  | ||||
| [May 9 2008] | ||||
|  | ||||
|  * ADDED: Allow specifying sto mode for an entire test set | ||||
|  | ||||
| [May 1 2008] | ||||
|  | ||||
|  * DOC: plunit and non termination | ||||
|  | ||||
| [Apr 28 2008] | ||||
|  | ||||
|  * ADDED: flag tests using `fixme'.  Suggested under the name `desired' by | ||||
|    Ulrich Neumerkel.  Let us see how this works out in practice! | ||||
|     | ||||
| [Apr 24 2008] | ||||
|  | ||||
|  * ENHANCED: When interrupting a thread, indicate which one. | ||||
|  | ||||
|  * ADDED: PlUnit: running_tests/0 to find what test is running. | ||||
|  | ||||
|  * FIXED: wrong dynamic declaration in plunit.pl | ||||
|  | ||||
| [Mar 11 2008] | ||||
|  | ||||
|  * ENHANCED: Provide file/line info for errors in setup and condition | ||||
|  | ||||
| [Feb 4 2008] | ||||
|  | ||||
|  * FIXED: STO independent of timings, Item#316 | ||||
|  | ||||
| [Jan 31 2008] | ||||
|  | ||||
|  * DOC: formatting | ||||
|  | ||||
| [Jan 30 2008] | ||||
|  | ||||
|  * ENHANCED: PlUnit: run_tests(Unit) raise error on non-existing unit. | ||||
|    Item#305, Ulrich Neumerkel. | ||||
|     | ||||
|  * ENHANCED: PlUnit: reporting STO and setting STO checking options | ||||
|  | ||||
| [Jan 29 2008] | ||||
|  | ||||
|  * CLEANUP: PlUnit: some reorganisation of the test loop, preparing for further | ||||
|    enhancements. | ||||
|     | ||||
| [Jan 28 2008] | ||||
|  | ||||
|  * ADDED: subsumes/2 and subsumes_chk/2 as built-in predicates. | ||||
|  | ||||
|  * ENHANCED: Really load no tests if not asked to do and provide better feedback | ||||
|    if no tests where loaded or all tests passed. | ||||
|     | ||||
|  * FIXED: DCG: cut transparency, output unification, preserve erroneous non-terminals | ||||
|  * ENHANCED: DCG: more tests | ||||
|  * FIXED: PlUnit version confusion | ||||
|     | ||||
| [2008-01-26] | ||||
|  | ||||
|  * FIXED: portability for SICStus 3.12 | ||||
|  * ENHANCED: new option sto | ||||
|  | ||||
| [Jan 14 2008] | ||||
|  | ||||
|  * FIXED: Non-determinism in run_tests/0 if there are blocked tests. | ||||
|  | ||||
| [Dec 21 2007] | ||||
|  | ||||
|  * FIXED: PlUnit printing of wrong answers if there are attributes.  Ulrich | ||||
|    Neumerkel, item#242 | ||||
| Oct 30, 2007 | ||||
|  | ||||
| * ENHANCED: Report error or failure in cleanup handler (Mark van Assem) | ||||
| * ENHANCED: Do not report '% All tests passed' if no tests are | ||||
|   executed after a make/0. | ||||
|  | ||||
| Sep 21, 2007 | ||||
|  | ||||
| * ENHANCED: Verify test-set and test options. | ||||
|  | ||||
| * ADDED: Allow for Var == Value as shorthand for true(Var == Value). | ||||
|  | ||||
| Sep 20, 2007 | ||||
|  | ||||
| * ENHANCED: Message on wrong error | ||||
|  | ||||
| * DOCS: Fixed typos.  Ulrich Neumerkel. | ||||
|  | ||||
| * FIXED: avoid a loop if a test throws an error holding a cyclic term. | ||||
|  | ||||
| * MODIFIED: errors are now verified using subsumes_chk/2 instead of =@=. | ||||
|  | ||||
| * PORT: =@= is now correctly bound to variant/2 for SICStus. | ||||
|  | ||||
| Aug 1, 2007 | ||||
|  | ||||
| * FIXED: Ran failing tests twice due to unwanted choicepoint. | ||||
|  | ||||
| Jul 30, 2007 | ||||
|  | ||||
| * ADDED: allow for error(Error) as alias for throws(error(Error, _)). | ||||
|  | ||||
| Mar 28, 2007 | ||||
|  | ||||
| * ADDED: option silent(Bool) to force completely silent operation. | ||||
|  | ||||
| Mar 6, 2007 | ||||
|  | ||||
| * Allow for single option without list | ||||
|  | ||||
| Nov 23, 2006 | ||||
|  | ||||
| * Force start of line after .... | ||||
|  | ||||
| Oct 22, 2006 | ||||
|  | ||||
| * Allow tracing body of tests.  Suggested by Lukas Degener. | ||||
|  | ||||
| Oct 17, 2006 | ||||
|  | ||||
| * Portability with SICStus 3.12. | ||||
|  | ||||
| Oct 13, 2006 | ||||
|  | ||||
| * Better message for tests that succeeded but should have failed | ||||
							
								
								
									
										132
									
								
								packages/plunit/Makefile.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								packages/plunit/Makefile.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| ################################################################ | ||||
| # SWI-Prolog `PlDoc' package | ||||
| # Author:    Jan Wielemaker. wielemak@science.uva.nl | ||||
| # Copyright: GPL (see COPYING or www.gnu.org | ||||
| ################################################################ | ||||
|  | ||||
| .SUFFIXES: .tex .dvi .doc .pl | ||||
|  | ||||
| SHELL=@SHELL@ | ||||
|  | ||||
| ifeq (@PROLOG_SYSTEM@,yap) | ||||
|  | ||||
| prefix = @prefix@ | ||||
| ROOTDIR = $(prefix) | ||||
| EROOTDIR = @exec_prefix@ | ||||
|  | ||||
| srcdir=@srcdir@ | ||||
|  | ||||
| BINDIR = $(EROOTDIR)/bin | ||||
| YAPLIBDIR=$(EROOTDIR)/lib/Yap | ||||
| # hack for SWI compatibility | ||||
| LIBDIR=$(EROOTDIR)/share/Yap | ||||
|  | ||||
| PL=@EXTEND_DYNLOADER_PATH@ $(DESTDIR)$(BINDIR)/yap $(DESTDIR)$(YAPLIBDIR)/startup.yss | ||||
| LN_S=@LN_S@ | ||||
| EXDIR=$(LIBDIR)/examples/plunit | ||||
|  | ||||
| else	# SWI | ||||
|  | ||||
| srcdir=. | ||||
|  | ||||
| PLBASE=@PLBASE@ | ||||
| PLARCH=@PLARCH@ | ||||
| PL=@PL@ | ||||
| XPCEBASE=$(PLBASE)/xpce | ||||
| PKGDOC=$(PLBASE)/doc/packages | ||||
| PCEHOME=../xpce | ||||
| LIBDIR=$(PLBASE)/library | ||||
| EXDIR=$(PKGDOC)/examples/plunit | ||||
|  | ||||
| endif | ||||
|  | ||||
| DESTDIR= | ||||
|  | ||||
| DOCTOTEX=$(PCEHOME)/bin/doc2tex | ||||
| PLTOTEX=$(PCEHOME)/bin/pl2tex | ||||
| vLATEX=latex | ||||
| DOC=plunit | ||||
| TEX=$(DOC).tex | ||||
| DVI=$(DOC).dvi | ||||
| PDF=$(DOC).pdf | ||||
| HTML=$(DOC).html | ||||
|  | ||||
| INSTALL=@INSTALL@ | ||||
| INSTALL_PROGRAM=@INSTALL_PROGRAM@ | ||||
| INSTALL_DATA=@INSTALL_DATA@ | ||||
|  | ||||
| LIBPL=		$(srcdir)/plunit.pl $(srcdir)/test_wizard.pl $(srcdir)/test_cover.pl | ||||
| LIBALL=		$(LIBPL) $(SUPPORT) | ||||
| EXAMPLES=	$(srcdir)/read.pl $(srcdir)/simple.pl | ||||
|  | ||||
| all:		 | ||||
| 		@echo "Nothing to do for this package" | ||||
|  | ||||
| install:	$(LIBPL) | ||||
| 		mkdir -p $(DESTDIR)$(LIBDIR) | ||||
| 		$(INSTALL_DATA) $(LIBALL) $(DESTDIR)$(LIBDIR) | ||||
| 		$(PL) -f none -g make -t halt | ||||
|  | ||||
| ln-install:	$(LIBPL) | ||||
| 		mkdir -p $(DESTDIR)$(LIBDIR) | ||||
| 		for f in $(LIBALL); do \ | ||||
| 		    rm -f $(DESTDIR)$(LIBDIR)/$$f; \ | ||||
| 		    ln -s `pwd`/$$f $(DESTDIR)$(LIBDIR); \ | ||||
| 		done | ||||
| 		$(PL) -f none -g make -t halt | ||||
|  | ||||
| rpm-install:	install | ||||
|  | ||||
| pdf-install:	install-examples | ||||
| 		$(INSTALL_DATA) $(DOC).pdf $(DESTDIR)$(PKGDOC) | ||||
|  | ||||
| html-install:	install-examples | ||||
| 		$(INSTALL_DATA) $(DOC).html $(DESTDIR)$(PKGDOC) | ||||
|  | ||||
| install-examples:: | ||||
| 		mkdir -p $(DESTDIR)$(EXDIR) | ||||
| 		(cd examples && $(INSTALL_DATA) $(EXAMPLES) $(DESTDIR)$(EXDIR)) | ||||
|  | ||||
| uninstall:: | ||||
| 		rm -f $(LIBDIR)/plunit.pl | ||||
| 		$(PL) -f none -g make -t halt | ||||
|  | ||||
| check:: | ||||
| 		true | ||||
|  | ||||
| ################################################################ | ||||
| # Documentation | ||||
| ################################################################ | ||||
|  | ||||
| doc:		$(PDF) $(HTML) | ||||
| pdf:		$(PDF) | ||||
| html:		$(HTML) | ||||
|  | ||||
| $(HTML):	$(TEX) | ||||
| 		latex2html $(DOC) | ||||
| 		mv html/index.html $@ | ||||
| 		rmdir html | ||||
|  | ||||
| $(PDF):		$(TEX) | ||||
| 		../../man/runtex --pdf $(DOC) | ||||
|  | ||||
| $(TEX):		$(DOCTOTEX) | ||||
|  | ||||
| .doc.tex: | ||||
| 		$(DOCTOTEX) $*.doc > $*.tex | ||||
| .pl.tex: | ||||
| 		$(PLTOTEX) $*.pl > $*.tex | ||||
|  | ||||
| ################################################################ | ||||
| # Clean | ||||
| ################################################################ | ||||
|  | ||||
| clean: | ||||
| 		rm -f *~ *% config.log | ||||
|  | ||||
| #		rm -f $(TEX) | ||||
| #		../../man/runtex --clean $(DOC) | ||||
| #		rm -rf html | ||||
|  | ||||
| distclean:	clean | ||||
| 		rm -f $(TARGETS) config.cache config.status Makefile | ||||
							
								
								
									
										46
									
								
								packages/plunit/README
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								packages/plunit/README
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| Prolog Unit Tests | ||||
| ================= | ||||
|  | ||||
|  | ||||
| ---++ TBD | ||||
|  | ||||
| 	* Add options to test units | ||||
| 		- Condition, setup, cleanup, blocked [OK] | ||||
| 	* Run the tests on make, etc.  [OK] | ||||
| 	* Report generation options | ||||
|  | ||||
| ---++ What to do with loaded tests? | ||||
|  | ||||
| 	* Keep them around | ||||
| 	* Remove them | ||||
| 	* Flag module as `volatile', so it is not saved. | ||||
|  | ||||
| ---++ Distinguish user/system/library tests | ||||
|  | ||||
| ---++ Deal with seperate test-files: | ||||
|  | ||||
| 	* load_test_files(+Options) | ||||
|  | ||||
| ---++ Volatile modules? | ||||
|  | ||||
| ---++ Conversion of system tests: | ||||
|  | ||||
| 	- test.pl internals [OK] | ||||
| 	- scripts called from test.pl [OK] | ||||
| 	- tests for packages | ||||
|  | ||||
| ---++ Run tests concurrently? | ||||
|  | ||||
| 	+ Tests applications for concurrent execution | ||||
| 	+ Saves time, notably on tests doing networking, timeout, etc. | ||||
| 	- Might be more complicated to understand | ||||
|  | ||||
| ---++ Test wizard | ||||
|  | ||||
| 	* Write tests to .plt files | ||||
| 	* Merge with existing content of .plt files | ||||
| 		- Read with comments and write back? | ||||
|  | ||||
| ---++ Coverage analysis | ||||
|  | ||||
| 	* See cover.pl | ||||
							
								
								
									
										22
									
								
								packages/plunit/examples/read.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								packages/plunit/examples/read.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
|  | ||||
| :- use_module(library(plunit)). | ||||
|  | ||||
| :- begin_tests(read). | ||||
|  | ||||
| :- use_module(library(system)). | ||||
|  | ||||
| :- use_module(library(readutil)). | ||||
|  | ||||
| create_file(Tmp) :- | ||||
| 	tmp_file(plunit, Tmp), | ||||
| 	open(Tmp, write, Out), | ||||
| 	write(Out, 'hello(World).\n'), | ||||
| 	close(Out). | ||||
|  | ||||
| test(read, [ setup(create_file(Tmp)), | ||||
| 	     cleanup(delete_file(Tmp)) | ||||
| 	   ]) :- | ||||
| 	read_file_to_terms(Tmp, Terms, []), | ||||
| 	Terms =@= [hello(_)]. | ||||
|  | ||||
| %:- end_tests(read). | ||||
							
								
								
									
										31
									
								
								packages/plunit/examples/simple.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								packages/plunit/examples/simple.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| :- module(ex_simple, []). | ||||
| :- use_module(library(plunit)). | ||||
|  | ||||
| :- begin_tests(lists). | ||||
|  | ||||
| test(true) :- | ||||
| 	true. | ||||
|  | ||||
| test(fail) :- | ||||
| 	\+ fail. | ||||
|  | ||||
| test(fail, [fail]) :- | ||||
| 	fail. | ||||
|  | ||||
| test(member) :- | ||||
| 	member(a, [a]), !. | ||||
|  | ||||
| test(member, [nondet]) :- | ||||
| 	member(_, [a]). | ||||
|  | ||||
| test(member, [true(X == a)]) :- | ||||
| 	member(X, [a]), !. | ||||
|  | ||||
| test(member, [all(V == [a,b,c])]) :- | ||||
| 	member(V, [a,b,c]). | ||||
|  | ||||
| test(append) :- | ||||
| 	append("aap", "noot", X), | ||||
| 	X == "aapnoot". | ||||
|  | ||||
| :- end_tests(lists). | ||||
							
								
								
									
										671
									
								
								packages/plunit/plunit.doc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										671
									
								
								packages/plunit/plunit.doc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,671 @@ | ||||
| \documentclass[11pt]{article} | ||||
| \usepackage{times} | ||||
| \usepackage{pl} | ||||
| \usepackage{html} | ||||
| \usepackage{plpage} | ||||
| \sloppy | ||||
| \makeindex | ||||
|  | ||||
| \onefile | ||||
| \htmloutput{html}				% Output directory | ||||
| \htmlmainfile{index}				% Main document file | ||||
| \bodycolor{white}				% Page colour | ||||
|  | ||||
| \renewcommand{\runningtitle}{Prolog Unit Tests} | ||||
|  | ||||
| \begin{document} | ||||
|  | ||||
| \title{Prolog Unit Tests} | ||||
| \author{Jan Wielemaker \\ | ||||
| 	HCS, \\ | ||||
| 	University of Amsterdam \\ | ||||
| 	The Netherlands \\ | ||||
| 	E-mail: \email{wielemak@science.uva.nl}} | ||||
|  | ||||
| \maketitle | ||||
|  | ||||
| \begin{abstract} | ||||
| This document describes a Prolog unit-test framework. This framework was | ||||
| initially developed for \url[SWI-Prolog]{http://www.swi-prolog.org}. The | ||||
| current version also runs on \url[SICStus | ||||
| Prolog]{http://www.sics.se/sicstus/}, providing a portable testing | ||||
| framework. See \secref{sicstus}. | ||||
| \end{abstract} | ||||
|  | ||||
| \pagebreak | ||||
| \tableofcontents | ||||
|  | ||||
| \vfill | ||||
| \vfill | ||||
|  | ||||
| \newpage | ||||
|  | ||||
| \section{Introduction} | ||||
| \label{sec:intro} | ||||
|  | ||||
| There is really no excuse not to write tests! | ||||
|  | ||||
| Automatic testing of software during development is probably the most | ||||
| important Quality Assurance measure. Tests can validate the final | ||||
| system, which is nice for your users.  However, most (Prolog) developers | ||||
| forget that it is not just a burden during development. | ||||
|  | ||||
| \begin{itemize} | ||||
|     \item Tests document how the code is supposed to be used. | ||||
|     \item Tests can validate claims you make on the Prolog | ||||
|           implementation.  Writing a test makes the claim | ||||
| 	  explicit. | ||||
|     \item Tests avoid big applications saying `No' after | ||||
|           modifications.  This saves time during development, | ||||
| 	  and it saves \emph{a lot} of time if you must return | ||||
| 	  to the application a few years later or you must | ||||
| 	  modify and debug someone else's application. | ||||
| \end{itemize} | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{A Unit Test box} | ||||
| \label{sec:unitbox} | ||||
|  | ||||
| Tests are written in pure Prolog and enclosed within the directives | ||||
| begin_tests/1,2 and end_tests/1. They can be embedded inside a normal | ||||
| source module, or be placed in a separate test-file that loads the files | ||||
| to be tested. Code inside a test box is normal Prolog code. The | ||||
| entry points are defined by rules using the head \term{test}{Name} or | ||||
| \term{test}{Name, Options}, where \arg{Name} is a ground term and | ||||
| \arg{Options} is a list describing additional properties of the test. | ||||
| Here is a very simple example: | ||||
|  | ||||
| \begin{code} | ||||
| :- begin_tests(lists). | ||||
| :- use_module(library(lists)). | ||||
|  | ||||
| test(reverse) :- | ||||
| 	reverse([a,b], [b,a]). | ||||
|  | ||||
| :- end_tests(lists). | ||||
| \end{code} | ||||
|  | ||||
| The optional second argument of the test-head defines additional processing | ||||
| options.  Defined options are: | ||||
|  | ||||
| \begin{description} | ||||
|     \termitem{blocked}{+Reason:atom} | ||||
| The test is currently disabled.   Tests are flagged as blocked if they | ||||
| cannot be run for some reason.  E.g.\ they crash Prolog, they rely on | ||||
| some service that is not available, they take too much resources, etc. | ||||
| Tests that fail but do not crash, etc.\ should be flagged using | ||||
| \term{fixme}{Fixme}. | ||||
|  | ||||
|     \termitem{fixme}{+Reason:atom} | ||||
| Similar to \term{blocked}{Reason}, but the test it executed anyway.  If | ||||
| it fails, a \const{-} is printed instead of the \const{.} character.  If | ||||
| it passes a \const{+} and if it passes with a choicepoint, \const{!}. | ||||
| A summary is printed at the end of the test run and the goal | ||||
| \term{test_report}{fixme} can be used to get details. | ||||
|  | ||||
|     \termitem{condition}{:Goal} | ||||
| Pre-condition for running the test.  If the condition fails | ||||
| the test is skipped.  The condition can be used as an alternative | ||||
| to the \const{setup} option.  The only difference is that failure | ||||
| of a condition skips the test and is considered an error when using | ||||
| the \const{setup} option. | ||||
|  | ||||
|     \termitem{cleanup}{:Goal} | ||||
| \arg{Goal} is always called after completion of the test-body, | ||||
| regardless of whether it fails, succeeds or throws an exception.  This | ||||
| option or call_cleanup/2 must be used by tests that require side-effects | ||||
| that must be reverted after the test completes.  \arg{Goal} may share | ||||
| variables with the test body. | ||||
|  | ||||
| \begin{code} | ||||
| create_file(Tmp) :- | ||||
| 	tmp_file(plunit, Tmp), | ||||
| 	open(Tmp, write, Out), | ||||
| 	write(Out, 'hello(World).\n'), | ||||
| 	close(Out). | ||||
|  | ||||
| test(read, [ setup(create_file(Tmp)), | ||||
| 	     cleanup(delete_file(Tmp)) | ||||
| 	   ]) :- | ||||
| 	read_file_to_terms(Tmp, Terms, []), | ||||
| 	Term = hello(_). | ||||
| \end{code} | ||||
|  | ||||
|     \termitem{setup}{:Goal} | ||||
| \arg{Goal} is run before the test-body.  Typically used together with | ||||
| the \const{cleanup} option to create and destroy the required execution | ||||
| environment. | ||||
|  | ||||
|     \termitem{forall}{:Generator} | ||||
| Run the same test for each solution of \arg{Generator}. Each run invokes | ||||
| the setup and cleanup handlers. This can be used to run the same test | ||||
| with different inputs.  If an error occurs, the test is reported as | ||||
| \mbox{\texttt{name (forall bindings = } <vars> \texttt{)}}, where | ||||
| <vars> indicates the bindings of variables in \arg{Generator}. | ||||
|  | ||||
|     \termitem{true}{AnswerTerm Cmp Value} | ||||
| Body must succeed deterministically. AnswerTerm is compared to Value | ||||
| using the comparison operator \arg{Cmp}. \arg{Cmp} is typically one of | ||||
| =/2, ==/2, =:=/2 or =@=/2,% | ||||
|     \footnote{The =@= predicate (denoted \emph{structural equivalence}) | ||||
| 	      is the same as variant/2 in SICStus.} | ||||
| but any test can be used. This is the same as inserting the test at the | ||||
| end of the conjunction, but it allows the test engine to distinguish | ||||
| between failure of copy_term/2 and producing the wrong value. Multiple | ||||
| variables must be combined in an arbitrary compound term. E.g.\ | ||||
| \verb$A1-A2 == v1-v2$ | ||||
|  | ||||
| \begin{code} | ||||
| test(copy, [ true(Copy =@= hello(X,X)) | ||||
| 	   ]) :- | ||||
| 	copy_term(hello(Y,Y), Copy). | ||||
| \end{code} | ||||
|  | ||||
|     \termitem{AnswerTerm Cmp Value} | ||||
| Equivalent to \term{true}{AnswerTerm Cmp Value} if \arg{Cmp} is one | ||||
| of the comparison operators given above. | ||||
|  | ||||
|     \termitem{fail}{} | ||||
| Body must fail. | ||||
|  | ||||
|     \termitem{throws}{Error} | ||||
| Body must throw \arg{Error}. The error is verified using | ||||
| \term{subsumes_chk}{Error, Generated}.  I.e.\ the generated error | ||||
| must be more specific than the specified \arg{Error}. | ||||
|  | ||||
|     \termitem{error}{Error} | ||||
| Body must throw \term{error}{Error, _Context}.  See \const{throws} | ||||
| for details. | ||||
|  | ||||
|     \termitem{all}{AnswerTerm Cmp Instances} | ||||
| Similar to \term{true}{AnswerTerm Cmp Values}, but used for non-deterministic | ||||
| predicates. Each element is compared using \arg{Cmp}. Order matters. For | ||||
| example: | ||||
|  | ||||
| \begin{code} | ||||
| test(or, all(X == [1,2])) :- | ||||
| 	( X = 1 ; X = 2 ). | ||||
| \end{code} | ||||
|  | ||||
|     \termitem{set}{AnswerTerm Cmp Instances} | ||||
| Similar to \term{all}{AnswerTerm Cmp Instances}, but ignores order and | ||||
| duplicates with respect to \arg{Cmp} in the comparison. Each element is | ||||
| compared using \arg{Cmp}.  | ||||
|  | ||||
|     \termitem{nondet}{} | ||||
| If this keyword appears in the option list, non-deterministic success | ||||
| of the body is not considered an error. | ||||
|  | ||||
|     \termitem{sto}{Terms} | ||||
| Declares that executing body is subject to occurs-check (STO).  The | ||||
| test is executed with \arg{Terms}.  \arg{Terms} is either | ||||
| \const{rational_trees} or \const{finite_trees}.  STO programs are not | ||||
| portable between different kinds of terms.  Only programs \emph{not} | ||||
| subject to occurs-check (NSTO) are portable\footnote{See 7.3.3 of | ||||
| ISO/IEC 13211-1 PROLOG: Part 1 - General Core, for a detailed | ||||
| discussion of STO and NSTO}.  Fortunately, most practical programs are | ||||
| NSTO.  Writing tests that are STO is still useful to ensure the | ||||
| robustness of a predicate.  In case sto4 and sto5 below, an infinite | ||||
| list (a rational tree) is created prior to calling the actual | ||||
| predicate.  Ideally, such cases produce a type error or fail silently. | ||||
|  | ||||
| \begin{code} | ||||
| test(sto1, [sto(rational_trees)]) :- | ||||
| 	X=s(X). | ||||
| test(sto2, [sto(finite_trees),fail]) :- | ||||
| 	X=s(X). | ||||
| test(sto3, [sto(rational_trees), fail]) :- | ||||
| 	X=s(X), fail. | ||||
| test(sto4, [sto(rational_trees),error(type_error(list,L))]) :- | ||||
| 	L = [_|L], length(L,_). | ||||
| test(sto5, [sto(rational_trees),fail]) :- | ||||
| 	L = [_|L], length(L,3). | ||||
| \end{code} | ||||
|  | ||||
| Programs that depend on STO cases tend to be inefficient, even | ||||
| incorrect, are hard to understand and debug, and terminate poorly.  It | ||||
| is therefore advisable to avoid STO programs whenever possible. | ||||
|  | ||||
| SWI's Prolog flag \prologflag{occurs_check} must not be modified | ||||
| within plunit tests. | ||||
|  | ||||
| \end{description} | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \subsection{Test Unit options} | ||||
| \label{sec:unitoptions} | ||||
|  | ||||
| \begin{description} | ||||
|     \predicate{begin_tests}{1}{+Name} | ||||
| Start named test-unit.  Same as \verb$begin_tests(Name, [])$. | ||||
|  | ||||
|     \predicate{begin_tests}{2}{+Name, +Options} | ||||
| Start named test-unit with options. Options provide conditional | ||||
| processing, setup and cleanup similar to individual tests (second | ||||
| argument of test/2 rules). | ||||
|  | ||||
| Defined options are: | ||||
|     \begin{description} | ||||
| 	\termitem{blocked}{+Reason} | ||||
| Test-unit has been blocked for the given \arg{Reason}. | ||||
|  | ||||
|         \termitem{condition}{:Goal} | ||||
| Executed before executing any of the tests.  If \arg{Goal} fails, | ||||
| the test of this unit is skipped. | ||||
|  | ||||
|         \termitem{setup}{:Goal} | ||||
| Executed before executing any of the tests.   | ||||
|  | ||||
|         \termitem{cleanup}{:Goal} | ||||
| Executed after completion of all tests in the unit. | ||||
|  | ||||
| 	\termitem{sto}{+Terms} | ||||
| Specify default for subject-to-occurs-check mode. See \secref{unitbox} | ||||
| for details on the sto option. | ||||
|     \end{description} | ||||
| \end{description} | ||||
|  | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \subsection{Writing the test body} | ||||
| \label{sec:testbody} | ||||
|  | ||||
| The test-body is ordinary Prolog code. Without any options, the body | ||||
| must be designed to succeed \emph{deterministically}. Any other result | ||||
| is considered a failure. One of the options \const{fail}, \const{true}, | ||||
| \const{throws}, \const{all} or \const{set} can be used to specify a | ||||
| different expected result. See \secref{unitbox} for details.  In this | ||||
| section we illustrate typical test-scenarios by testing SWI-Prolog | ||||
| built-in and library predicates. | ||||
|  | ||||
| \subsubsection{Testing deterministic predicates} | ||||
| \label{sec:testdet} | ||||
|  | ||||
| Deterministic predicates are predicates that must succeed exactly once | ||||
| and, for well behaved predicates, leave no choicepoints. Typically they | ||||
| have zero or more input- and zero or more output arguments. The test | ||||
| goal supplies proper values for the input arguments and verifies the | ||||
| output arguments. Verification can use test-options or be explicit in | ||||
| the body.  The tests in the example below are equivalent. | ||||
|  | ||||
| \begin{code} | ||||
| test(add) :- | ||||
| 	A is 1 + 2, | ||||
| 	A =:= 3. | ||||
|  | ||||
| test(add, [true(A =:= 3)]) :- | ||||
| 	A is 1 + 2. | ||||
| \end{code} | ||||
|  | ||||
| The test engine verifies that the test-body does not leave a | ||||
| choicepoint.  We illustrate that using the test below: | ||||
|  | ||||
| \begin{code} | ||||
| test(member) :- | ||||
| 	member(b, [a,b,c]). | ||||
| \end{code} | ||||
|  | ||||
| Although this test succeeds, member/2 leaves a choicepoint which is | ||||
| reported by the test subsystem.  To make the test silent, use one of | ||||
| the alternatives below. | ||||
|  | ||||
| \begin{code} | ||||
| test(member) :- | ||||
| 	member(b, [a,b,c]), !. | ||||
|  | ||||
| test(member, [nondet]) :- | ||||
| 	member(b, [a,b,c]). | ||||
| \end{code} | ||||
|  | ||||
| \subsubsection{Testing semi-deterministic predicates} | ||||
| \label{sec:testsemidet} | ||||
|  | ||||
| Semi-deterministic predicates are predicates that either fail or succeed | ||||
| exactly once and, for well behaved predicates, leave no choicepoints. | ||||
| Testing such predicates is the same as testing deterministic | ||||
| predicates.  Negative tests must be specified using the option | ||||
| \const{fail} or by negating the body using \verb$\+/1$. | ||||
|  | ||||
| \begin{code} | ||||
| test(is_set) :- | ||||
| 	\+ is_set([a,a]). | ||||
|  | ||||
| test(is_set, [fail]) :- | ||||
| 	is_set([a,a]). | ||||
| \end{code} | ||||
|  | ||||
|  | ||||
| \subsubsection{Testing non-deterministic predicates} | ||||
| \label{sec:testnondet} | ||||
|  | ||||
| Non-deterministic predicates succeed zero or more times.  Their results | ||||
| are tested either using findall/3 or setof/3 followed by a value-check | ||||
| or using the \const{all} or \const{set} options.  The following are | ||||
| equivalent tests: | ||||
|  | ||||
| \begin{code} | ||||
| test(member) :- | ||||
| 	findall(X, member(X, [a,b,c]), Xs), | ||||
| 	Xs == [a,b,c]. | ||||
|  | ||||
| test(member, all(X == [a,b,c])) :- | ||||
| 	member(X, [a,b,c]). | ||||
| \end{code} | ||||
|  | ||||
| \subsubsection{Testing error conditions} | ||||
| \label{sec:testerror} | ||||
|  | ||||
| Error-conditions are tested using the option \term{throws}{Error} or | ||||
| by wrapping the test in a catch/3.  The following tests are equivalent: | ||||
|  | ||||
| \begin{code} | ||||
| test(div0) :- | ||||
|      catch(A is 1/0, error(E, _), true), | ||||
|      E =@= evaluation_error(zero_divisor). | ||||
|  | ||||
| test(div0, [error(evaluation_error(zero_divisor))]) :- | ||||
|      A is 1/0. | ||||
| \end{code} | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Using separate test files} | ||||
| \label{sec:testfiles} | ||||
|  | ||||
| Test-units can be embedded in normal Prolog source-files. Alternatively, | ||||
| tests for a source-file can be placed in another file alongside the file | ||||
| to be tested. Test files use the extension \fileext{plt}. The predicate | ||||
| load_test_files/1 can load all files that are related to source-files | ||||
| loaded into the current project. | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Running the test-suite} | ||||
| \label{sec:running} | ||||
|  | ||||
| At any time, the tests can be executed by loading the program and | ||||
| running run_tests/0 or run_tests(+Unit).   | ||||
|  | ||||
| \begin{description} | ||||
|     \predicate{run_tests}{0}{} | ||||
| Run all test-units. | ||||
|  | ||||
|     \predicate{run_tests}{1}{+Spec} | ||||
| Run only the specified tests.  \arg{Spec} can be a list to run multiple | ||||
| tests.  A single specification is either the name of a test unit or | ||||
| a term <Unit>:<Tests>, running only the specified test.  <Tests> is | ||||
| either the name of a test or a list of names. Running particular | ||||
| tests is particularly useful for tracing a test:% | ||||
| \footnote{Unfortunately the body of the test is called through | ||||
| meta-calling, so it cannot be traced. The called user-code can be traced | ||||
| normally though.} | ||||
|  | ||||
| \begin{code} | ||||
| ?- gtrace, run_tests(lists:member). | ||||
| \end{code} | ||||
| \end{description} | ||||
|  | ||||
| To identify nonterminating tests, interrupt the looping process with | ||||
| \emph{Control-C}. The test name and location will be displayed. | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Tests and production systems} | ||||
| \label{sec:state} | ||||
|  | ||||
| Most applications do not want the test-suite to end up in the | ||||
| final application.  There are several ways to achieve this.  One | ||||
| is to place all tests in separate files and not to load the tests | ||||
| when creating the production environment.  Alternatively, use the | ||||
| directive below before loading the application. | ||||
|  | ||||
| \begin{code} | ||||
| :- set_test_options([load(never)]). | ||||
| \end{code} | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Controlling the test suite} | ||||
| \label{sec:options} | ||||
|  | ||||
| \begin{description} | ||||
|     \predicate{set_test_options}{1}{+Options} | ||||
| Defined options are: | ||||
|  | ||||
| \begin{description} | ||||
|     \termitem{load}{+Load} | ||||
| Determines whether or not tests are loaded. When \const{never}, | ||||
| everything between begin_tests/1 and end_tests/1 is simply ignored. | ||||
| When \const{always}, tests are always loaded.  Finally, when using | ||||
| the default value \const{normal}, tests are loaded if the code is | ||||
| not compiled with optimisation turned on. | ||||
|  | ||||
|     \termitem{run}{+Run} | ||||
| Specifies when tests are run. Using \const{manual}, tests can only be | ||||
| run using run_tests/0 or run_tests/1. Using \const{make}, tests will be | ||||
| run for reloaded files, but not for files loaded the first time. Using | ||||
| \const{make(all)} make/0 will run all test-suites, not only those that | ||||
| belong to files that are reloaded. | ||||
|  | ||||
|     \termitem{silent}{+Bool} | ||||
| When \const{true} (default is \const{false}), send informational | ||||
| messages using the `silent' level.  In practice this means there | ||||
| is no output except for errors. | ||||
|  | ||||
|     \termitem{sto}{+Bool} | ||||
| When \const{true} (default \const{false}), assume tests are not subject | ||||
| to occurs check (non-STO) and verify this if the Prolog implementation | ||||
| supports testing this. | ||||
| \end{description} | ||||
|  | ||||
|     \predicate{load_test_files}{1}{+Options} | ||||
| Load \fileext{plt} test-files that belong to the currently loaded | ||||
| sources.      | ||||
|  | ||||
|     \predicate{running_tests}{0}{} | ||||
| Print all currently running tests to the terminal.  It can be used | ||||
| to find running thread in multi-threaded test operation or find the | ||||
| currently running test if a test appears to be blocking. | ||||
|  | ||||
|     \predicate{test_report}{1}{+What} | ||||
| Print report on the executed tests.  \arg{What} defines the type | ||||
| of report.  Currently this only supports \const{fixme}, providing | ||||
| details on how the fixme-flagged tests proceeded. | ||||
| \end{description} | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Auto-generating tests} | ||||
| \label{sec:wizard} | ||||
|  | ||||
| Prolog is an interactive environment. Where users of non-interactive | ||||
| systems tend to write tests as code, Prolog developers tend to run | ||||
| queries interactively during development. This interactive testing is | ||||
| generally faster, but the disadvantage is that the tests are lost at the | ||||
| end of the session. The test-wizard tries to combine the advantages. It | ||||
| collects toplevel queries and saves them to a specified file.  Later, | ||||
| it extracts these queries from the file and locates the predicates that | ||||
| are tested by the queries.  It runs the query and creates a test clause | ||||
| from the query. | ||||
|  | ||||
| Auto-generating test cases is experimentally supported through the | ||||
| library \pllib{test_wizard}. We briefly introduce the functionality | ||||
| using examples. First step is to log the queries into a file. This is | ||||
| accomplished with the commands below. \file{Queries.pl} is the name in | ||||
| which to store all queries. The user can choose any filename for this | ||||
| purpose.   Multiple Prolog instances can share the same name, as data | ||||
| is appended to this file and write is properly locked to avoid file | ||||
| corruption. | ||||
|  | ||||
| \begin{code} | ||||
| :- use_module(library(test_wizard)). | ||||
| :- set_prolog_flag(log_query_file, 'Queries.pl'). | ||||
| \end{code} | ||||
|  | ||||
| Next, we will illustrate using the library by testing the predicates | ||||
| from library \pllib{lists}.  To generate test cases we just make calls | ||||
| on the terminal.  Note that all queries are recorded and the system will | ||||
| select the appropriate ones when generating the test unit for a | ||||
| particular module. | ||||
|  | ||||
| \begin{code} | ||||
| ?- member(b, [a,b]). | ||||
| Yes | ||||
| ?- reverse([a,b], [b|A]). | ||||
| A = [a] ; | ||||
| No | ||||
| \end{code} | ||||
|  | ||||
| Now we can generate the test-cases for the module list using | ||||
| make_tests/3: | ||||
|  | ||||
| \begin{code} | ||||
| ?- make_tests(lists, 'Queries.pl', current_output). | ||||
| :- begin_tests(lists). | ||||
|  | ||||
| test(member, [nondet]) :- | ||||
|         member(b, [a, b]). | ||||
| test(reverse, [true(A==[a])]) :- | ||||
|         reverse([a, b], [b|A]). | ||||
|  | ||||
| :- end_tests(lists). | ||||
| \end{code} | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Coverage analysis} | ||||
| \label{sec:cover} | ||||
|  | ||||
| An important aspect of tests is to know which parts of program | ||||
| is used (\emph{covered}) by the tests.  An experimental analysis | ||||
| is provided by the library \pllib{test_cover}.   | ||||
|  | ||||
| \begin{description} | ||||
|     \predicate{show_coverage}{1}{:Goal} | ||||
| Run \arg{Goal} and write a report on which percentage of the clauses in | ||||
| each file are used by the program and which percentage of the clauses | ||||
| always fail. | ||||
| \end{description} | ||||
|  | ||||
| We illustrate this here using CHAT, a natural language question and | ||||
| answer application by David H.D. Warren and Fernando C.N. Pereira. | ||||
|  | ||||
| \begin{code} | ||||
| 1 ?- show_coverage(test_chat). | ||||
| Chat Natural Language Question Answering Test | ||||
| ... | ||||
|  | ||||
| ================================================================== | ||||
|                          Coverage by File | ||||
| ================================================================== | ||||
| File                                        Clauses    %Cov %Fail | ||||
| ================================================================== | ||||
| /staff/jan/lib/prolog/chat/xgrun.pl                5   100.0   0.0 | ||||
| /staff/jan/lib/prolog/chat/newg.pl               186    89.2  18.3 | ||||
| /staff/jan/lib/prolog/chat/clotab.pl              28    89.3   0.0 | ||||
| /staff/jan/lib/prolog/chat/newdic.pl             275    35.6   0.0 | ||||
| /staff/jan/lib/prolog/chat/slots.pl              128    74.2   1.6 | ||||
| /staff/jan/lib/prolog/chat/scopes.pl             132    70.5   3.0 | ||||
| /staff/jan/lib/prolog/chat/templa.pl              67    55.2   1.5 | ||||
| /staff/jan/lib/prolog/chat/qplan.pl              106    75.5   0.9 | ||||
| /staff/jan/lib/prolog/chat/talkr.pl               60    20.0   1.7 | ||||
| /staff/jan/lib/prolog/chat/ndtabl.pl              42    59.5   0.0 | ||||
| /staff/jan/lib/prolog/chat/aggreg.pl              47    48.9   2.1 | ||||
| /staff/jan/lib/prolog/chat/world0.pl             131    71.8   1.5 | ||||
| /staff/jan/lib/prolog/chat/rivers.pl              41   100.0   0.0 | ||||
| /staff/jan/lib/prolog/chat/cities.pl              76    43.4   0.0 | ||||
| /staff/jan/lib/prolog/chat/countr.pl             156   100.0   0.0 | ||||
| /staff/jan/lib/prolog/chat/contai.pl             334   100.0   0.0 | ||||
| /staff/jan/lib/prolog/chat/border.pl             857    98.6   0.0 | ||||
| /staff/jan/lib/prolog/chat/chattop.pl            139    43.9   0.7 | ||||
| ================================================================== | ||||
| \end{code} | ||||
|  | ||||
| Using \verb$?- show_coverage(run_tests).$, this library currently only | ||||
| shows some rough quality measure for test-suite.  Later versions should | ||||
| provide a report to the developer identifying which clauses are covered, | ||||
| not covered and always failed. | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Portability of the test-suite} | ||||
| \label{sec:porting} | ||||
|  | ||||
| One of the reasons to have tests is to simplify migrating code between | ||||
| Prolog implementations. Unfortunately creating a portable test-suite | ||||
| implies a poor integration into the development environment. Luckily, | ||||
| the specification of the test-system proposed here can be ported quite | ||||
| easily to most Prolog systems sufficiently compatible to SWI-Prolog to | ||||
| consider porting your application. Most important is to have support for | ||||
| term_expansion/2. | ||||
|  | ||||
| In the current system, test units are compiled into sub-modules of the | ||||
| module in which they appear.  Few Prolog systems allow for sub-modules | ||||
| and therefore ports may have to fall-back to inject the code in the | ||||
| surrounding module.  This implies that support predicates used inside | ||||
| the test unit should not conflict with predicates of the module being | ||||
| tested. | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \subsection{PlUnit on SICStus} | ||||
| \label{sec:sicstus} | ||||
|  | ||||
| The directory of \file{plunit.pl} and \file{swi.pl} must be in the | ||||
| \const{library} search-path.  With PLUNITDIR replaced accordingly, | ||||
| add the following into your \file{.sicstusrc} or \file{sicstus.ini}. | ||||
|  | ||||
| \begin{code} | ||||
| :- set_prolog_flag(language, iso). % for maximal compatibility | ||||
| library_directory('PLUNITDIR'). | ||||
| \end{code} | ||||
|  | ||||
| The current version runs under SICStus 3.  Open issues: | ||||
|  | ||||
| \begin{itemize} | ||||
|  | ||||
|     \item Some messages are unformatted because SICStus 3 reports | ||||
|           all ISO errors as instantiation errors. | ||||
|  | ||||
|     \item Only \file{plunit.pl}.  Both coverage analysis and the test | ||||
|     	  generation wizard currently require SWI-Prolog. | ||||
|  | ||||
|     \item The \const{load} option \const{normal} is the same as \const{always}. | ||||
| 	  Use \exam{set_test_options(load, never)} to avoid loading the | ||||
| 	  test suites. | ||||
|  | ||||
|     \item The \const{run} option is not supported. | ||||
|  | ||||
|     \item Tests are loaded into the enclosing module instead of a separate | ||||
|           test module. This means that predicates in the test module must | ||||
| 	  not conflict with the enclosing module, nor with other test | ||||
| 	  modules loaded into the same module. | ||||
| \end{itemize} | ||||
|  | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||||
| \section{Motivation of choices} | ||||
| \label{sec:motivation} | ||||
|  | ||||
| \subsection*{Easy to understand and flexible} | ||||
|  | ||||
| There are two approaches for testing. In one extreme the tests are | ||||
| written using declarations dealing with setup, cleanup, running and | ||||
| testing the result. In the other extreme a test is simply a Prolog goal | ||||
| that is supposed to succeed. We have chosen to allow for any mixture of | ||||
| these approaches. Written down as test/1 we opt for the simple | ||||
| succeeding goal approach. Using options to the test the user can choose | ||||
| for a more declarative specification.  The user can mix both approaches. | ||||
|  | ||||
| The body of the test appears at the position of a clause-body. This | ||||
| simplifies identification of the test body and ensures proper layout and | ||||
| colouring support from the editor without the need for explicit support | ||||
| of the unit test module. Only clauses of test/1 and test/2 may be marked | ||||
| as non-called in environments that perform cross-referencing. | ||||
|  | ||||
| %\subsection*{Well integrated} | ||||
|  | ||||
| \printindex | ||||
|  | ||||
| \end{document} | ||||
|  | ||||
							
								
								
									
										1510
									
								
								packages/plunit/plunit.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1510
									
								
								packages/plunit/plunit.pl
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										251
									
								
								packages/plunit/test_cover.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										251
									
								
								packages/plunit/test_cover.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,251 @@ | ||||
| /*  $Id$ | ||||
|  | ||||
|     Part of SWI-Prolog | ||||
|  | ||||
|     Author:        Jan Wielemaker | ||||
|     E-mail:        wielemak@science.uva.nl | ||||
|     WWW:           http://www.swi-prolog.org | ||||
|     Copyright (C): 1985-2006, University of Amsterdam | ||||
|  | ||||
|     This program is free software; you can redistribute it and/or | ||||
|     modify it under the terms of the GNU General Public License | ||||
|     as published by the Free Software Foundation; either version 2 | ||||
|     of the License, or (at your option) any later version. | ||||
|  | ||||
|     This program 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 General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public | ||||
|     License along with this library; if not, write to the Free Software | ||||
|     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  | ||||
|     As a special exception, if you link this library with other files, | ||||
|     compiled with a Free Software compiler, to produce an executable, this | ||||
|     library does not by itself cause the resulting executable to be covered | ||||
|     by the GNU General Public License. This exception does not however | ||||
|     invalidate any other reasons why the executable file might be covered by | ||||
|     the GNU General Public License. | ||||
| */ | ||||
|  | ||||
| :- module(prolog_cover, | ||||
| 	  [ show_coverage/1,		% :Goal | ||||
| 	    covered_clauses/4		% +Goal, -Result, -Succeeded, -Failed | ||||
| 	  ]). | ||||
| :- use_module(library(ordsets)). | ||||
|  | ||||
| :- set_prolog_flag(generate_debug_info, false). | ||||
|  | ||||
| /** <module> Clause cover analysis | ||||
|  | ||||
| The purpose of this module is to find which part of the program has been | ||||
| use by a certain goal. Usage is defined   in  terms of clauses that have | ||||
| fired, seperated in clauses that  succeeded   at  least once and clauses | ||||
| that failed on each occasion. | ||||
|  | ||||
| This module relies on the  SWI-Prolog   tracer  hooks. It modifies these | ||||
| hooks and collects the results, after   which  it restores the debugging | ||||
| environment.  This has some limitations: | ||||
|  | ||||
| 	* The performance degrades significantly (about 10 times) | ||||
| 	* It is not possible to use the debugger using coverage analysis | ||||
| 	* The cover analysis tool is currently not thread-safe. | ||||
|  | ||||
| The result is  represented  as  a   list  of  clause-references.  As the | ||||
| references to clauses of dynamic predicates  cannot be guaranteed, these | ||||
| are omitted from the result. | ||||
|  | ||||
| @bug	Relies heavily on SWI-Prolog internals. We have considered using | ||||
| 	a meta-interpreter for this purpose, but it is nearly impossible | ||||
| 	to do 100% complete meta-interpretation of Prolog.  Example | ||||
| 	problem areas include handling cuts in control-structures | ||||
| 	and calls from non-interpreted meta-predicates. | ||||
| */ | ||||
|  | ||||
|  | ||||
| :- dynamic | ||||
| 	entered/1,			% clauses entered | ||||
| 	exited/1.			% clauses completed | ||||
|  | ||||
| :- module_transparent | ||||
| 	covering/1, | ||||
| 	covering/4. | ||||
|  | ||||
| %%	show_coverage(Goal) | ||||
| % | ||||
| %	Report on coverage by Goal | ||||
|  | ||||
| show_coverage(Goal) :- | ||||
| 	covered_clauses(Goal, Result, Succeeded, Failed), | ||||
| 	file_coverage(Succeeded, Failed), | ||||
| 	return(Result). | ||||
|  | ||||
| return(true). | ||||
| return(fail) :- !, fail. | ||||
| return(error(E)) :- | ||||
| 	throw(E). | ||||
|  | ||||
| %%	covered_clauses(:Goal, -Result, -Succeeded, -Failed) is det. | ||||
| % | ||||
| %	Run Goal as once/1. Unify Result with   one of =true=, =fail= or | ||||
| %	error(Error). | ||||
| % | ||||
| %	@param	Succeeded Ordered set of succeeded clauses | ||||
| %	@param	Failed	  Ordered set of clauses that are entered but | ||||
| %			  never succeeded. | ||||
|  | ||||
| covered_clauses(Goal, Result, Succeeded, Failed) :- | ||||
| 	asserta(user:prolog_trace_interception(Port, Frame, _, continue) :- | ||||
| 			prolog_cover:assert_cover(Port, Frame), Ref), | ||||
| 	port_mask([unify,exit], Mask), | ||||
| 	'$visible'(V, Mask), | ||||
| 	'$leash'(L, Mask), | ||||
| 	trace, | ||||
| 	call_with_result(Goal, Result), | ||||
| 	set_prolog_flag(debug, false), | ||||
| 	covered(Ref, V, L, Succeeded, Failed). | ||||
|  | ||||
| %%	call_with_result(:Goal, -Result) is det. | ||||
| % | ||||
| %	Run Goal as once/1. Unify Result with   one of =true=, =fail= or | ||||
| %	error(Error). | ||||
|  | ||||
| call_with_result(Goal, Result) :- | ||||
| 	(   catch(Goal, E, true) | ||||
| 	->  (   var(E) | ||||
| 	    ->	Result = true | ||||
| 	    ;	Result = error(E) | ||||
| 	    ) | ||||
| 	;   Result = false | ||||
| 	). | ||||
|  | ||||
| port_mask([], 0). | ||||
| port_mask([H|T], Mask) :- | ||||
| 	port_mask(T, M0), | ||||
| 	'$syspreds':'$port_bit'(H, Bit),	% Private stuff | ||||
| 	Mask is M0 \/ Bit. | ||||
|  | ||||
| %%	assert_cover(+Port, +Frame) is det. | ||||
| % | ||||
| %	Assert coverage of the current clause. We monitor two ports: the | ||||
| %	_unify_ port to see which  clauses   we  entered, and the _exit_ | ||||
| %	port to see which completed successfully. | ||||
|  | ||||
| assert_cover(unify, Frame) :- | ||||
| 	running_static_pred(Frame), | ||||
| 	prolog_frame_attribute(Frame, clause, Cl), !, | ||||
| 	assert_entered(Cl). | ||||
| assert_cover(exit, Frame) :- | ||||
| 	running_static_pred(Frame), | ||||
| 	prolog_frame_attribute(Frame, clause, Cl), !, | ||||
| 	assert_exited(Cl). | ||||
| assert_cover(_, _). | ||||
|  | ||||
| %%	running_static_pred(+Frame) is semidet. | ||||
| % | ||||
| %	True if Frame is not running a dynamic predicate. | ||||
|  | ||||
| running_static_pred(Frame) :- | ||||
| 	prolog_frame_attribute(Frame, goal, Goal), | ||||
| 	\+ predicate_property(Goal, dynamic). | ||||
|  | ||||
| %%	assert_entered(+Ref) is det. | ||||
| %%	assert_exited(+Ref) is det. | ||||
| % | ||||
| %	Add Ref to the set of entered or exited	clauses. | ||||
|  | ||||
| assert_entered(Cl) :- | ||||
| 	entered(Cl), !. | ||||
| assert_entered(Cl) :- | ||||
| 	assert(entered(Cl)). | ||||
|  | ||||
| assert_exited(Cl) :- | ||||
| 	exited(Cl), !. | ||||
| assert_exited(Cl) :- | ||||
| 	assert(exited(Cl)). | ||||
|  | ||||
| %%	covered(+Ref, +VisibleMask, +LeashMask, -Succeeded, -Failed) is	det. | ||||
| % | ||||
| %	Restore state and collect failed and succeeded clauses. | ||||
|  | ||||
| covered(Ref, V, L, Succeeded, Failed) :- | ||||
| 	'$visible'(_, V), | ||||
| 	'$leash'(_, L), | ||||
| 	erase(Ref), | ||||
| 	findall(Cl, (entered(Cl), \+exited(Cl)), Failed0), | ||||
| 	findall(Cl, retract(exited(Cl)), Succeeded0), | ||||
| 	retractall(entered(Cl)), | ||||
| 	sort(Failed0, Failed), | ||||
| 	sort(Succeeded0, Succeeded). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	     REPORTING		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	file_coverage(+Succeeded, +Failed) is det. | ||||
| % | ||||
| %	Write a report on  the  clauses   covered  organised  by file to | ||||
| %	current output. | ||||
|  | ||||
| file_coverage(Succeeded, Failed) :- | ||||
| 	format('~N~n~`=t~78|~n'), | ||||
| 	format('~tCoverage by File~t~78|~n'), | ||||
| 	format('~`=t~78|~n'), | ||||
| 	format('~w~t~w~64|~t~w~72|~t~w~78|~n', | ||||
| 	       ['File', 'Clauses', '%Cov', '%Fail']), | ||||
| 	format('~`=t~78|~n'), | ||||
| 	forall(source_file(File), | ||||
| 	       file_coverage(File, Succeeded, Failed)), | ||||
| 	format('~`=t~78|~n'). | ||||
|  | ||||
| file_coverage(File, Succeeded, Failed) :- | ||||
| 	findall(Cl, clause_source(Cl, File, _), Clauses), | ||||
| 	sort(Clauses, All), | ||||
| 	(   ord_intersect(All, Succeeded) | ||||
| 	->  true | ||||
| 	;   ord_intersect(All, Failed) | ||||
| 	), !, | ||||
| 	ord_intersection(All, Failed, FailedInFile), | ||||
| 	ord_intersection(All, Succeeded, SucceededInFile), | ||||
| 	ord_subtract(All, SucceededInFile, UnCov1), | ||||
| 	ord_subtract(UnCov1, FailedInFile, Uncovered), | ||||
| 	length(All, AC), | ||||
| 	length(Uncovered, UC), | ||||
| 	length(FailedInFile, FC), | ||||
| 	CP is 100-100*UC/AC, | ||||
| 	FCP is 100*FC/AC, | ||||
| 	summary(File, 56, SFile), | ||||
| 	format('~w~t ~D~64| ~t~1f~72| ~t~1f~78|~n', [SFile, AC, CP, FCP]). | ||||
| file_coverage(_,_,_). | ||||
|  | ||||
|  | ||||
| summary(Atom, MaxLen, Summary) :- | ||||
| 	atom_length(Atom, Len), | ||||
| 	(   Len < MaxLen | ||||
| 	->  Summary = Atom | ||||
| 	;   SLen is MaxLen - 5, | ||||
| 	    sub_atom(Atom, _, SLen, 0, End), | ||||
| 	    atom_concat('...', End, Summary) | ||||
| 	). | ||||
|  | ||||
|  | ||||
| %%	clause_source(+Clause, -File, -Line) is det. | ||||
| %%	clause_source(-Clause, +File, -Line) is det. | ||||
|  | ||||
| clause_source(Clause, File, Line) :- | ||||
| 	nonvar(Clause), !, | ||||
| 	clause_property(Clause, file(File)), | ||||
| 	clause_property(Clause, line_count(Line)). | ||||
| clause_source(Clause, File, Line) :- | ||||
| 	source_file(Pred, File), | ||||
| 	\+ predicate_property(Pred, multifile), | ||||
| 	nth_clause(Pred, _Index, Clause), | ||||
| 	clause_property(Clause, line_count(Line)). | ||||
| clause_source(Clause, File, Line) :- | ||||
| 	Pred = _:_, | ||||
| 	predicate_property(Pred, multifile), | ||||
| 	nth_clause(Pred, _Index, Clause), | ||||
| 	clause_property(Clause, file(File)), | ||||
| 	clause_property(Clause, line_count(Line)). | ||||
							
								
								
									
										211
									
								
								packages/plunit/test_wizard.pl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								packages/plunit/test_wizard.pl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,211 @@ | ||||
| :- module(test_wizard, | ||||
| 	  [ make_tests/3,		% +Module, +File, +Out | ||||
| 	    make_test/3			% +Callable, -Module, -Test | ||||
| 	  ]). | ||||
| :- use_module(library(time)). | ||||
| :- use_module(library(lists)). | ||||
| :- use_module(library(listing)). | ||||
| :- use_module(library(readutil)). | ||||
|  | ||||
| /** <module> Test Generation Wizard | ||||
|  | ||||
| Tasks | ||||
|  | ||||
| 	* Accumulate user queries | ||||
| 	* Suggest tests from user queries | ||||
| */ | ||||
|  | ||||
| setting(max_time(5)). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	 UNIT GENERATION	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	make_tests(+Module, +File, +Out) is det. | ||||
| % | ||||
| %	Create tests from queries stored in File and write the tests for | ||||
| %	Module to the stream Out. | ||||
|  | ||||
| make_tests(Module, File, Out) :- | ||||
| 	read_file_to_terms(File, Queries, []), | ||||
| 	findall(Test, (   member(Q, Queries), | ||||
| 			  make_test(Q, Module, Test)), Tests), | ||||
| 	(   Tests == [] | ||||
| 	->  true | ||||
| 	;   format(Out, ':- begin_tests(~q).~n~n', [Module]), | ||||
| 	    maplist(portray_clause(Out), Tests), | ||||
| 	    format(Out, '~n:- end_tests(~q).~n', [Module]) | ||||
| 	). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	 TEST GENERATION	* | ||||
| 		 *******************************/ | ||||
|  | ||||
| %%	make_test(+Query:callable, -Module, -Test:term) is det. | ||||
| % | ||||
| %	Generate a test from a query. Test   is  returned as a clause of | ||||
| %	test/1  or  test/2  to  be   inserted  between  begin_tests  and | ||||
| %	end_tests. | ||||
|  | ||||
| make_test(Query0, Module, (test(Name, Options) :- Query)) :- | ||||
| 	find_test_module(Query0, Module, Query), | ||||
| 	pred_name(Query, Name), | ||||
| 	setting(max_time(Max)), | ||||
| 	test_result(Module:Query, Max, Options). | ||||
|  | ||||
| %%	find_test_module(+QuerySpec, ?Module, -Query). | ||||
| % | ||||
| %	Find module to test from a query. Note that it is very common | ||||
| %	for toplevel usage to rely on SWI-Prolog's DWIM. | ||||
| % | ||||
| %	@tbd	What if multiple modules match?  We can select the | ||||
| %		local one or ask the user. | ||||
|  | ||||
| find_test_module(Var, _, _) :- | ||||
| 	var(Var), !, fail. | ||||
| find_test_module(M:Query, M0, Query) :- !, | ||||
| 	M0 = M. | ||||
| find_test_module(Query, M, Query) :- | ||||
| 	current_predicate(_, M:Query), | ||||
| 	\+ predicate_property(M:Query, imported_from(_M2)). | ||||
|  | ||||
| %%	pred_name(+Callable, -Name) is det. | ||||
| % | ||||
| %	Suggest a name for the test. In   the  plunit framework the name | ||||
| %	needs not be unique, so we simply take the predicate name. | ||||
|  | ||||
| pred_name(Callable, Name) :- | ||||
| 	strip_module(Callable, _, Term), | ||||
| 	functor(Term, Name, _Arity). | ||||
|  | ||||
| %%	test_result(+Callable, +Maxtime, -Result) is det. | ||||
| % | ||||
| %	Try running goal and get meaningful results.  Results are: | ||||
| % | ||||
| %		* true(Templ == Var) | ||||
| %		* fail | ||||
| %		* all(Templ == Bindings) | ||||
| %		* throws(Error) | ||||
| %		* timeout | ||||
|  | ||||
| test_result(Callable, Maxtime, Result) :- | ||||
| 	term_variables(Callable, Vars), | ||||
| 	make_template(Vars, Templ), | ||||
| 	catch(call_with_time_limit(Maxtime, | ||||
| 				   findall(Templ-Det, | ||||
| 					   call_test(Callable, Det), | ||||
| 					   Bindings)), | ||||
| 	      E, true), | ||||
| 	(   var(E) | ||||
| 	->  success(Bindings, Templ, Result) | ||||
| 	;   error(E, Result) | ||||
| 	). | ||||
|  | ||||
| %%	success(+Bindings, +Templ, -Result) is det. | ||||
| % | ||||
| %	Create test-results from non-error cases. | ||||
|  | ||||
| success([], _, [fail]) :- !. | ||||
| success([[]-true],  _, []) :- !. | ||||
| success([S1-true],  Templ, [ true(Templ == S1) ]) :- !. | ||||
| success([[]-false], _, [ nondet ]) :- !. | ||||
| success([S1-false], Templ, [ true(Templ == S1), nondet ]) :- !. | ||||
| success(ListDet, Templ, [all(Templ == List)]) :- | ||||
| 	strip_det(ListDet, List). | ||||
|  | ||||
| strip_det([], []). | ||||
| strip_det([H-_|T0], [H|T]) :- | ||||
| 	strip_det(T0, T). | ||||
|  | ||||
| %%	error(+ErrorTerm, -Result) | ||||
|  | ||||
| error(Error0, [throws(Error)]) :- | ||||
| 	generalise_error(Error0, Error). | ||||
|  | ||||
|  | ||||
| generalise_error(error(Formal, _), error(Formal, _)) :- !. | ||||
| generalise_error(Term, Term). | ||||
|  | ||||
|  | ||||
| %%	make_template(+Vars, -Template) is det. | ||||
| % | ||||
| %	Make a nice looking template | ||||
|  | ||||
| make_template([], []) :- !. | ||||
| make_template([One], One) :- !. | ||||
| make_template([One, Two], One-Two) :- !. | ||||
| make_template(List, Vars) :- | ||||
| 	Vars =.. [v|List]. | ||||
|  | ||||
| %%	call_test(:Goal, -Det) is nondet. | ||||
| % | ||||
| %	True if Goal succeeded.  Det is unified to =true= if Goal left | ||||
| %	no choicepoints and =false= otherwise. | ||||
|  | ||||
| call_test(Goal, Det) :- | ||||
| 	Goal, | ||||
| 	deterministic(Det). | ||||
|  | ||||
|  | ||||
| 		 /******************************* | ||||
| 		 *	     COLLECT		* | ||||
| 		 *******************************/ | ||||
|  | ||||
| /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - | ||||
| Collect toplevel queries if the Prolog flag log_query_file points to the | ||||
| name of a writeable  file.  The  file   is  opened  in  append-mode  for | ||||
| exclusive write to allow for concurrent   operation from multiple Prolog | ||||
| systems using the same logfile. | ||||
|  | ||||
| The file is written in  UTF-8   encoding  and  using ignore_ops(true) to | ||||
| ensure it can be read. | ||||
| - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ | ||||
|  | ||||
| :- multifile | ||||
| 	user:message_hook/3. | ||||
|  | ||||
| user:message_hook(toplevel_goal(Goal0, Bindings), _Level, _Lines) :- | ||||
| 	open_query_log(Out), | ||||
| 	bind_vars(Bindings), | ||||
| 	clean_goal(Goal0, Goal), | ||||
| 	call_cleanup(format(Out, '~W.~n', [Goal, [ numbervars(true), | ||||
| 						   quoted(true), | ||||
| 						   ignore_ops(true) | ||||
| 						 ]]), close(Out)), | ||||
| 	fail. | ||||
|  | ||||
| clean_goal(Var, _) :- | ||||
| 	var(Var), !, fail. | ||||
| clean_goal(user:Goal, Goal) :- !. | ||||
| clean_goal(Goal, Goal). | ||||
|  | ||||
| bind_vars([]). | ||||
| bind_vars([Name=Var|T]) :- | ||||
| 	Var = '$VAR'(Name), | ||||
| 	bind_vars(T). | ||||
|  | ||||
| open_query_log(Out) :- | ||||
| 	current_prolog_flag(log_query_file, File), | ||||
| 	exists_file(File), !, | ||||
| 	open(File, append, Out, | ||||
| 	     [ encoding(utf8), | ||||
| 	       lock(write) | ||||
| 	     ]). | ||||
| open_query_log(Out) :- | ||||
| 	current_prolog_flag(log_query_file, File), | ||||
| 	access_file(File, write), !, | ||||
| 	open(File, write, Out, | ||||
| 	     [ encoding(utf8), | ||||
| 	       lock(write), | ||||
| 	       bom(true) | ||||
| 	     ]), | ||||
| 	format(Out, | ||||
| 	       '/* SWI-Prolog query log.  This file contains all syntactically\n   \ | ||||
| 	           correct queries issued in this directory.  It is used by the\n   \ | ||||
| 		   test wizard to generate unit tests.\n\ | ||||
| 		*/~n~n', []). | ||||
|  | ||||
|  | ||||
|  | ||||
		Reference in New Issue
	
	Block a user