Replace some commands and environments within a TeX document by evaluating code inside a jupyter kernel
Project description
TexSurgery
Replaces some commands and environments within a TeX document by evaluating code inside a jupyter kernel.
Much like sagetex, but with the following differences:
sagetexcollects all the code using LaTeX and only then runssageto get the LaTeX output, which definitely works, but this conflicts with some interesting LaTeX packages and is slower than a direct conversion.TexSurgeryworks in any language with a jupyter kernel. In particular, you don't need to installsagemath.
Installation
Install python package
`python3 -m pip install texsurgery`
Install tex library
This is usually not required, but recommended. The instructions are different for MiKTeX and TeX Live:
- `TeX Live`: Install the package with `tlmgr --verify-repo=none install texsurgery`
- `MiKTeX`: Use the graphical interface.
- _If the above doesn't work_: Install the CTAN LaTeX package manually: `https://ctan.org/pkg/texsurgery`
- _If all of the above still don't work_: Copy the file `texsurgery.sty` from the folder `texsurgery/tex/texsurgery` into the current working folder where your `tex` document lies.
Installation from source
git clone https://framagit.org/pang/texsurgery.git
cd texsurgery
python3 -m pip install -U .
Basic usage
The most simple use case is to insert a jupyter kernel output in a latex document. In order to do so, the document must include a line like
\usepackage[python3]{texsurgery}
to tell texsurgery which kernel to use. Then texsurgery can perform the following substitutions:
Environments
run
Code blocks like
\begin{run}
code
\end{run}
will pass code to the kernel, and then be replaced by the output (both the input and the output may consist on several lines).
runsilent
Works like run but no output is shown.
eval
allows to make a substitution of a single line input in the middle of a line. For example
The result of 2+2 is \eval{2+2}.
will be replaced to
The result of 2+2 is 4.
Formatting
Optional parameters type and format can be passed to eval in order to format the result.
type can be any python type, plus string and tex. The output is converted to the corresponding type (string works like str, if the outout is a string between quotes, the quotes are eliminated; tex comverts a latex expression with escaed charcters to its unescaped version).
format can be any python format string, or upper or lower.
If the type is not provided, it will be infered from the format if possible.
So for example, the following code
\eval[type=float,format=.2f]{3*5}
\eval[type=float,format=8.2f]{3*5}
\eval[format=upper]{'a'+'b'}
\eval[type=string,format=upper]{'a'+'b'}
\eval[type=tex]{'\\frac{3}{5}'}
will result in
15.00
15.00
'AB'
AB
\frac{3}{5}
evalstr
evaltex
sage
sinput
sif
srepl
Command line usage
You can call it from the command line as follows:
texsurgery input_file.tex -o output_file.tex
will perform the code substitutions in input_file.tex and write the result in output_file.tex
The possible parameters are:
-h, --helpshows the help message and exits,input_filethe file to read from. If none given, the standard input will be used.--output_file output_file, -o output_filewrites the result in output file. If it is not provided, the result is directed to the standard output-texjust perform the code substitution in the latex content (this is the default)-pdfcreate a pdf file from the resulting modified tex. If the-ooption is not provided, texsurgery will try to guess the name of the resulting file. This option deppends onpdflatexbeing installed in the system.--pdflatex-optionsoptions to be passed to pdflatex (it requires the-pdfoption).
Since it can use standard input/output, it is pipe friendly. So, for example:
texsurgery input_file.tex | pandoc --from latex -o output_file.html
will perform the code substitutions and convert the result to a html (provided you have pandoc installed).
Likewise, if you have a markdown file input.md with the lines
\usepackage[python3]{texsurgery}
The result of multiplying 3 and 5 is \eval{3*5}
And run
pandoc input.md --from markdown --to latex -s | texsurgery
you will get the corresponding latex file with the \usepackage line removed, and the \eval{3*5} text substituted by the corresponding result 15.
In fact, since texsurgery does not require fully correct .tex code, you can directly run texsurgery in the markdown file!
Testing
If you installed from source, the following command will perform some common tests, and specific tests for some of the kernels that are installed:
python3 -m unittest tests
Selectors
find and findall
texsurgery can also gather information using css-style selectors:
.. doctest::
>>> from texsurgery.texsurgery import TexSurgery
>>> tex = open('../tests/test_find.tex').read()
>>> # An environment, which can be question or questionmultx,
>>> # which contains an environment runsilent, and captures its content
>>> TexSurgery(tex).findall('question,questionmultx runsilent')
[('questionmultx', [('runsilent', '\na = randint(1,10)\n')]), ('questionmultx', [('runsilent', '\na = randint(2,10)\n')]), ('question', [('runsilent', '\na = randint(2,10)\nf = sin(a*x)\nfd = f.derivative(x)\n')])]
>>> # An environment, which can be question or questionmultx,
>>> # which contains an environment choices
>>> # which contains a command \correctchoice, and captures its argument
>>> TexSurgery(tex).findall('question,questionmultx choices \correctchoice{choice}')
[('question', [('choices', [('\\correctchoice', {'choice': '$\\sage{fd}$'})])])]
>>> # An environment questionmultx which contains a command
>>> # \AMCnumericChoices with two arbitrary arguments
>>> TexSurgery(tex).findall('questionmultx \AMCnumericChoices[_nargs=2]')
[('questionmultx', [('\\AMCnumericChoices', {'arg0': '\\eval{8+a}', 'arg1': 'digits=2,sign=false,scoreexact=3'})]), ('questionmultx', [('\\AMCnumericChoices', {'arg0': '\\eval{8*a}', 'arg1': 'digits=2,sign=false,scoreexact=3'})])]
>>> # Environment questionmultx, with first argument exactly equal to basic-multiplication
>>> TexSurgery(tex).find(r'questionmultx{@basic-multiplication}')
('questionmultx', {'@basic-multiplication': 'basic-multiplication'}, '\n\\begin{runsilent}\na = randint(2,10)\n\\end{runsilent}\nWhat is $8*\\eval{a}$?\n\\AMCnumericChoices{\\eval{8*a}}{digits=2,sign=false,scoreexact=3}\n')
>>> # Command \copygroup, with any first argument, and second argument
>>> # exactly equal to BigGroupe
>>> TexSurgery(tex).findall(r'\copygroup{category}{@BigGroupe}')
[('\\copygroup', {'category': 'cat1', '@BigGroupe': 'BigGroupe'}), ('\\copygroup', {'category': 'cat2', '@BigGroupe': 'BigGroupe'})]
>>> # Command \subsection, with any title as first argument,
>>> # with \label{seed}
>>> TexSurgery(tex).find('\\subsection[label="student id"]{title}')
('\\subsection', {'title': 'Student identification'})
>>> # Command \subsection, with any title as first argument,
>>> # with \label{seed} and in this subsection (which is ended by a \section)
>>> # there is a run environment
>>> TexSurgery(tex).find('\\subsection{title}#seed:next run')
('\\subsection', ('run', "\nprint('The random seed is ', seed)\n"))
insertAfter and replace
texsurgery can perform search-and-replace, and search-and-insert-after using the css-style selectors:
.. doctest::
>>> from texsurgery.texsurgery import TexSurgery
>>> tex = r'''\begin{choices}
... \wrongchoice{$\sage{fd + a}$}
... \correctchoice{$\sage{fd}$}
... \wrongchoice{$\sage{fd*a}$}
... \end{choices}
... '''
>>> ts = TexSurgery(tex)
>>> ts = ts.replace(r'\correctchoice{choice}', r'\correctchoice{$\sage{f.derivative(x)}$}')
>>> ts.src
'\\begin{choices}\n\\wrongchoice{$\\sage{fd + a}$}\n\\correctchoice{$\\sage{f.derivative(x)}$}\n\\wrongchoice{$\\sage{fd*a}$}\n\\end{choices}\n'
shuffle
texsurgery can also shuffle some TexElements nested within a parent using the css-style selectors:
.. doctest::
>>> from texsurgery.texsurgery import TexSurgery
>>> tex = r'''\begin{choices}
... \wrongchoice{$\sage{fd + a}$}
... \correctchoice{$\sage{fd}$}
... \wrongchoice{$\sage{fd*a}$}
... \end{choices}
... '''
>>> ts = TexSurgery(tex)
>>> ts = ts.shuffle('choices', r'\correctchoice{choice},\wrongchoice{choice}', randomseed=1)
>>> ts.src
'\\begin{choices}\n\\correctchoice{$\\sage{fd}$}\n\\wrongchoice{$\\sage{fd*a}$}\n\\wrongchoice{$\\sage{fd + a}$}\n\\end{choices}\n'
Kernels
TexSurgery can use several jupyter kernels in the same document. In order to do so, you have to declare them with a line like
\usepackage[sagemath,python3]{texsurgery}
The first one is the default. To use another one, pass the option to the corresponding environment. For example
\begin{run}
1^1
\end{run}
will be transformed into
1
but, because of the different way that sagemath and python handle the ^ operator, this
\begin{run}[python3]
1^1
\end{run}
will be transformed into
0
instead.
This can be useful for example to include graphics generated by different systems in the same document.
Example
Start with this LaTeX code:
% Any jupyter kernel is available
\usepackage[sagemath]{texsurgery}
% Compatible with any other LaTeX package
\usepackage[bloc,completemulti]{automultiplechoice}
% Example of user macros
\providecommand{\abs}[1]{\lvert#1\rvert}
\newcommand{\R}{\mathbb{R}}
% TexSurgery can replace some \commands before pdflatex runs
\begin{minipage}{.85\linewidth}
Student: {\bf \name \; \surname}, \quad ID: {\bf \id}
\end{minipage}
\begin{question}{derivative-sin}
\qvariant{1} \qtags{derivative}
% TexSurgery will run code in a jupyter kernel
\begin{runsilent}
set_random_seed(\seed)
a = randint(2,10)
f = sin(a*x)
fd = f.derivative(x)
\end{runsilent}
% TexSurgery will eval code in a jupyter kernel
% and replace \eval{expr} with the output from the kernel
% \sage{expr} is just an alias for \eval{latex(expr)}
What is the first derivative of $\sage{f}$?
\begin{choices}
\correctchoice{$\sage{fd}$}
\wrongchoice{$\sage{fd*a}$}
\wrongchoice{$\sage{fd + a}$}
\end{choices}
\begin{explain}
\begin{run}
# TexSurgery will run code in the jupyter kernel
# and replace this environment with the full output
\end{run}
\end{explain}
\end{question}
and run this python code:
from texsurgery.texsurgery import TexSurgery
student_vars = dict(name='Fulano', surname='de Tal', seed='1', id='314159')
ts = TexSurgery(tex_source).data_surgery(student_vars).code_surgery()
in order to transform it into this:
% Compatible with any other LaTeX package
\usepackage[bloc,completemulti]{automultiplechoice}
% Example of user macros
\providecommand{\abs}[1]{\lvert#1\rvert}
\newcommand{\R}{\mathbb{R}}
\begin{minipage}{.85\linewidth}
Student: {\bf Fulano \; de Tal}, \quad ID: {\bf 314159}
\end{minipage}
\begin{question}{derivative-sin}
\qvariant{1} \qtags{derivative}
What is the first derivative of $\sin\left(7 \, x\right)$?
\begin{choices}
\correctchoice{$7 \, \cos\left(7 \, x\right)$}
\wrongchoice{$49 \, \cos\left(7 \, x\right)$}
\wrongchoice{$7 \, \cos\left(7 \, x\right) + 7$}
\end{choices}
\begin{explain}
\begin{run}
# TexSurgery will run code in the jupyter kernel
# and replace this environment with the full output
\end{run}
\end{explain}
\end{question}
Thanks
To all our colleagues that gave feedback to the early versions, specially Fabricio from ETSIN.UPM and Carlos from ETSIAAB.UPM
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file texsurgery-0.6.3.tar.gz.
File metadata
- Download URL: texsurgery-0.6.3.tar.gz
- Upload date:
- Size: 29.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.4.2 importlib_metadata/4.6.3 pkginfo/1.7.1 requests/2.22.0 requests-toolbelt/0.9.1 tqdm/4.62.0 CPython/3.8.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ce839e4d11dca439d7275402ec1233f60bb3a8bf50ee4989e55486132a9a9df6
|
|
| MD5 |
87457ef72b2ab13711c8b71329f0076d
|
|
| BLAKE2b-256 |
89044eda5e327067412ec5d248007661e021de08a78a5eb66569ea816e6368de
|
File details
Details for the file texsurgery-0.6.3-py3-none-any.whl.
File metadata
- Download URL: texsurgery-0.6.3-py3-none-any.whl
- Upload date:
- Size: 31.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.4.2 importlib_metadata/4.6.3 pkginfo/1.7.1 requests/2.22.0 requests-toolbelt/0.9.1 tqdm/4.62.0 CPython/3.8.10
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b8583e6e41c49b742f4d9087b3fbc8d147caff60ea340ea1f7c8c97a426cdf03
|
|
| MD5 |
0fb07a6fc86715de65adf08ef6cf4c99
|
|
| BLAKE2b-256 |
e076bfe48cc2b455f8d9da8693912ab4f113576542593e64f305e218c31a87b6
|