nixpkgs/pkgs/test/texlive/default.nix

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

982 lines
28 KiB
Nix
Raw Normal View History

{
lib,
stdenv,
buildEnv,
runCommand,
fetchurl,
file,
texlive,
writeShellScript,
writeText,
texliveInfraOnly,
texliveConTeXt,
texliveSmall,
texliveMedium,
texliveFull,
}:
2020-04-10 23:47:44 +00:00
2023-08-26 12:37:30 +00:00
rec {
mkTeXTest = lib.makeOverridable (
{
name,
format,
text,
texLive ? texliveSmall,
2023-08-26 12:37:30 +00:00
options ? "-interaction=errorstopmode",
preTest ? "",
postTest ? "",
...
}@attrs:
runCommand "texlive-test-tex-${name}"
(
{
nativeBuildInputs = [ texLive ] ++ attrs.nativeBuildInputs or [ ];
text = builtins.toFile "${name}.tex" text;
}
// builtins.removeAttrs attrs [
"nativeBuildInputs"
"text"
"texLive"
]
)
''
export HOME="$(mktemp -d)"
mkdir "$out"
cd "$out"
cp "$text" "$name.tex"
${preTest}
$format $options "$name.tex"
${postTest}
''
);
2023-03-08 22:37:01 +00:00
tlpdbNix =
runCommand "texlive-test-tlpdb-nix"
{
nixpkgsTlpdbNix = ../../tools/typesetting/tex/texlive/tlpdb.nix;
tlpdbNix = texlive.tlpdb.nix;
}
''
mkdir -p "$out"
diff -u "''${nixpkgsTlpdbNix}" "''${tlpdbNix}" | tee "$out/tlpdb.nix.patch"
'';
# test two completely different font discovery mechanisms, both of which were once broken:
# - lualatex uses its own luaotfload script (#220228)
# - xelatex uses fontconfig (#228196)
opentype-fonts = lib.recurseIntoAttrs rec {
lualatex = mkTeXTest {
name = "opentype-fonts-lualatex";
format = "lualatex";
texLive = texliveSmall.withPackages (ps: [ ps.libertinus-fonts ]);
text = ''
\documentclass{article}
\usepackage{fontspec}
\setmainfont{Libertinus Serif}
\begin{document}
\LaTeX{} is great
\end{document}
'';
};
xelatex = lualatex.override {
name = "opentype-fonts-xelatex";
format = "xelatex";
};
};
2023-03-08 22:37:01 +00:00
chktex =
runCommand "texlive-test-chktex"
{
2020-04-10 23:47:44 +00:00
nativeBuildInputs = [
(texlive.withPackages (ps: [ ps.chktex ]))
2020-04-10 23:47:44 +00:00
];
input = builtins.toFile "chktex-sample.tex" ''
\documentclass{article}
\begin{document}
\LaTeX is great
\end{document}
'';
}
''
# chktex is supposed to return 2 when it (successfully) finds warnings, but no errors,
# see http://git.savannah.nongnu.org/cgit/chktex.git/commit/?id=ec0fb9b58f02a62ff0bfec98b997208e9d7a5998
(set +e; chktex -v -nall -w1 "$input" 2>&1; [ $? = 2 ] || exit 1; set -e) | tee "$out"
# also check that the output does indeed contain "One warning printed"
2020-04-10 23:47:44 +00:00
grep "One warning printed" "$out"
'';
context = mkTeXTest {
name = "texlive-test-context";
format = "context";
texLive = texliveConTeXt;
# check that the PDF has been created: we have hit cases of context
# failing with exit status 0 due to a misconfigured texlive
postTest = ''
if [[ ! -f "$name".pdf ]] ; then
echo "ConTeXt test failed: file '$name.pdf' not found"
exit 1
fi
'';
text = ''
\starttext
\startsection[title={ConTeXt test document}]
This is an {\em incredibly} simple ConTeXt document.
\stopsection
\stoptext
'';
};
dvipng = lib.recurseIntoAttrs {
# https://github.com/NixOS/nixpkgs/issues/75605
basic =
runCommand "texlive-test-dvipng-basic"
{
nativeBuildInputs = [
file
texliveMedium
];
input = fetchurl {
name = "test_dvipng.tex";
url = "http://git.savannah.nongnu.org/cgit/dvipng.git/plain/test_dvipng.tex?id=b872753590a18605260078f56cbd6f28d39dc035";
sha256 = "1pjpf1jvwj2pv5crzdgcrzvbmn7kfmgxa39pcvskl4pa0c9hl88n";
};
}
''
cp "$input" ./document.tex
latex document.tex
dvipng -T tight -strict -picky document.dvi
for f in document*.png; do
file "$f" | tee output
grep PNG output
done
mkdir "$out"
mv document*.png "$out"/
'';
# test dvipng's limited capability to render postscript specials via GS
ghostscript =
runCommand "texlive-test-ghostscript"
{
nativeBuildInputs = [
file
(texliveSmall.withPackages (ps: [ ps.dvipng ]))
];
input = builtins.toFile "postscript-sample.tex" ''
\documentclass{minimal}
\begin{document}
Ni
\special{ps:
newpath
0 0 moveto
7 7 rlineto
0 7 moveto
7 -7 rlineto
stroke
showpage
}
\end{document}
'';
gs_trap = writeShellScript "gs_trap.sh" ''
exit 1
'';
}
''
cp "$gs_trap" ./gs
export PATH=$PWD:$PATH
# check that the trap works
gs && exit 1
cp "$input" ./document.tex
latex document.tex
dvipng -T 1in,1in -strict -picky document.dvi
for f in document*.png; do
file "$f" | tee output
grep PNG output
done
mkdir "$out"
mv document*.png "$out"/
'';
};
2020-04-10 23:47:44 +00:00
# https://github.com/NixOS/nixpkgs/issues/75070
dvisvgm =
runCommand "texlive-test-dvisvgm"
{
nativeBuildInputs = [
file
texliveMedium
];
2020-04-10 23:47:44 +00:00
input = builtins.toFile "dvisvgm-sample.tex" ''
\documentclass{article}
\begin{document}
mwe
\end{document}
'';
}
''
cp "$input" ./document.tex
2020-04-10 23:47:44 +00:00
latex document.tex
dvisvgm document.dvi -n -o document_dvi.svg
cat document_dvi.svg
file document_dvi.svg | grep SVG
2020-04-10 23:47:44 +00:00
pdflatex document.tex
dvisvgm -P document.pdf -n -o document_pdf.svg
cat document_pdf.svg
file document_pdf.svg | grep SVG
2020-04-10 23:47:44 +00:00
mkdir "$out"
mv document*.svg "$out"/
'';
texdoc =
runCommand "texlive-test-texdoc"
{
2021-06-24 02:16:37 +00:00
nativeBuildInputs = [
2024-08-16 21:15:23 +00:00
(texlive.withPackages (ps: [
ps.luatex
ps.texdoc
ps.texdoc.texdoc
]))
2021-06-24 02:16:37 +00:00
];
}
''
texdoc --version
2021-06-24 02:16:37 +00:00
texdoc --debug --list texdoc | tee "$out"
grep texdoc.pdf "$out"
'';
2021-07-04 10:47:33 +00:00
2023-08-26 12:37:30 +00:00
# check that the default language is US English
defaultLanguage = lib.recurseIntoAttrs rec {
# language.def
etex = mkTeXTest {
name = "default-language-etex";
format = "etex";
text = ''
\catcode`\@=11
\ifnum\language=\lang@USenglish \message{[tests.texlive] Default language is US English.}
\else\errmessage{[tests.texlive] Error: default language is NOT US English.}\fi
\ifnum\language=0\message{[tests.texlive] Default language has id 0.}
\else\errmessage{[tests.texlive] Error: default language does NOT have id 0.}\fi
\bye
'';
};
# language.dat
latex = mkTeXTest {
name = "default-language-latex";
format = "latex";
text = ''
\makeatletter
\ifnum\language=\l@USenglish \GenericWarning{}{[tests.texlive] Default language is US English}
\else\GenericError{}{[tests.texlive] Error: default language is NOT US English}{}{}\fi
\ifnum\language=0\GenericWarning{}{[tests.texlive] Default language has id 0}
\else\GenericError{}{[tests.texlive] Error: default language does NOT have id 0}{}{}\fi
\stop
'';
};
# language.dat.lua
luatex = etex.override {
name = "default-language-luatex";
format = "luatex";
};
};
2023-08-26 12:37:30 +00:00
# check that all languages are available, including synonyms
allLanguages =
let
hyphenBase = texlive.pkgs.hyphen-base;
texLive = texliveFull;
in
2023-08-26 12:37:30 +00:00
lib.recurseIntoAttrs {
# language.def
etex = mkTeXTest {
name = "all-languages-etex";
format = "etex";
inherit hyphenBase texLive;
text = ''
\catcode`\@=11
\input kvsetkeys.sty
\def\CheckLang#1{
\ifcsname lang@#1\endcsname\message{[tests.texlive] Found language #1}
\else\errmessage{[tests.texlive] Error: missing language #1}\fi
}
\comma@parse{@texLanguages@}\CheckLang
\bye
'';
preTest = ''
texLanguages="$(sed -n -E 's/^\\addlanguage\s*\{([^}]+)\}.*$/\1/p' < "$hyphenBase"/tex/generic/config/language.def)"
texLanguages="''${texLanguages//$'\n'/,}"
substituteInPlace "$name.tex" --subst-var texLanguages
'';
};
# language.dat
latex = mkTeXTest {
name = "all-languages-latex";
format = "latex";
inherit hyphenBase texLive;
text = ''
\makeatletter
\@for\Lang:=italian,@texLanguages@\do{
\ifcsname l@\Lang\endcsname
\GenericWarning{}{[tests.texlive] Found language \Lang}
\else
\GenericError{}{[tests.texlive] Error: missing language \Lang}{}{}
\fi
}
\stop
'';
preTest = ''
texLanguages="$(sed -n -E 's/^([^%= \t]+).*$/\1/p' < "$hyphenBase"/tex/generic/config/language.dat)"
texLanguages="''${texLanguages//$'\n'/,}"
substituteInPlace "$name.tex" --subst-var texLanguages
'';
};
# language.dat.lua
luatex = mkTeXTest {
name = "all-languages-luatex";
format = "luatex";
inherit hyphenBase texLive;
text = ''
\directlua{
require('luatex-hyphen.lua')
langs = '@texLanguages@,'
texio.write('\string\n')
for l in langs:gmatch('([^,]+),') do
if luatexhyphen.lookupname(l) \string~= nil then
texio.write('[tests.texlive] Found language '..l..'.\string\n')
else
error('[tests.texlive] Error: missing language '..l..'.', 2)
end
end
}
\bye
'';
preTest = ''
texLanguages="$(sed -n -E 's/^.*\[("|'\''')(.*)("|'\''')].*$/\2/p' < "$hyphenBase"/tex/generic/config/language.dat.lua)"
texLanguages="''${texLanguages//$'\n'/,}"
substituteInPlace "$name.tex" --subst-var texLanguages
'';
};
};
2021-07-04 10:47:33 +00:00
# test that language files are generated as expected
hyphen-base =
runCommand "texlive-test-hyphen-base"
{
hyphenBase = texlive.pkgs.hyphen-base;
schemeFull = texliveFull;
schemeInfraOnly = texliveInfraOnly;
2021-07-04 10:47:33 +00:00
}
''
mkdir -p "$out"/{scheme-infraonly,scheme-full}
2021-07-04 10:47:33 +00:00
# create language files with no hyphenation patterns
cat "$hyphenBase"/tex/generic/config/language.us >language.dat
cat "$hyphenBase"/tex/generic/config/language.us.def >language.def
cat "$hyphenBase"/tex/generic/config/language.us.lua >language.dat.lua
2021-07-04 10:47:33 +00:00
cat >>language.dat.lua <<EOF
}
EOF
2021-07-04 10:47:33 +00:00
cat >>language.def <<EOF
%%% No changes may be made beyond this point.
2021-07-04 10:47:33 +00:00
\uselanguage {USenglish} %%% This MUST be the last line of the file.
EOF
2021-07-04 10:47:33 +00:00
for fname in language.{dat,def,dat.lua} ; do
diff --ignore-matching-lines='^\(%\|--\) Generated by ' -u \
{"$hyphenBase","$schemeFull"/share/texmf-var}/tex/generic/config/"$fname" \
2021-07-04 10:47:33 +00:00
| tee "$out/scheme-full/$fname.patch"
diff --ignore-matching-lines='^\(%\|--\) Generated by ' -u \
{,"$schemeInfraOnly"/share/texmf-var/tex/generic/config/}"$fname" \
2021-07-04 10:47:33 +00:00
| tee "$out/scheme-infraonly/$fname.patch"
done
'';
2021-07-12 19:37:51 +00:00
# verify that the restricted mode gets enabled when
# needed (detected by checking if it disallows --gscmd)
repstopdf =
runCommand "texlive-test-repstopdf"
{
nativeBuildInputs = [ (texlive.withPackages (ps: [ ps.epstopdf ])) ];
}
''
! (epstopdf --gscmd echo /dev/null 2>&1 || true) | grep forbidden >/dev/null
(repstopdf --gscmd echo /dev/null 2>&1 || true) | grep forbidden >/dev/null
mkdir "$out"
'';
# verify that the restricted mode gets enabled when
# needed (detected by checking if it disallows --gscmd)
rpdfcrop =
runCommand "texlive-test-rpdfcrop"
{
nativeBuildInputs = [ (texlive.withPackages (ps: [ ps.pdfcrop ])) ];
}
''
! (pdfcrop --gscmd echo $(command -v pdfcrop) 2>&1 || true) | grep 'restricted mode' >/dev/null
(rpdfcrop --gscmd echo $(command -v pdfcrop) 2>&1 || true) | grep 'restricted mode' >/dev/null
mkdir "$out"
'';
2023-06-28 22:06:06 +00:00
# check that all binaries run successfully, in the following sense:
# (1) run --version, -v, --help, -h successfully; or
# (2) run --help, -h, or no argument with error code but show help text; or
# (3) run successfully on a test.tex or similar file
# we ignore the binaries that cannot be tested as above, and are either
# compiled binaries or trivial shell wrappers
binaries =
let
# TODO known broken binaries
broken = [
# *.inc files in source container rather than run
"texaccents"
# 'Error initialising QuantumRenderer: no suitable pipeline found'
"tlcockpit"
] ++ lib.optional stdenv.hostPlatform.isDarwin "epspdftk"; # wish shebang is a script, not a binary!
2023-06-28 22:06:06 +00:00
# (1) binaries requiring -v
shortVersion = [
"devnag"
"diadia"
"pmxchords"
"ptex2pdf"
"simpdftex"
"ttf2afm"
];
# (1) binaries requiring --help or -h
help = [
"arlatex"
"bundledoc"
"cachepic"
"checklistings"
"dvipos"
"extractres"
"fig4latex"
"fragmaster"
2024-10-28 01:29:19 +00:00
"kpsewhere"
"latex-git-log"
"ltxfileinfo"
"mendex"
"pdflatexpicscale"
"perltex"
"pn2pdf"
"psbook"
"psnup"
"psresize"
"purifyeps"
2023-08-28 17:27:22 +00:00
"simpdftex"
"tex2xindy"
"texluac"
"texluajitc"
"upmendex"
"urlbst"
"yplan"
];
2024-10-28 01:29:19 +00:00
shortHelp = [
"adhocfilelist"
"authorindex"
"bbl2bib"
"bibdoiadd"
"bibmradd"
"biburl2doi"
"bibzbladd"
"bookshelf-listallfonts"
"bookshelf-mkfontsel"
"ctanupload"
"disdvi"
"dvibook"
"dviconcat"
"getmapdl"
"latex2man"
"listings-ext.sh"
"pygmentex"
];
2023-06-28 22:06:06 +00:00
# (2) binaries that return non-zero exit code even if correctly asked for help
2024-10-28 01:29:19 +00:00
ignoreExitCode = [
"authorindex"
"bookshelf-listallfonts"
"bookshelf-mkfontsel"
"dvibook"
"dviconcat"
"dvipos"
"extractres"
"fig4latex"
"fragmaster"
"latex2man"
"latex-git-log"
"listings-ext.sh"
"psbook"
"psnup"
"psresize"
"purifyeps"
"tex2xindy"
"texluac"
"texluajitc"
];
2023-06-28 22:06:06 +00:00
# (2) binaries that print help on no argument, returning non-zero exit code
noArg = [
"a2ping"
"bg5+latex"
"bg5+pdflatex"
"bg5latex"
"bg5pdflatex"
"cef5latex"
"cef5pdflatex"
"ceflatex"
"cefpdflatex"
"cefslatex"
"cefspdflatex"
"chkdvifont"
"dvi2fax"
"dvired"
"dviselect"
"dvitodvi"
"epsffit"
"findhyph"
"gbklatex"
"gbkpdflatex"
"komkindex"
"kpsepath"
"listbib"
"listings-ext"
"mag"
"mathspic"
"mf2pt1"
"mk4ht"
"mkt1font"
"mkgrkindex"
"musixflx"
"pdf2ps"
"pdfclose"
"pdftosrc"
"pdfxup"
"pedigree"
"pfb2pfa"
"pk2bm"
"prepmx"
"ps2pk"
"psselect"
"pstops"
"rubibtex"
"rubikrotation"
"sjislatex"
"sjispdflatex"
"srcredact"
"t4ht"
"teckit_compile"
"tex4ht"
"texdiff"
"texdirflatten"
"texplate"
"tie"
"ttf2kotexfont"
"ttfdump"
"vlna"
"vpl2ovp"
"vpl2vpl"
"yplan"
];
# (3) binaries requiring a .tex file
contextTest = [ "htcontext" ];
latexTest = [
"de-macro"
"e2pall"
"htlatex"
"htxelatex"
"makeindex"
"pslatex"
"rumakeindex"
"tpic2pdftex"
"wordcount"
"xhlatex"
];
texTest = [
"fontinst"
"htmex"
"httex"
"httexi"
"htxetex"
];
2023-06-28 22:06:06 +00:00
# tricky binaries or scripts that are obviously working but are hard to test
# (e.g. because they expect user input no matter the arguments)
# (printafm comes from ghostscript, not texlive)
ignored = [
# compiled binaries
"dt2dv"
"dv2dt"
"dvi2tty"
"dvidvi"
"dvispc"
"otp2ocp"
"outocp"
"pmxab"
# GUI scripts that accept no argument or crash without a graphics server; please test manualy
"epspdftk"
"texdoctk"
"tlshell"
"xasy"
# requires Cinderella, not open source and not distributed via Nixpkgs
"ketcindy"
];
# binaries that need a combined scheme and cannot work standalone
needScheme = [
# pfarrei: require working kpse to find lua module
"a5toa4"
# bibexport: requires kpsewhich
"bibexport"
# crossrefware: require bibtexperllibs under TEXMFROOT
"bbl2bib"
"bibdoiadd"
"bibmradd"
"biburl2doi"
"bibzbladd"
"checkcites"
"ltx2crossrefxml"
# epstopdf: requires kpsewhich
"epstopdf"
"repstopdf"
# requires kpsewhich
"memoize-extract.pl"
"memoize-extract.py"
# require other texlive binaries in PATH
"allcm"
"allec"
"chkweb"
"fontinst"
"ht*"
"installfont-tl"
"kanji-config-updmap-sys"
"kanji-config-updmap-user"
"kpse*"
"latexfileversion"
"mkocp"
"mkofm"
"mtxrunjit"
"pdftex-quiet"
"pslatex"
"rumakeindex"
"texconfig"
"texconfig-sys"
"texexec"
"texlinks"
"texmfstart"
"typeoutfileinfo"
"wordcount"
"xdvi"
"xhlatex"
# misc luatex binaries searching for luatex in PATH
"citeproc-lua"
"context"
"contextjit"
"ctanbib"
"digestif"
"epspdf"
"l3build"
"luafindfont"
"luaotfload-tool"
"luatools"
"make4ht"
"pmxchords"
"tex4ebook"
"texblend"
"texdoc"
"texfindpkg"
"texlogsieve"
"xindex"
# requires full TEXMFROOT (e.g. for config)
"mktexfmt"
"mktexmf"
"mktexpk"
"mktextfm"
"psnup"
"psresize"
"pstops"
"tlmgr"
"updmap"
"webquiz"
# texlive-scripts: requires texlive.infra's TeXLive::TLUtils under TEXMFROOT
"fmtutil"
"fmtutil-sys"
"fmtutil-user"
# texlive-scripts: not used in nixpkgs, need updmap in PATH
"updmap-sys"
"updmap-user"
];
# simple test files
contextTestTex = writeText "context-test.tex" ''
\starttext
A simple test file.
\stoptext
'';
latexTestTex = writeText "latex-test.tex" ''
2023-06-28 22:06:06 +00:00
\documentclass{article}
\begin{document}
A simple test file.
\end{document}
'';
texTestTex = writeText "tex-test.tex" ''
Hello.
\bye
'';
# link all binaries in single derivation
binPackages = lib.catAttrs "out" (lib.attrValues texlive.pkgs);
binaries = buildEnv {
name = "texlive-binaries";
paths = binPackages;
};
2023-06-28 22:06:06 +00:00
in
runCommand "texlive-test-binaries"
{
inherit
binaries
contextTestTex
latexTestTex
texTestTex
;
texliveScheme = texliveFull;
}
2023-06-28 22:06:06 +00:00
''
loadables="$(command -v bash)"
loadables="''${loadables%/bin/bash}/lib/bash"
enable -f "$loadables/realpath" realpath
2023-06-28 22:06:06 +00:00
mkdir -p "$out"
export HOME="$(mktemp -d)"
declare -i binCount=0 ignoredCount=0 brokenCount=0 failedCount=0
cp "$contextTestTex" context-test.tex
cp "$latexTestTex" latex-test.tex
cp "$texTestTex" tex-test.tex
2023-06-28 22:06:06 +00:00
testBin () {
path="$(realpath "$bin")"
path="''${path##*/}"
2023-06-28 22:06:06 +00:00
if [[ -z "$ignoreExitCode" ]] ; then
PATH="$path" "$bin" $args >"$out/$base.log" 2>&1
ret=$?
if [[ $ret == 0 ]] && grep -i 'command not found' "$out/$base.log" >/dev/null ; then
echo "command not found when running '$base''${args:+ $args}'"
return 1
fi
return $ret
2023-06-28 22:06:06 +00:00
else
PATH="$path" "$bin" $args >"$out/$base.log" 2>&1
2023-06-28 22:06:06 +00:00
ret=$?
if [[ $ret == 0 ]] && grep -i 'command not found' "$out/$base.log" >/dev/null ; then
echo "command not found when running '$base''${args:+ $args}'"
return 1
fi
2023-06-28 22:06:06 +00:00
if ! grep -Ei '(Example:|Options:|Syntax:|Usage:|improper command|SYNOPSIS)' "$out/$base.log" >/dev/null ; then
echo "did not find usage info when running '$base''${args:+ $args}'"
return $ret
fi
fi
}
for bin in "$binaries"/bin/* ; do
2023-06-28 22:06:06 +00:00
base="''${bin##*/}"
args=
ignoreExitCode=
binCount=$((binCount + 1))
# ignore non-executable files (such as context.lua)
if [[ ! -x "$bin" ]] ; then
ignoredCount=$((ignoredCount + 1))
continue
fi
2023-06-28 22:06:06 +00:00
case "$base" in
${lib.concatStringsSep "|" ignored})
ignoredCount=$((ignoredCount + 1))
continue ;;
${lib.concatStringsSep "|" broken})
brokenCount=$((brokenCount + 1))
continue ;;
${lib.concatStringsSep "|" help})
args=--help ;;
${lib.concatStringsSep "|" shortHelp})
args=-h ;;
${lib.concatStringsSep "|" noArg})
;;
${lib.concatStringsSep "|" contextTest})
args=context-test.tex ;;
${lib.concatStringsSep "|" latexTest})
args=latex-test.tex ;;
${lib.concatStringsSep "|" texTest})
args=tex-test.tex ;;
2023-06-28 22:06:06 +00:00
${lib.concatStringsSep "|" shortVersion})
args=-v ;;
ebong)
touch empty
args=empty ;;
ht)
args='latex latex-test.tex' ;;
2023-06-28 22:06:06 +00:00
pdf2dsc)
args='--help --help --help' ;;
typeoutfileinfo)
args=/dev/null ;;
*)
args=--version ;;
esac
case "$base" in
${lib.concatStringsSep "|" (ignoreExitCode ++ noArg)})
ignoreExitCode=1 ;;
esac
case "$base" in
${lib.concatStringsSep "|" needScheme})
bin="$texliveScheme/bin/$base"
if [[ ! -f "$bin" ]] ; then
ignoredCount=$((ignoredCount + 1))
continue
fi ;;
esac
2023-06-28 22:06:06 +00:00
if testBin ; then : ; else # preserve exit code
echo "failed '$base''${args:+ $args}' (exit code: $?)"
sed 's/^/ > /' < "$out/$base.log"
2023-06-28 22:06:06 +00:00
failedCount=$((failedCount + 1))
fi
done
echo "tested $binCount binaries: $ignoredCount ignored, $brokenCount broken, $failedCount failed"
2023-06-28 22:06:06 +00:00
[[ $failedCount = 0 ]]
'';
2023-06-30 00:07:49 +00:00
# check that all scripts have a Nix shebang
shebangs =
let
binPackages = lib.catAttrs "out" (lib.attrValues texlive.pkgs);
in
runCommand "texlive-test-shebangs" { } (
''
echo "checking that all texlive scripts shebangs are in '$NIX_STORE'"
declare -i scriptCount=0 invalidCount=0
''
+ (lib.concatMapStrings (pkg: ''
for bin in '${pkg.outPath}'/bin/* ; do
grep -I -q . "$bin" || continue # ignore binary files
[[ -x "$bin" ]] || continue # ignore non-executable files (such as context.lua)
scriptCount=$((scriptCount + 1))
read -r cmdline < "$bin"
read -r interp <<< "$cmdline"
if [[ "$interp" != "#!$NIX_STORE"/* && "$interp" != "#! $NIX_STORE"/* ]] ; then
echo "error: non-nix shebang '$interp' in script '$bin'"
invalidCount=$((invalidCount + 1))
fi
done
'') binPackages)
+ ''
echo "checked $scriptCount scripts, found $invalidCount non-nix shebangs"
[[ $invalidCount -gt 0 ]] && exit 1
mkdir -p "$out"
''
);
2023-06-30 00:07:49 +00:00
# verify that the precomputed licensing information in default.nix
# does indeed match the metadata of the individual packages.
#
# This is part of the test suite (and not the normal evaluation) to save
# time for "normal" evaluations. To be more in line with the other tests, this
# also builds a derivation, even though it is essentially an eval-time assertion.
licenses =
let
concatLicenses = builtins.foldl' (acc: el: if builtins.elem el acc then acc else acc ++ [ el ]);
# converts a license to its attribute name in lib.licenses
licenseToAttrName =
license: builtins.head (builtins.attrNames (lib.filterAttrs (n: v: license == v) lib.licenses));
lt = (a: b: a < b);
2023-06-30 00:07:49 +00:00
savedLicenses = scheme: scheme.meta.license;
savedLicensesAttrNames = scheme: map licenseToAttrName (savedLicenses scheme);
2023-06-30 00:07:49 +00:00
correctLicenses =
scheme:
builtins.foldl' (
acc: pkg: concatLicenses acc (lib.toList (pkg.meta.license or [ ]))
) [ ] scheme.passthru.requiredTeXPackages;
2023-06-30 00:07:49 +00:00
correctLicensesAttrNames = scheme: lib.sort lt (map licenseToAttrName (correctLicenses scheme));
2023-06-30 00:07:49 +00:00
hasLicenseMismatch =
scheme:
(lib.isDerivation scheme) && (savedLicensesAttrNames scheme) != (correctLicensesAttrNames scheme);
2023-11-04 20:02:25 +00:00
incorrectSchemes = lib.filterAttrs (n: hasLicenseMismatch) (texlive.combined // texlive.schemes);
2023-06-30 00:07:49 +00:00
prettyPrint = name: scheme: ''
license info for ${name} is incorrect! Note that order is enforced.
saved: [ ${lib.concatStringsSep " " (savedLicensesAttrNames scheme)} ]
correct: [ ${lib.concatStringsSep " " (correctLicensesAttrNames scheme)} ]
'';
2023-06-30 00:07:49 +00:00
errorText = lib.concatStringsSep "\n\n" (lib.mapAttrsToList prettyPrint incorrectSchemes);
in
2023-06-30 00:07:49 +00:00
runCommand "texlive-test-license"
{
inherit errorText;
}
(
2023-06-30 00:07:49 +00:00
if (incorrectSchemes == { }) then
"echo everything is fine! > $out"
else
2023-06-30 00:07:49 +00:00
''
echo "$errorText"
false
''
);
# verify that all fixed hashes are present
# this is effectively an eval-time assertion, converted into a derivation for
# ease of testing
2024-08-16 21:15:23 +00:00
fixedHashes =
let
fods = lib.concatMap (
2024-08-16 21:15:23 +00:00
p:
lib.optional (p ? tex && lib.isDerivation p.tex) p.tex
++ lib.optional (p ? texdoc) p.texdoc
++ lib.optional (p ? texsource) p.texsource
++ lib.optional (p ? tlpkg) p.tlpkg
2024-08-16 21:15:23 +00:00
) (lib.attrValues texlive.pkgs);
errorText = lib.concatMapStrings (
p:
lib.optionalString (
!p ? outputHash
) "${p.pname}-${p.tlOutputName} does not have a fixed output hash\n"
) fods;
in
runCommand "texlive-test-fixed-hashes"
{
inherit errorText;
passAsFile = [ "errorText" ];
}
''
if [[ -s "$errorTextPath" ]] ; then
cat "$errorTextPath"
echo Failed: some TeX Live packages do not have fixed output hashes. Please read UPGRADING.md for how to generate a new fixed-hashes.nix.
exit 1
else
touch "$out"
fi
'';
2020-04-10 23:47:44 +00:00
}