This article DOES NOT include any material on dtx writing or l3build for dtx files.
Why we need l3build ?
when you are developing a complex package or class: there are many commands in it, while that some later commands may affect the previous ones.
Then you shoud make sure the previous commands still work after you add new commands. Thus, you need a test system.
when package(class) A relys on package(class) B, and you change something in B, you should make sure that A still works after the change. Thus, you need a test system to test functions in A as well.
Setup
build script
Create a file build.lua in the root directory of your project, which specifies the build process
write test file
Create your test files in testfiles directory, this folder can be set in build.lua by variable testfiledir, the default is ./testfiles. The default test file extention should be .lvt, but i change it to .tex in the following Example project.
how to enable test ?
Just add file regression-test.tex in your test file, then you can use the test commands provided by l3build. Some remarks about the regression-test.tex:
Test files are written in a TEX dialect using the support file regression-test.tex, which should be \input at the very beginning of each test. Additional customizations to this driver can be included in a local regression-test.cfg file, which will be loaded automatically if found.
Here is a test file template:
1 2 3 4 5 6
\input{regression-test.tex} \documentclass{article} \START % your test code here \ExplSyntaxOff \END
See later for more details about the test commands provided by l3build.
Warning: If you use the wrong syntax and using l3build save <test-file-name> to generate the *.tlg file in your testfile dir, then following l3build check will pass Even the commands is wrong.
commands for test
The l3build provides a series of commands for package or class writer to write test. Here i list some mostlt used commands:
\START ... \END: start and end the test
\SEPARATOR inserts a long line of = symbols to break up the log output
\TRUE, \FALSE, \YES, \NO output those strings to the log file.
\TEST{⟨title⟩}{⟨contents⟩} command runs its ⟨contents⟩ in a group and surrounds the generated log lines with some \SEPARATORs and a ⟨title⟩.
\TESTEXP{⟨title⟩}{⟨contents⟩} surrounds its ⟨contents⟩ with \TYPE and formatting to match \TEST;
\BEGINTEST{⟨title⟩} . . . \ENDTEST is an environment form of \TEST, allowing verbatim material, etc. to appear.
\ASSERT{⟨arg1⟩}{⟨arg2⟩} and \ASSERTSTR{⟨arg1⟩}{⟨arg2⟩} Test if the full expansion of ⟨arg1⟩ and ⟨arg2⟩ are the same: on a token basis in \ASSERT and on a string basis in \ASSERTSTR. Depending on the outcome, record either PASSED or FAILED in the .log.
Command \START tolds the test system that the useful part of the .log has started (for that there is lots of material in a raw log that you don’t want). and \END tells the system that the useful part of the .log has ended. (You can use \OMIT ... \TIMO to ignore parts of the log as well)
\TYPE is used to write material to the .log file, like LaTeX \typeout, but it allows long input.
This is a generated file for the l3build validation system.
Don't change this file in any respect.
> \box52= \hbox(6.94444+0.83333)x91.35481 .\OT1/cmr/m/n/10 h .\OT1/cmr/m/n/10 e .\OT1/cmr/m/n/10 l .\OT1/cmr/m/n/10 l .\OT1/cmr/m/n/10 o .\glue 3.33333 plus 1.66666 minus 1.11111 .\OT1/cmr/m/it/10 w .\OT1/cmr/m/it/10 o .\OT1/cmr/m/it/10 r .\OT1/cmr/m/it/10 l .\OT1/cmr/m/it/10 d .\kern 1.03334 .\glue 3.33333 plus 1.66666 minus 1.11111 .\mathon .\OML/cmm/m/it/10 a .\glue(\thickmuskip) 2.77771 plus 2.77771 .\OT1/cmr/m/n/10 = .\glue(\thickmuskip) 2.77771 plus 2.77771 .\OML/cmm/m/it/10 b .\glue(\medmuskip) 2.22217 plus 1.11108 minus 2.22217 .\OT1/cmr/m/n/10 + .\glue(\medmuskip) 2.22217 plus 1.11108 minus 2.22217 .\OML/cmm/m/it/10 c .\mathoff
! OK. <argument> \l_tmp_box
l.43 \box_show:N \l_tmp_box
END-TEST-LOG
) (see the transcript file for additional information) No pages of output.
test entire page
A simple example:
1 2 3 4 5 6 7
\input{regression-test.tex} \documentclass{article} \START \showoutput % Test content here \vfil\break \END
This code will generate a pdf, the output log:
1 2 3 4 5 6 7 8 9 10 11
START-TEST-LOG
This is a generated file for the l3build validation system.
Don't change this file in any respect.
END-TEST-LOG
) No pages of output.
output normalization
Eliminate the differences between different systems. Consist of 2 parts:
removing ignoring lines
Lines before the \START, after the \END and within \OMIT/\TIMO blocks
Entirely blank lines, including those consisting only of spaces.
Lines starting \openin or \openout.
lines loading *.fd files.
modify others to give consistent test output
Removal spaces at the start of lines
Removal of ./ at start of file names.
… etc
see the documentation for more details
test reference
There are many test goals (kinds), in the following example project, we use the log-based test. The default extention of your test file is .lvt.
l3build doc: A standard test will run the file ⟨name⟩.lvt using one or more engines, but will not carry out any additional processing.
What if we want a complex test goal ? See the l3build doc, it says that:
Regression tests check whether changes introduced in the code modify the test output. Especially while developing a complex package there is not yet a baseline to save a test goal with. It might then be easier to formulate the expected effects and outputs of tests directly. To achieve this, you may create an .lve instead of a .tlg file.
if the .tlg file not exsit, l3build check will throw an error:
1
Error: failed to find any reference or expectation file for test1!
How to setup the (default) test reference ? Create and Save your test reference by running:
1
l3build save <test-file-name>
Then a <test-file-name>.tlg file will be created in your testfiledir. The ⟨name⟩.tlg file is then used in all subsequent checks against the ⟨name⟩.lvt test.
run tests
Run the tests by running l3build check <tlg-file-name> or l3build check, then the test files will be compared with the saved reference
Use l3build clean to remove the test tempoaray files.
l3build check mechanism
file extention
.lvt(est): Extension of log-based test files. .tlg will be its reference or expectation file and will be generated. .lve(xpectation): Extension of auto-generating test file output, .pvt: Extension of PDF-based test files, .tpf will be its reference or expectation file and will be generated.
There are some different test systems examples below.
We don’t pack tests inside source files, like docstrip.
syntax base test system
l3build check will automatically check the syntax error, if the latter macro causes a syntax error. The output log will be like:
1 2 3 4 5
--- 3,6 ---- ============================================================ TEST 1: command~cmdA~Test ============================================================ ! Bad grouping: 2!
in the file test1.pdftex.diff as the before Test Example.
log base test system
The test system use the file *.tlg to check, that is, if the original log is:
1 2 3 4 5 6 7
This is a generated file for the l3build validation system. Don't change this file in any respect. ============================================================ TEST 1: LVTtest ============================================================ TRUE ============================================================
If we change it to:
1 2 3 4 5 6 7
This is a generated file for the l3build validation system. Don't change this file in any respect. ============================================================ TEST 1: LVTtest ============================================================ FALSE ============================================================
Then the l3build check will throw an check fail. To better understand the log basis test, there is a simple example:
This is a generated file for the l3build validation system. Don't change this file in any respect. ============================================================ TEST 1: LVTtest ============================================================ TRUE ============================================================
Absolutely, the check will fail, but if we change the LVTtest.lvt to:
> l3build check -c config-TU TUtest Running checks on TUtest (1/1) This is XeTeX, Version 3.141592653-2.6-0.999996 (TeX Live 2024/Arch Linux) (preloaded format=xelatex) restricted \write18 enabled. entering extended mode LaTeX2e <2023-11-01> patch level 1 L3 programming layer <2024-02-20> (./TUtest.lvt (/usr/share/texmf-dist/tex/latex/base/article.cls Document Class: article 2023/05/17 v1.4n Standard LaTeX document class (/usr/share/texmf-dist/tex/latex/base/size10.clo)) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec.sty (/usr/share/texmf-dist/tex/latex/l3packages/xparse/xparse.sty (/usr/share/texmf-dist/tex/latex/l3kernel/expl3.sty (/usr/share/texmf-dist/tex/latex/l3backend/l3backend-xetex.def))) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec-xetex.sty (/usr/share/texmf-dist/tex/latex/base/fontenc.sty) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec.cfg))) No file TUtest.aux. (/usr/share/texmf-dist/tex/latex/base/ts1cmr.fd) [1] (./TUtest.aux) ) Output written on TUtest.xdv (1 page, 500 bytes). Transcript written on TUtest.log. This is LuaHBTeX, Version 1.18.0 (TeX Live 2024/Arch Linux) restricted system commands enabled. LaTeX2e <2023-11-01> patch level 1 L3 programming layer <2024-02-20> (./TUtest.lvt (/usr/share/texmf-dist/tex/latex/base/article.cls Document Class: article 2023/05/17 v1.4n Standard LaTeX document class (/usr/share/texmf-dist/tex/latex/base/size10.clo)) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec.sty (/usr/share/texmf-dist/tex/latex/l3packages/xparse/xparse.sty (/usr/share/texmf-dist/tex/latex/l3kernel/expl3.sty (/usr/share/texmf-dist/tex/latex/l3backend/l3backend-luatex.def))) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec-luatex.sty (/usr/share/texmf-dist/tex/latex/base/fontenc.sty) (/usr/share/texmf-dist/tex/latex/fontspec/fontspec.cfg))) No file TUtest.aux. (/usr/share/texmf-dist/tex/latex/base/ts1cmr.fd) [1{/var/lib/texmf/fonts/map/pdftex/updmap/pdftex.map}] (./TUtest.aux)) 406 words of node memory still in use: 3 hlist, 1 vlist, 1 rule, 2 glue, 3 kern, 1 glyph, 4 attribute, 48 glue_spec, 4 attribute_list, 1 write nodes avail lists: 2:22,3:4,4:1,5:23,6:2,7:50,9:18 </usr/share/texmf-dist/fonts/opentype/public/lm/lmroman10-regular.otf> Output written on TUtest.pdf (1 page, 3288 bytes). Transcript written on TUtest.log.
All checks passed
or using command:
1 2 3 4 5
l3build check -c config-TU Running checks on TUtest (1/1)
All checks passed
Testing support
In the following , we will use ./ to denotes the root directory of the project, that is the directory contains the build.lua file.
introduction
Sometimes, your test needs some additional support files, like graphics, text file, package modules and so on. The case is that: l3build will usually copy the matching support files in ./support to the ./build/test/ directory in a flatten way. If the package you testing need the top level folder of these modules, like:
that is to remove the top level folder modules/ in the command. Then you can put the ztikzmodule.gnuplot.tex in the default ./support/ folder directly(need an additional copy) or specifies the folder contains this module(no copy needed) and set it as support file by:
(personal recommanded – tree kept) set the variable testuppdir to you folder the contains your modules and style(class) file, like ./code (no additional copy needed). Assume that your package/class(module) structure will be: