Take this minimal LaTeX document:
% minimal.tex
\documentclass{article}
\begin{document}
Hello World!
\end{document}
And this Ruby (2.5.1) script:
require 'open3'
Open3.popen3(
'xelatex', '-interaction=nonstopmode', '"minimal.tex"'
) do |_stdin, stdout, stderr, wait_thr|
log = stdout.readlines + stderr.readlines
exit_status = wait_thr.value
puts exit_status
puts log.join('').strip!
end
Running the script, we get:
This is XeTeX, Version 3.14159265-2.6-0.99999 (TeX Live 2018) (preloaded format=xelatex)
...
! I can't find file `'"minimal.tex"''.
This is surprising. Not only because the file is there, but because both
Open3.popen3('xelatex -interaction=nonstopmode "minimal.tex"') do ... andOpen3.popen3('lualatex', '-interaction=nonstopmode', '"minimal.tex"') do ...work just fine!
What is special about the combination of xelatex and popen3?
Open3.popen3(
'xelatex', '-interaction=nonstopmode', 'minimal.tex'
)
works fine.
The error message tells you, the file "minimal.tex" does not exist.
With the varargs form of popen3, the commands array is passed to Process#spawn where xelatex is called with arg1=-interaction=nonstopmode and arg2=minimal.tex. The arguments are not passed to the shell. Side note, this way you can also call other processes with "complex" args, e.g. Open3.popen3('mkdir', 'directory with spaces and $SHELL').
Ad Open3.popen3('xelatex -interaction=nonstopmode "minimal.tex"'): here, the entire string is considered a command that is sent to the standard shell, and this shell takes care of the double quotes.
Ad Open3.popen3('lualatex', '-interaction=nonstopmode', '"minimal.tex"'): here I can only guess that lualatex ignores the double quotes.
See also: https://ruby-doc.org/core-2.2.0/Process.html#method-c-exec