2023-07-10 22:28:36 +02:00
|
|
|
{ lib, stdenv, buildEnv, runCommand, fetchurl, file, texlive, writeShellScript, writeText }:
|
2020-04-11 01:47:44 +02:00
|
|
|
|
2023-08-26 14:37:30 +02:00
|
|
|
rec {
|
|
|
|
|
|
|
|
mkTeXTest = lib.makeOverridable (
|
|
|
|
{ name
|
|
|
|
, format
|
|
|
|
, text
|
|
|
|
, texLive ? texlive.combined.scheme-small
|
|
|
|
, 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 23:37:01 +01:00
|
|
|
|
2023-04-22 21:22:39 +02:00
|
|
|
tlpdbNix = runCommand "texlive-test-tlpdb-nix" {
|
2023-03-12 22:04:39 +01:00
|
|
|
nixpkgsTlpdbNix = ../../tools/typesetting/tex/texlive/tlpdb.nix;
|
2023-04-22 21:22:39 +02:00
|
|
|
tlpdbNix = texlive.tlpdb.nix;
|
2023-03-12 22:04:39 +01:00
|
|
|
}
|
|
|
|
''
|
|
|
|
mkdir -p "$out"
|
|
|
|
diff -u "''${nixpkgsTlpdbNix}" "''${tlpdbNix}" | tee "$out/tlpdb.nix.patch"
|
|
|
|
'';
|
|
|
|
|
2023-08-26 14:37:30 +02:00
|
|
|
# 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 = texlive.combine { inherit (texlive) scheme-medium 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 23:37:01 +01:00
|
|
|
|
2021-08-15 17:12:23 +02:00
|
|
|
chktex = runCommand "texlive-test-chktex" {
|
2020-04-11 01:47:44 +02:00
|
|
|
nativeBuildInputs = [
|
2023-08-26 14:37:30 +02:00
|
|
|
(texlive.combine { inherit (texlive) scheme-infraonly chktex; })
|
2020-04-11 01:47:44 +02:00
|
|
|
];
|
|
|
|
input = builtins.toFile "chktex-sample.tex" ''
|
|
|
|
\documentclass{article}
|
|
|
|
\begin{document}
|
|
|
|
\LaTeX is great
|
|
|
|
\end{document}
|
|
|
|
'';
|
|
|
|
} ''
|
|
|
|
chktex -v -nall -w1 "$input" 2>&1 | tee "$out"
|
|
|
|
grep "One warning printed" "$out"
|
|
|
|
'';
|
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
dvipng = lib.recurseIntoAttrs {
|
|
|
|
# https://github.com/NixOS/nixpkgs/issues/75605
|
2021-08-15 17:12:23 +02:00
|
|
|
basic = runCommand "texlive-test-dvipng-basic" {
|
2021-05-03 01:35:47 +02:00
|
|
|
nativeBuildInputs = [ file texlive.combined.scheme-medium ];
|
|
|
|
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
|
2020-04-11 01:47:44 +02:00
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
latex document.tex
|
|
|
|
dvipng -T tight -strict -picky document.dvi
|
|
|
|
for f in document*.png; do
|
|
|
|
file "$f" | tee output
|
|
|
|
grep PNG output
|
|
|
|
done
|
2020-04-11 01:47:44 +02:00
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
mkdir "$out"
|
|
|
|
mv document*.png "$out"/
|
2021-03-19 01:55:00 +01:00
|
|
|
'';
|
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
# test dvipng's limited capability to render postscript specials via GS
|
2021-08-15 17:12:23 +02:00
|
|
|
ghostscript = runCommand "texlive-test-ghostscript" {
|
2023-08-26 14:37:30 +02:00
|
|
|
nativeBuildInputs = [ file (texlive.combine { inherit (texlive) scheme-small dvipng; }) ];
|
2021-05-03 01:35:47 +02:00
|
|
|
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
|
2021-03-19 01:55:00 +01:00
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
cp "$input" ./document.tex
|
2021-03-19 01:55:00 +01:00
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
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
|
2021-03-19 01:55:00 +01:00
|
|
|
|
2021-05-03 01:35:47 +02:00
|
|
|
mkdir "$out"
|
|
|
|
mv document*.png "$out"/
|
|
|
|
'';
|
|
|
|
};
|
2021-03-19 01:55:00 +01:00
|
|
|
|
2020-04-11 01:47:44 +02:00
|
|
|
# https://github.com/NixOS/nixpkgs/issues/75070
|
2021-08-15 17:12:23 +02:00
|
|
|
dvisvgm = runCommand "texlive-test-dvisvgm" {
|
2020-04-11 01:47:44 +02:00
|
|
|
nativeBuildInputs = [ file texlive.combined.scheme-medium ];
|
|
|
|
input = builtins.toFile "dvisvgm-sample.tex" ''
|
|
|
|
\documentclass{article}
|
|
|
|
\begin{document}
|
|
|
|
mwe
|
|
|
|
\end{document}
|
|
|
|
'';
|
|
|
|
} ''
|
|
|
|
cp "$input" ./document.tex
|
|
|
|
|
|
|
|
latex document.tex
|
|
|
|
dvisvgm document.dvi -n -o document_dvi.svg
|
|
|
|
cat document_dvi.svg
|
|
|
|
file document_dvi.svg | grep SVG
|
|
|
|
|
|
|
|
pdflatex document.tex
|
|
|
|
dvisvgm -P document.pdf -n -o document_pdf.svg
|
|
|
|
cat document_pdf.svg
|
|
|
|
file document_pdf.svg | grep SVG
|
|
|
|
|
|
|
|
mkdir "$out"
|
|
|
|
mv document*.svg "$out"/
|
|
|
|
'';
|
2021-06-24 04:16:37 +02:00
|
|
|
|
2021-08-15 17:12:23 +02:00
|
|
|
texdoc = runCommand "texlive-test-texdoc" {
|
2021-06-24 04:16:37 +02:00
|
|
|
nativeBuildInputs = [
|
2023-08-26 14:37:30 +02:00
|
|
|
(texlive.combine {
|
|
|
|
inherit (texlive) scheme-infraonly luatex texdoc;
|
2021-06-24 04:16:37 +02:00
|
|
|
pkgFilter = pkg: lib.elem pkg.tlType [ "run" "bin" "doc" ];
|
|
|
|
})
|
|
|
|
];
|
|
|
|
} ''
|
|
|
|
texdoc --version
|
|
|
|
|
|
|
|
texdoc --debug --list texdoc | tee "$out"
|
|
|
|
grep texdoc.pdf "$out"
|
|
|
|
'';
|
2021-07-04 12:47:33 +02:00
|
|
|
|
2023-08-26 14:37:30 +02: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 14:37:30 +02:00
|
|
|
# check that all languages are available, including synonyms
|
|
|
|
allLanguages = let hyphenBase = lib.head texlive.hyphen-base.pkgs; texLive = texlive.combined.scheme-full; in
|
|
|
|
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 12:47:33 +02:00
|
|
|
# test that language files are generated as expected
|
2021-08-15 17:12:23 +02:00
|
|
|
hyphen-base = runCommand "texlive-test-hyphen-base" {
|
2021-07-04 12:47:33 +02:00
|
|
|
hyphenBase = lib.head texlive.hyphen-base.pkgs;
|
|
|
|
schemeFull = texlive.combined.scheme-full;
|
|
|
|
schemeInfraOnly = texlive.combined.scheme-infraonly;
|
|
|
|
} ''
|
|
|
|
mkdir -p "$out"/{scheme-infraonly,scheme-full}
|
|
|
|
|
|
|
|
# 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
|
|
|
|
|
|
|
|
cat >>language.dat.lua <<EOF
|
|
|
|
}
|
|
|
|
EOF
|
|
|
|
|
|
|
|
cat >>language.def <<EOF
|
|
|
|
%%% No changes may be made beyond this point.
|
|
|
|
|
|
|
|
\uselanguage {USenglish} %%% This MUST be the last line of the file.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
for fname in language.{dat,def,dat.lua} ; do
|
|
|
|
diff --ignore-matching-lines='^\(%\|--\) Generated by ' -u \
|
2023-02-05 20:32:09 +01:00
|
|
|
{"$hyphenBase","$schemeFull"/share/texmf-var}/tex/generic/config/"$fname" \
|
2021-07-04 12:47:33 +02:00
|
|
|
| tee "$out/scheme-full/$fname.patch"
|
|
|
|
diff --ignore-matching-lines='^\(%\|--\) Generated by ' -u \
|
2023-02-05 20:32:09 +01:00
|
|
|
{,"$schemeInfraOnly"/share/texmf-var/tex/generic/config/}"$fname" \
|
2021-07-04 12:47:33 +02:00
|
|
|
| tee "$out/scheme-infraonly/$fname.patch"
|
|
|
|
done
|
|
|
|
'';
|
2021-07-12 21:37:51 +02:00
|
|
|
|
|
|
|
# test that fmtutil.cnf is fully regenerated on scheme-full
|
2021-08-15 17:12:23 +02:00
|
|
|
fmtutilCnf = runCommand "texlive-test-fmtutil.cnf" {
|
2021-07-12 21:37:51 +02:00
|
|
|
kpathsea = lib.head texlive.kpathsea.pkgs;
|
|
|
|
schemeFull = texlive.combined.scheme-full;
|
|
|
|
} ''
|
|
|
|
mkdir -p "$out"
|
|
|
|
|
|
|
|
diff --ignore-matching-lines='^# Generated by ' -u \
|
2023-02-05 20:32:09 +01:00
|
|
|
{"$kpathsea","$schemeFull"/share/texmf-var}/web2c/fmtutil.cnf \
|
2021-07-12 21:37:51 +02:00
|
|
|
| tee "$out/fmtutil.cnf.patch"
|
|
|
|
'';
|
2023-05-14 17:05:22 +02:00
|
|
|
|
|
|
|
# verify that the restricted mode gets enabled when
|
|
|
|
# needed (detected by checking if it disallows --gscmd)
|
|
|
|
repstopdf = runCommand "texlive-test-repstopdf" {
|
|
|
|
nativeBuildInputs = [ (texlive.combine { inherit (texlive) scheme-infraonly 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"
|
|
|
|
'';
|
|
|
|
|
2023-07-10 22:28:36 +02:00
|
|
|
# verify that the restricted mode gets enabled when
|
|
|
|
# needed (detected by checking if it disallows --gscmd)
|
|
|
|
rpdfcrop = runCommand "texlive-test-rpdfcrop" {
|
|
|
|
nativeBuildInputs = [ (texlive.combine { inherit (texlive) scheme-infraonly 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-29 00:06:06 +02: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
|
2023-07-10 22:28:36 +02:00
|
|
|
broken = [
|
|
|
|
# *.inc files in source container rather than run
|
|
|
|
"texaccents"
|
|
|
|
|
|
|
|
# 'Error initialising QuantumRenderer: no suitable pipeline found'
|
|
|
|
"tlcockpit"
|
|
|
|
] ++ lib.optional stdenv.isDarwin "epspdftk"; # wish shebang is a script, not a binary!
|
|
|
|
|
2023-06-29 00:06:06 +02: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"
|
2023-07-10 22:28:36 +02:00
|
|
|
"kpsewhere" "latex-git-log" "ltxfileinfo" "mendex" "perltex" "pn2pdf" "psbook" "psnup" "psresize" "purifyeps"
|
|
|
|
"simpdftex" "tex2xindy" "texluac" "texluajitc" "urlbst" "yplan" ];
|
|
|
|
shortHelp = [ "adhocfilelist" "authorindex" "bbl2bib" "bibdoiadd" "bibmradd" "biburl2doi" "bibzbladd" "ctanupload"
|
2023-07-10 22:28:36 +02:00
|
|
|
"disdvi" "dvibook" "dviconcat" "getmapdl" "latex2man" "listings-ext.sh" "pygmentex" ];
|
2023-06-29 00:06:06 +02:00
|
|
|
# (2) binaries that return non-zero exit code even if correctly asked for help
|
|
|
|
ignoreExitCode = [ "authorindex" "dvibook" "dviconcat" "dvipos" "extractres" "fig4latex" "fragmaster" "latex2man"
|
2023-07-10 22:28:36 +02:00
|
|
|
"latex-git-log" "listings-ext.sh" "psbook" "psnup" "psresize" "purifyeps" "tex2xindy" "texluac"
|
|
|
|
"texluajitc" ];
|
2023-06-29 00:06:06 +02: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"
|
2023-07-10 22:28:36 +02:00
|
|
|
"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" "pdftosrc" "pdfxup" "pedigree" "pfb2pfa" "pk2bm" "prepmx"
|
|
|
|
"ps2pk" "psselect" "pstops" "rubibtex" "rubikrotation" "sjislatex" "sjispdflatex" "srcredact" "t4ht" "tex4ht"
|
|
|
|
"texdiff" "texdirflatten" "texplate" "tie" "ttf2kotexfont" "ttfdump" "vlna" "vpl2ovp" "vpl2vpl" "yplan" ];
|
2023-07-10 22:28:36 +02:00
|
|
|
# (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-29 00:06:06 +02: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)
|
2023-07-10 22:28:36 +02:00
|
|
|
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
|
2023-08-07 20:24:55 +02:00
|
|
|
"epspdftk" "texdoctk" "tlshell" "xasy"
|
2023-07-10 22:28:36 +02:00
|
|
|
|
|
|
|
# requires Cinderella, not open source and not distributed via Nixpkgs
|
|
|
|
"ketcindy"
|
|
|
|
];
|
2023-07-10 22:28:36 +02:00
|
|
|
# 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"
|
|
|
|
|
|
|
|
# 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" "texdoc" "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"
|
|
|
|
];
|
2023-07-10 22:28:36 +02:00
|
|
|
|
|
|
|
# simple test files
|
|
|
|
contextTestTex = writeText "context-test.tex" ''
|
|
|
|
\starttext
|
|
|
|
A simple test file.
|
|
|
|
\stoptext
|
|
|
|
'';
|
|
|
|
latexTestTex = writeText "latex-test.tex" ''
|
2023-06-29 00:06:06 +02:00
|
|
|
\documentclass{article}
|
|
|
|
\begin{document}
|
|
|
|
A simple test file.
|
|
|
|
\end{document}
|
|
|
|
'';
|
2023-07-10 22:28:36 +02:00
|
|
|
texTestTex = writeText "tex-test.tex" ''
|
|
|
|
Hello.
|
|
|
|
\bye
|
|
|
|
'';
|
2023-07-10 22:28:36 +02:00
|
|
|
|
|
|
|
# link all binaries in single derivation
|
|
|
|
allPackages = with lib; concatLists (catAttrs "pkgs" (filter isAttrs (attrValues texlive)));
|
|
|
|
binPackages = lib.filter (p: p.tlType == "bin") allPackages;
|
|
|
|
binaries = buildEnv { name = "texlive-binaries"; paths = binPackages; };
|
2023-06-29 00:06:06 +02:00
|
|
|
in
|
2023-07-10 22:28:36 +02:00
|
|
|
runCommand "texlive-test-binaries"
|
|
|
|
{
|
2023-07-10 22:28:36 +02:00
|
|
|
inherit binaries contextTestTex latexTestTex texTestTex;
|
2023-07-10 22:28:36 +02:00
|
|
|
texliveScheme = texlive.combined.scheme-full;
|
|
|
|
}
|
2023-06-29 00:06:06 +02:00
|
|
|
''
|
2023-07-10 22:28:36 +02:00
|
|
|
loadables="$(command -v bash)"
|
|
|
|
loadables="''${loadables%/bin/bash}/lib/bash"
|
|
|
|
enable -f "$loadables/realpath" realpath
|
2023-06-29 00:06:06 +02:00
|
|
|
mkdir -p "$out"
|
|
|
|
export HOME="$(mktemp -d)"
|
|
|
|
declare -i binCount=0 ignoredCount=0 brokenCount=0 failedCount=0
|
2023-07-10 22:28:36 +02:00
|
|
|
cp "$contextTestTex" context-test.tex
|
|
|
|
cp "$latexTestTex" latex-test.tex
|
|
|
|
cp "$texTestTex" tex-test.tex
|
2023-06-29 00:06:06 +02:00
|
|
|
|
|
|
|
testBin () {
|
2023-07-10 22:28:36 +02:00
|
|
|
path="$(realpath "$bin")"
|
|
|
|
path="''${path##*/}"
|
2023-06-29 00:06:06 +02:00
|
|
|
if [[ -z "$ignoreExitCode" ]] ; then
|
2023-07-10 22:28:36 +02:00
|
|
|
PATH="$path" "$bin" $args >"$out/$base.log" 2>&1
|
2023-07-10 22:28:36 +02: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
|
|
|
|
return $ret
|
2023-06-29 00:06:06 +02:00
|
|
|
else
|
2023-07-10 22:28:36 +02:00
|
|
|
PATH="$path" "$bin" $args >"$out/$base.log" 2>&1
|
2023-06-29 00:06:06 +02:00
|
|
|
ret=$?
|
2023-07-10 22:28:36 +02:00
|
|
|
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-29 00:06:06 +02: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
|
|
|
|
}
|
|
|
|
|
2023-07-10 22:28:36 +02:00
|
|
|
for bin in "$binaries"/bin/* ; do
|
2023-06-29 00:06:06 +02:00
|
|
|
base="''${bin##*/}"
|
|
|
|
args=
|
|
|
|
ignoreExitCode=
|
|
|
|
binCount=$((binCount + 1))
|
|
|
|
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})
|
|
|
|
;;
|
2023-07-10 22:28:36 +02:00
|
|
|
${lib.concatStringsSep "|" contextTest})
|
|
|
|
args=context-test.tex ;;
|
|
|
|
${lib.concatStringsSep "|" latexTest})
|
|
|
|
args=latex-test.tex ;;
|
|
|
|
${lib.concatStringsSep "|" texTest})
|
|
|
|
args=tex-test.tex ;;
|
2023-06-29 00:06:06 +02:00
|
|
|
${lib.concatStringsSep "|" shortVersion})
|
|
|
|
args=-v ;;
|
2023-07-10 22:28:36 +02:00
|
|
|
ebong)
|
|
|
|
touch empty
|
|
|
|
args=empty ;;
|
|
|
|
ht)
|
|
|
|
args='latex latex-test.tex' ;;
|
2023-06-29 00:06:06 +02:00
|
|
|
pdf2dsc)
|
|
|
|
args='--help --help --help' ;;
|
|
|
|
typeoutfileinfo)
|
|
|
|
args=/dev/null ;;
|
|
|
|
*)
|
|
|
|
args=--version ;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
case "$base" in
|
|
|
|
${lib.concatStringsSep "|" (ignoreExitCode ++ noArg)})
|
|
|
|
ignoreExitCode=1 ;;
|
|
|
|
esac
|
|
|
|
|
2023-07-10 22:28:36 +02:00
|
|
|
case "$base" in
|
|
|
|
${lib.concatStringsSep "|" needScheme})
|
|
|
|
bin="$texliveScheme/bin/$base"
|
|
|
|
if [[ ! -f "$bin" ]] ; then
|
|
|
|
ignoredCount=$((ignoredCount + 1))
|
|
|
|
continue
|
|
|
|
fi ;;
|
|
|
|
esac
|
|
|
|
|
2023-06-29 00:06:06 +02:00
|
|
|
if testBin ; then : ; else # preserve exit code
|
|
|
|
echo "failed '$base''${args:+ $args}' (exit code: $?)"
|
2023-07-10 22:28:36 +02:00
|
|
|
sed 's/^/ > /' < "$out/$base.log"
|
2023-06-29 00:06:06 +02:00
|
|
|
failedCount=$((failedCount + 1))
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
2023-07-10 22:28:36 +02:00
|
|
|
echo "tested $binCount binaries: $ignoredCount ignored, $brokenCount broken, $failedCount failed"
|
2023-06-29 00:06:06 +02:00
|
|
|
[[ $failedCount = 0 ]]
|
|
|
|
'';
|
2023-06-30 02:07:49 +02:00
|
|
|
|
2023-07-10 22:28:36 +02:00
|
|
|
# check that all scripts have a Nix shebang
|
|
|
|
shebangs = let
|
|
|
|
allPackages = with lib; concatLists (catAttrs "pkgs" (filter isAttrs (attrValues texlive)));
|
|
|
|
binPackages = lib.filter (p: p.tlType == "bin") allPackages;
|
|
|
|
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
|
|
|
|
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 02:07:49 +02: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);
|
|
|
|
|
|
|
|
savedLicenses = scheme: scheme.meta.license;
|
|
|
|
savedLicensesAttrNames = scheme: map licenseToAttrName (savedLicenses scheme);
|
|
|
|
|
|
|
|
correctLicenses = scheme: builtins.foldl'
|
|
|
|
(acc: pkg: concatLicenses acc (lib.toList (pkg.meta.license or [])))
|
|
|
|
[]
|
|
|
|
scheme.passthru.packages;
|
|
|
|
correctLicensesAttrNames = scheme:
|
|
|
|
lib.sort lt
|
|
|
|
(map licenseToAttrName (correctLicenses scheme));
|
|
|
|
|
|
|
|
hasLicenseMismatch = scheme:
|
|
|
|
(lib.isDerivation scheme) &&
|
|
|
|
(savedLicensesAttrNames scheme) != (correctLicensesAttrNames scheme);
|
|
|
|
incorrectSchemes = lib.filterAttrs
|
|
|
|
(n: hasLicenseMismatch)
|
|
|
|
texlive.combined;
|
|
|
|
prettyPrint = name: scheme:
|
|
|
|
''
|
|
|
|
license info for ${name} is incorrect! Note that order is enforced.
|
|
|
|
saved: [ ${lib.concatStringsSep " " (savedLicensesAttrNames scheme)} ]
|
|
|
|
correct: [ ${lib.concatStringsSep " " (correctLicensesAttrNames scheme)} ]
|
|
|
|
'';
|
|
|
|
errorText = lib.concatStringsSep "\n\n" (lib.mapAttrsToList prettyPrint incorrectSchemes);
|
|
|
|
in
|
|
|
|
runCommand "texlive-test-license" {
|
|
|
|
inherit errorText;
|
|
|
|
}
|
|
|
|
(if (incorrectSchemes == {})
|
|
|
|
then "echo everything is fine! > $out"
|
|
|
|
else ''
|
|
|
|
echo "$errorText"
|
|
|
|
false
|
|
|
|
'');
|
2020-04-11 01:47:44 +02:00
|
|
|
}
|