File::Temp->safe_level(File::Temp::MEDIUM);
use
version;
our
$VERSION
= version->declare(
"0.8.8_1"
);
our
$FULLVERSION
=
"LaTeXML version $LaTeXML::VERSION"
. (
$LaTeXML::Version::REVISION
?
"; revision $LaTeXML::Version::REVISION"
:
''
);
our
$IDENTITY
=
"$FindBin::Script ($LaTeXML::FULLVERSION)"
;
our
$LOG_STACK
= 0;
our
@COMPARABLE
=
qw(preload paths verbosity strict comments inputencoding includestyles documentid mathparse)
;
sub
new {
my
(
$class
,
$config
) =
@_
;
$config
= LaTeXML::Common::Config->new()
unless
(
defined
$config
);
my
$self
=
bless
{
opts
=>
$config
->options,
ready
=> 0,
log
=>
q{}
,
runtime
=> {},
latexml
=>
undef
},
$class
;
my
$debug_directives
=
$$self
{opts}->{debug};
$LaTeXML::DEBUG
{latexml} = 1
if
(
ref
$debug_directives
eq
'ARRAY'
) && (
grep
{ /latexml/i }
@$debug_directives
);
eval
{
$config
->check; };
if
($@) {
$self
->bind_log;
NoteLog($@);
$$self
{
log
} .=
$self
->flush_log; }
return
$self
; }
sub
prepare_session {
my
(
$self
,
$config
) =
@_
;
eval
{
$config
->check; };
if
($@) {
$self
->bind_log;
NoteLog($@);
$$self
{
log
} .=
$self
->flush_log; }
my
$opts
=
$config
->options;
my
$opts_comparable
= {
map
{
$_
=>
$$opts
{
$_
} }
@COMPARABLE
};
my
$self_opts_comparable
= {
map
{
$_
=>
$$self
{opts}{
$_
} }
@COMPARABLE
};
my
$something_to_do
;
$something_to_do
= LaTeXML::Util::ObjectDB::compare(
$opts_comparable
,
$self_opts_comparable
) ? 0 : 1;
$$self
{opts} =
$opts
;
$self
->initialize_session
if
(
$something_to_do
|| (!
$$self
{ready}));
return
;
}
sub
initialize_session {
my
(
$self
) =
@_
;
$$self
{runtime} = {}
unless
$$self
{runtime};
my
$runtime
=
$$self
{runtime};
$$runtime
{TTL} =
$$self
{opts}{timeout}
unless
defined
$$runtime
{TTL};
$self
->bind_log;
foreach
my
$subname
(
keys
%LaTeXML::Package::Pool::
) {
delete
$LaTeXML::Package::Pool::
{
$subname
};
}
my
$latexml
;
eval
{
local
$SIG
{
'ALRM'
} =
sub
{
die
"Fatal:conversion:init Failed to initialize LaTeXML state\n"
};
alarm
(
$$runtime
{TTL});
$latexml
= new_latexml(
$$self
{opts});
$$runtime
{TTL} =
alarm
(0);
1;
};
$$latexml
{state}->noteStatus(
'fatal'
)
if
$latexml
&& $@;
if
($@) {
Debug($@);
Note(
"Initialization complete: "
.
$latexml
->getStatusMessage .
". Aborting."
)
if
defined
$latexml
;
$$self
{
log
} .=
$self
->flush_log;
$$self
{ready} = 0;
return
;
}
else
{
my
$init_status
=
$latexml
->getStatusMessage;
if
(
$init_status
=~ /error/i) {
Note(
"Initialization complete: "
.
$init_status
.
". Aborting."
);
$$self
{
log
} .=
$self
->flush_log;
$$self
{ready} = 0;
return
;
}
}
$$self
{
log
} .=
$self
->flush_log;
$$self
{latexml} =
$latexml
;
$$self
{ready} = 1;
return
;
}
sub
convert {
my
(
$self
,
$source
) =
@_
;
$$self
{runtime} = {
TTL
=>
$$self
{opts}{timeout} };
my
$runtime
=
$$self
{runtime};
$self
->initialize_session
unless
$$self
{ready};
if
(!
$$self
{ready}) {
return
{
result
=>
undef
,
log
=>
$self
->flush_log,
status
=>
"Initialization failed."
,
status_code
=> 3 }; }
$self
->bind_log;
my
$opts
=
$$self
{opts};
(
$$runtime
{status},
$$runtime
{status_code}) = (
undef
,
undef
);
Note(
"$LaTeXML::IDENTITY"
);
NoteLog(
"invoked as [$0 "
.
join
(
' '
,
@ARGV
) .
"]"
);
Note((
$$opts
{recursive} ?
"recursive "
:
""
) .
"processing started "
.
localtime
());
my
(
$current_preamble
,
$current_postamble
);
if
(
$$opts
{whatsin} eq
"math"
) {
$current_preamble
=
'literal:\begin{document}\ensuremathfollows'
;
$current_postamble
=
'literal:\ensuremathpreceeds\end{document}'
; }
elsif
(
$$opts
{whatsin} eq
'fragment'
) {
$current_preamble
=
$$opts
{preamble} ||
'standard_preamble.tex'
;
$current_postamble
=
$$opts
{postamble} ||
'standard_postamble.tex'
; }
elsif
(
$$opts
{whatsin} =~ /^archive/) {
$$opts
{archive_sourcedirectory} =
$$opts
{sourcedirectory};
my
$sandbox_directory
= File::Temp->newdir(
TMPDIR
=> 1);
$$opts
{sourcedirectory} =
$sandbox_directory
;
$source
= unpack_source(
$source
,
$sandbox_directory
);
if
(!
defined
$source
) {
$$opts
{sourcedirectory} =
$$opts
{archive_sourcedirectory};
my
$log
=
$self
->flush_log;
$log
.=
"\nFatal:invalid:Archive Can't detect a source TeX file!\nStatus:conversion:3\n"
;
return
{
result
=>
undef
,
log
=>
$log
,
status
=>
"Fatal:invalid:Archive Can't detect a source TeX file!"
,
status_code
=> 3 }; }
if
((
$$opts
{whatsout} =~ /^archive/) && (!
$$opts
{destination})) {
$$opts
{placeholder_destination} = 1;
$$opts
{destination} = pathname_name(
$source
) .
".zip"
; } }
if
(
$$opts
{whatsout} =~ /^archive/) {
$$opts
{archive_sitedirectory} =
$$opts
{sitedirectory};
$$opts
{archive_destination} =
$$opts
{destination};
my
$destination_name
=
$$opts
{destination} ? pathname_name(
$$opts
{destination}) :
'document'
;
my
$sandbox_directory
= File::Temp->newdir(
TMPDIR
=> 1);
my
$extension
=
$$opts
{
format
};
$extension
=~ s/\d+$//;
$extension
=~ s/^epub|mobi$/xhtml/;
my
$sandbox_destination
=
"$destination_name.$extension"
;
$$opts
{sitedirectory} =
$sandbox_directory
;
if
(
$$opts
{
format
} eq
'epub'
) {
$$opts
{resource_directory} = File::Spec->catdir(
$sandbox_directory
,
'OPS'
);
$$opts
{destination} = pathname_concat(File::Spec->catdir(
$sandbox_directory
,
'OPS'
),
$sandbox_destination
); }
else
{
$$opts
{destination} = pathname_concat(
$sandbox_directory
,
$sandbox_destination
); } }
if
((!
$$opts
{destination})
&& (
$$opts
{dographics} ||
$$opts
{picimages} ||
grep
{
$_
eq
'images'
or
$_
eq
'svg'
} @{
$$opts
{math_formats} })) {
Warn(
"expected"
,
"options"
,
undef
,
"must supply --destination to support auxilliary files"
,
" disabling: --nomathimages --nographicimages --nopictureimages"
);
$$opts
{dographics} = 0;
$$opts
{picimages} = 0;
removeMathFormat(
$opts
,
'images'
);
removeMathFormat(
$opts
,
'svg'
);
maybeAddMathFormat(
$opts
,
'pmml'
); }
my
$latexml
=
$$self
{latexml};
$latexml
->withState(
sub
{
my
(
$state
) =
@_
;
$$state
{status} = {};
my
$stomach
=
$$state
{stomach};
delete
$$stomach
{rescued_boxes}
if
$$stomach
{rescued_boxes};
$state
->pushDaemonFrame;
$state
->assignValue(
'_authlist'
,
$$opts
{authlist},
'global'
);
$state
->assignValue(
'REMOTE_REQUEST'
, (!
$$opts
{
local
}),
'global'
);
});
my
(
$digested
,
$dom
,
$serialized
) = (
undef
,
undef
,
undef
);
eval
{
alarm
(
$$runtime
{TTL});
my
$mode
= (
$$opts
{type} eq
'auto'
) ?
'TeX'
:
$$opts
{type};
$digested
=
$latexml
->digestFile(
$source
,
preamble
=>
$current_preamble
,
postamble
=>
$current_postamble
,
mode
=>
$mode
,
noinitialize
=> 1);
$$runtime
{TTL} =
alarm
(0); };
my
$eval_report
= $@;
if
(!
$digested
&&
$eval_report
) {
eval
{
alarm
(
$$runtime
{TTL});
$digested
=
$latexml
->withState(
sub
{
return
$latexml
->finishDigestion; });
$$runtime
{TTL} =
alarm
(0); };
$eval_report
.= $@
if
$@; }
my
$core_target
=
$$opts
{
format
};
if
(
$core_target
ne
'tex'
and
$core_target
ne
'box'
) {
$core_target
=
'xml'
; }
if
(
$digested
) {
eval
{
alarm
(
$$runtime
{TTL});
$latexml
->withState(
sub
{
if
(
$core_target
eq
'tex'
) {
$serialized
= LaTeXML::Core::Token::UnTeX(
$digested
); }
elsif
(
$core_target
eq
'box'
) {
$serialized
= (
$$opts
{verbosity} > 0 ?
$digested
->stringify :
$digested
->toString); }
elsif
(
$core_target
eq
'xml'
) {
$dom
=
$latexml
->convertDocument(
$digested
); } });
$$runtime
{TTL} =
alarm
(0); };
$eval_report
.= $@
if
$@;
if
(!
$dom
&& $@ &&
$core_target
eq
'xml'
) {
$dom
=
$latexml
->withState(
sub
{
my
(
$state
) =
@_
;
my
$rescued
=
$$state
{rescued_document};
$rescued
->finalize()
if
$rescued
;
return
$rescued
; }); } }
$$runtime
{status} =
$latexml
->getStatusMessage;
$$runtime
{status_code} =
$latexml
->getStatusCode;
$latexml
->showProfile();
if
(
$eval_report
) {
$$runtime
{status} .=
"\n"
.
$eval_report
.
"\n"
;
$$runtime
{status_code} = 3; }
$latexml
->withState(
sub
{
my
(
$state
) =
@_
;
$$opts
{searchpaths} =
$state
->lookupValue(
'SEARCHPATHS'
);
if
(
$state
->lookupValue(
'LEXEMATIZE_MATH'
)) {
$$opts
{math_formats} ||= [];
push
@{
$$opts
{math_formats} },
'lexemes'
;
$$opts
{parallelmath} = 1
if
(@{
$$opts
{math_formats} } > 1); }
$state
->popDaemonFrame;
});
if
(
$LaTeXML::UNSAFE_FATAL
) {
$LaTeXML::UNSAFE_FATAL
= 0;
$$self
{ready} = 0;
}
Note((
$$opts
{recursive} ?
"recursive "
:
""
) .
"Conversion complete: "
.
$$runtime
{status});
if
(
$serialized
) {
if
(
$$opts
{whatsin} =~ /^archive/) {
rmtree(
$$opts
{sourcedirectory});
$$opts
{sourcedirectory} =
$$opts
{archive_sourcedirectory}; }
return
{
result
=>
$serialized
,
log
=>
$self
->flush_log,
status
=>
$$runtime
{status},
status_code
=>
$$runtime
{status_code} };
}
my
$result
=
$dom
;
if
(
$$opts
{post} &&
$dom
) {
if
(!
$dom
->documentElement) {
$$dom
{document}->setDocumentElement(
$$dom
{document}->createElement(
"document"
));
}
$result
=
$self
->convert_post(
$dom
);
}
if
(
$$opts
{whatsin} =~ /^archive/) {
rmtree(
$$opts
{sourcedirectory});
$$opts
{sourcedirectory} =
$$opts
{archive_sourcedirectory}; }
if
(
$$opts
{whatsout} =~ /^archive/) {
rmtree(
$$opts
{sitedirectory});
$$opts
{sitedirectory} =
$$opts
{archive_sitedirectory};
$$opts
{destination} =
$$opts
{archive_destination};
if
(
delete
$$opts
{placeholder_destination}) {
delete
$$opts
{destination}; } }
undef
$serialized
;
my
$ref_result
=
ref
(
$result
) ||
''
;
if
(
$$opts
{
format
} eq
'dom'
) {
$serialized
=
$result
; }
elsif
(
$ref_result
=~ /^(:?LaTe)?XML/) {
if
(
$$opts
{
format
} =~ /^jats|x(ht)?ml$/) {
if
(
$ref_result
=~ /Document$/) {
$serialized
=
$result
->toString(1);
if
(
$serialized
and
$ref_result
eq
'LaTeXML::Core::Document'
) {
$serialized
= Encode::encode(
'UTF-8'
,
$serialized
); }
}
else
{
$serialized
=
$result
->toString(1, 1);
} }
elsif
(
$$opts
{
format
} =~ /^html/) {
if
(
ref
(
$result
) =~ /^LaTeXML::(Post::)?Document$/) {
$serialized
=
$result
->getDocument->toStringHTML; }
else
{
local
$XML::LibXML::setTagCompression
= 1;
$serialized
=
$result
->toString(1, 1); } } }
else
{
$serialized
=
$result
; }
Note(
"Status:conversion:"
. (
$$runtime
{status_code} ||
'0'
));
return
{
result
=>
$serialized
,
log
=>
$self
->flush_log,
status
=>
$$runtime
{status},
'status_code'
=>
$$runtime
{status_code} };
}
our
%DAEMON_CACHE
= ();
our
%CONFIG_CACHE
= ();
sub
get_converter {
my
(
$self
,
$config
,
$key
) =
@_
;
if
(!
$key
&&
$config
) {
$key
=
$config
->get(
'cache_key'
) ||
$config
->get(
'profile'
); }
my
$converter
=
$key
&&
$DAEMON_CACHE
{
$key
};
if
(!
defined
$converter
) {
$config
||=
$CONFIG_CACHE
{
$key
};
$converter
= LaTeXML->new(
$config
->clone);
if
(
$key
) {
$DAEMON_CACHE
{
$key
} =
$converter
;
$CONFIG_CACHE
{
$key
} =
$config
; } }
return
$converter
; }
sub
convert_post {
my
(
$self
,
$dom
) =
@_
;
my
$opts
=
$$self
{opts};
my
$runtime
=
$$self
{runtime};
my
(
$xslt
,
$parallel
,
$math_formats
,
$format
,
$verbosity
,
$defaultresources
,
$embed
) =
map
{
$$opts
{
$_
} }
qw(stylesheet parallelmath math_formats format verbosity defaultresources embed)
;
SetVerbosity(
$verbosity
)
if
defined
$verbosity
;
my
%PostOPS
= (
validate
=>
$$opts
{validate},
sourceDirectory
=>
$$opts
{sourcedirectory},
siteDirectory
=>
$$opts
{sitedirectory},
resource_directory
=>
$$opts
{resource_directory},
searchpaths
=>
$$opts
{searchpaths},
nocache
=> 1,
destination
=>
$$opts
{destination},
is_html
=>
$$opts
{is_html});
if
(
$PostOPS
{destination}) {
my
(
$dir
,
$name
,
$ext
) = pathname_split(
$PostOPS
{destination});
$PostOPS
{destinationDirectory} =
$dir
||
'.'
; }
$parallel
=
$parallel
|| 0;
my
$DOCUMENT
= LaTeXML::Post::Document->new(
$dom
,
%PostOPS
);
my
@procs
= ();
my
$dbfile
=
$$opts
{dbfile};
if
(
defined
$dbfile
&& !-f
$dbfile
) {
if
(
my
$dbdir
= pathname_directory(
$dbfile
)) {
pathname_mkdir(
$dbdir
); } }
my
$DB
= LaTeXML::Util::ObjectDB->new(
dbfile
=>
$dbfile
,
%PostOPS
);
if
(
$format
eq
'epub'
) {
$self
->check_TOC(
$DOCUMENT
); }
if
(
$$opts
{
split
}) {
push
(
@procs
, LaTeXML::Post::Split->new(
split_xpath
=>
$$opts
{splitpath},
splitnaming
=>
$$opts
{splitnaming},
db
=>
$DB
,
%PostOPS
)); }
my
$scanner
= (
$$opts
{scan} ||
$DB
) && (LaTeXML::Post::Scan->new(
db
=>
$DB
,
labelids
=> (
$$opts
{splitnaming} && (
$$opts
{splitnaming} =~ /^label/) ? 1 : 0),
%PostOPS
));
push
(
@procs
,
$scanner
)
if
$$opts
{scan};
if
(!(
$$opts
{prescan})) {
if
(
$$opts
{
index
}) {
push
(
@procs
, LaTeXML::Post::MakeIndex->new(
db
=>
$DB
,
permuted
=>
$$opts
{permutedindex},
split
=>
$$opts
{splitindex},
scanner
=>
$scanner
,
%PostOPS
)); }
push
(
@procs
, LaTeXML::Post::MakeBibliography->new(
db
=>
$DB
,
bibliographies
=>
$$opts
{bibliographies},
split
=>
$$opts
{splitbibliography},
scanner
=>
$scanner
,
%PostOPS
));
if
(
$$opts
{crossref}) {
push
(
@procs
, LaTeXML::Post::CrossRef->new(
db
=>
$DB
,
urlstyle
=>
$$opts
{urlstyle},
extension
=>
$$opts
{extension},
(
$$opts
{numbersections} ? (
number_sections
=> 1) : ()),
(
$$opts
{navtoc} ? (
navigation_toc
=>
$$opts
{navtoc}) : ()),
%PostOPS
)); }
if
(
$$opts
{picimages}) {
push
(
@procs
, LaTeXML::Post::PictureImages->new(
%PostOPS
));
}
else
{
push
(
@procs
, LaTeXML::Post::PictureImages->new(
empty_only
=> 1,
%PostOPS
)); }
if
(
$$opts
{dographics}) {
my
@g_options
= ();
if
(
$$opts
{graphicsmaps} &&
scalar
(@{
$$opts
{graphicsmaps} })) {
my
@maps
=
map
{ [
split
(/\./,
$_
)] } @{
$$opts
{graphicsmaps} };
push
(
@g_options
, (
graphics_types
=> [
map
{
$$_
[0] }
@maps
],
type_properties
=> {
map
{ (
$$_
[0] => {
destination_type
=> (
$$_
[1] ||
$$_
[0]) }) }
@maps
})); }
push
(
@procs
, LaTeXML::Post::Graphics->new(
@g_options
,
%PostOPS
));
}
if
(
$$opts
{svg}) {
push
(
@procs
, LaTeXML::Post::SVG->new(
%PostOPS
)); }
if
(
@$math_formats
) {
my
@mprocs
= ();
foreach
my
$fmt
(
@$math_formats
) {
if
(
$fmt
eq
'xmath'
) {
push
(
@mprocs
, LaTeXML::Post::XMath->new(
%PostOPS
)); }
elsif
(
$fmt
eq
'pmml'
) {
push
(
@mprocs
, LaTeXML::Post::MathML::Presentation->new(
linelength
=>
$$opts
{linelength},
(
defined
$$opts
{plane1} ? (
plane1
=>
$$opts
{plane1}) : (
plane1
=> 1)),
(
$$opts
{hackplane1} ? (
hackplane1
=> 1) : ()),
(
defined
$$opts
{invisibletimes} ? (
invisibletimes
=>
$$opts
{invisibletimes}) :
(
invisibletimes
=> 1)),
%PostOPS
)); }
elsif
(
$fmt
eq
'cmml'
) {
push
(
@mprocs
, LaTeXML::Post::MathML::Content->new(
(
defined
$$opts
{plane1} ? (
plane1
=>
$$opts
{plane1}) : (
plane1
=> 1)),
(
$$opts
{hackplane1} ? (
hackplane1
=> 1) : ()),
%PostOPS
)); }
elsif
(
$fmt
eq
'om'
) {
push
(
@mprocs
, LaTeXML::Post::OpenMath->new(
(
defined
$$opts
{plane1} ? (
plane1
=>
$$opts
{plane1}) : (
plane1
=> 1)),
(
$$opts
{hackplane1} ? (
hackplane1
=> 1) : ()),
%PostOPS
)); }
elsif
(
$fmt
eq
'images'
) {
push
(
@mprocs
, LaTeXML::Post::MathImages->new(
magnification
=>
$$opts
{mathimagemag},
%PostOPS
)); }
elsif
(
$fmt
eq
'svg'
) {
push
(
@mprocs
, LaTeXML::Post::MathImages->new(
magnification
=>
$$opts
{mathimagemag},
imagetype
=>
'svg'
,
%PostOPS
)); }
elsif
(
$fmt
eq
'mathtex'
) {
push
(
@mprocs
, LaTeXML::Post::TeXMath->new(
%PostOPS
)); }
elsif
(
$fmt
eq
'unicodemath'
) {
push
(
@mprocs
, LaTeXML::Post::UnicodeMath->new(
%PostOPS
)); }
elsif
(
$fmt
eq
'lexemes'
) {
push
(
@mprocs
, LaTeXML::Post::LexMath->new(
%PostOPS
)); }
}
if
(
$parallel
) {
my
$main
=
shift
(
@mprocs
);
$main
->setParallel(
@mprocs
);
push
(
@procs
,
$main
); }
else
{
push
(
@procs
,
@mprocs
); }
}
if
(
$xslt
) {
my
$parameters
= {
LATEXML_VERSION
=>
"'$LaTeXML::VERSION'"
};
my
@searchpaths
= (
'.'
,
$DOCUMENT
->getSearchPaths);
foreach
my
$css
(@{
$$opts
{css} }) {
push
(@{
$$parameters
{CSS} },
$css
); }
foreach
my
$js
(@{
$$opts
{javascript} }) {
push
(@{
$$parameters
{JAVASCRIPT} },
$js
); }
if
(
$$opts
{icon}) {
$$parameters
{ICON} =
$$opts
{icon}; }
if
(!
defined
$$opts
{timestamp}) {
$$opts
{timestamp} =
defined
$ENV
{SOURCE_DATE_EPOCH} ?
gmtime
(
$ENV
{SOURCE_DATE_EPOCH}) :
localtime
(); }
if
(
$$opts
{timestamp}) {
$$parameters
{TIMESTAMP} =
"'"
.
$$opts
{timestamp} .
"'"
; }
foreach
my
$parm
(@{
$$opts
{xsltparameters} }) {
if
(
$parm
=~ /^\s*(\w+)\s*:\s*(.*)$/) {
$$parameters
{$1} =
"'"
. $2 .
"'"
; }
else
{
warn
"xsltparameter not in recognized format: 'name:value' got: '$parm'\n"
; }
}
push
(
@procs
, LaTeXML::Post::XSLT->new(
stylesheet
=>
$xslt
,
parameters
=>
$parameters
,
searchpaths
=> [
@searchpaths
],
noresources
=> (
defined
$$opts
{defaultresources}) && !
$$opts
{defaultresources},
%PostOPS
));
}
}
if
(
$$opts
{destination} &&
((
$$opts
{
local
} && (
$$opts
{whatsout} eq
'document'
))
|| (
$$opts
{whatsout} =~ /^archive/))) {
push
(
@procs
, LaTeXML::Post::Writer->new(
format
=>
$format
,
omit_doctype
=>
$$opts
{omit_doctype},
%PostOPS
));
}
my
@postdocs
;
my
$latexmlpost
= LaTeXML::Post->new();
eval
{
alarm
(
$$runtime
{TTL});
@postdocs
=
$latexmlpost
->ProcessChain(
$DOCUMENT
,
@procs
);
$$runtime
{TTL} =
alarm
(0);
1; };
if
($@) {
$$runtime
{status_code} = 3;
local
$@ =
'Fatal:conversion:unknown '
. $@
unless
$@ =~ /^\n?\S
*Fatal
:/s;
Debug($@); }
if
((
$$opts
{whatsout} =~ /^archive/) && (
$format
!~ /^x?html|xml/)) {
my
$manifest_maker
= LaTeXML::Post::Manifest->new(
db
=>
$DB
,
format
=>
$format
,
log
=>
$$opts
{
log
},
%PostOPS
);
$manifest_maker
->process(
@postdocs
); }
if
(
$$opts
{
log
} && (
$$opts
{whatsout} =~ /^archive/) && (!pathname_is_absolute(
$$opts
{
log
}))) {
my
$destination_directory
=
$PostOPS
{destinationDirectory};
my
$log_file
= pathname_absolute(
$$opts
{
log
},
$destination_directory
);
if
(pathname_is_contained(
$log_file
,
$destination_directory
)) {
NoteLog((
$$opts
{recursive} ?
"recursive "
:
""
) .
"Post-processing complete: "
.
$latexmlpost
->getStatusMessage);
NoteLog((
$$opts
{recursive} ?
"recursive "
:
""
) .
"processing finished "
.
localtime
());
my
$archive_log_status_code
= max(
$$runtime
{status_code},
$latexmlpost
->getStatusCode);
Note(
"Status:conversion:"
.
$archive_log_status_code
);
open
my
$log_fh
,
'>'
,
$log_file
;
print
$log_fh
$self
->flush_log;
close
$log_fh
;
$self
->bind_log; }
else
{ Error(
"I/O"
,
"log"
,
"The target log file isn't contained in the destination directory!"
); } }
my
(
$postdoc
) = pack_collection(
collection
=> [
@postdocs
],
whatsout
=>
$$opts
{whatsout},
format
=>
$format
,
%PostOPS
);
$DB
->finish;
my
$post_status
=
$latexmlpost
->getStatusMessage;
if
(
$post_status
ne
$$runtime
{status}) {
$$runtime
{status} .=
"\n"
.
$post_status
;
}
$$runtime
{status_code} = max(
$$runtime
{status_code},
$latexmlpost
->getStatusCode);
Note((
$$opts
{recursive} ?
"recursive "
:
""
) .
"Post-processing complete: "
.
$latexmlpost
->getStatusMessage);
NoteLog((
$$opts
{recursive} ?
"recursive "
:
""
) .
"processing finished "
.
localtime
());
if
(
$$opts
{destination} &&
$$opts
{
local
} && (
$$opts
{whatsout} eq
'document'
)) {
undef
$postdoc
; }
return
$postdoc
; }
sub
check_TOC {
my
(
$self
,
$document
) =
@_
;
if
(!
$document
->findnode(
'//ltx:TOC[@lists="toc"]'
)) {
my
@s
= (
qw(ltx:part ltx:chapter ltx:section ltx:subsection ltx:subsubsection
ltx:paragraph ltx:subparagraph ltx:appendix ltx:index ltx:bibliography)
);
$document
->prependNodes(
$document
->getDocumentElement,
[
'ltx:TOC'
, {
lists
=>
'toc'
,
scope
=>
'global'
,
select
=>
join
(
' | '
,
@s
),
class
=>
'ltx_nodisplay'
}]); }
return
; }
sub
new_latexml {
my
(
$opts
) =
@_
;
my
$preloads
=
$$opts
{preload};
my
(
@pre
,
@str_pre
);
foreach
my
$pre
(
@$preloads
) {
if
(pathname_is_literaldata(
$pre
)) {
push
@str_pre
,
$pre
;
}
else
{
push
@pre
,
$pre
;
}
}
my
$includepathpis
= !(
exists
$$opts
{xsltparameters} &&
(
grep
{
$_
eq
'LATEXML_VERSION:TEST'
} @{
$$opts
{xsltparameters} }));
SetVerbosity(
$$opts
{verbosity})
if
defined
$$opts
{verbosity};
my
$latexml
= LaTeXML::Core->new(
preload
=> [
@pre
],
searchpaths
=> [@{
$$opts
{paths} }],
graphicspaths
=> [
'.'
],
includecomments
=>
$$opts
{comments},
includepathpis
=>
$includepathpis
,
inputencoding
=>
$$opts
{inputencoding},
includestyles
=>
$$opts
{includestyles},
documentid
=>
$$opts
{documentid},
nomathparse
=>
$$opts
{nomathparse},
mathparse
=>
$$opts
{mathparse});
if
(
my
@baddirs
=
grep
{ !-d
$_
} @{
$$opts
{paths} }) {
warn
"\n$LaTeXML::IDENTITY : these path directories do not exist: "
.
join
(
', '
,
@baddirs
) .
"\n"
; }
$latexml
->withState(
sub
{
my
(
$state
) =
@_
;
$latexml
->initializeState(
'TeX.pool'
, @{
$$latexml
{preload} || [] });
});
$latexml
->digestFile(
$_
,
noinitialize
=> 1)
foreach
(
@str_pre
);
return
$latexml
;
}
sub
bind_log {
my
(
$self
) =
@_
;
my
$opts
=
$$self
{opts};
if
(
my
$logfile
=
$$opts
{
log
}) {
my
$whatsout
=
$$opts
{whatsout};
if
(
$whatsout
&& (
$whatsout
!~ /^archive/)) {
UseLog(
$logfile
, 1); }
else
{
UseLog(\
$$self
{
log
}, 1);
}
}
else
{
UseLog(\
$$self
{
log
}, 1); }
return
; }
sub
flush_log {
my
(
$self
) =
@_
;
UseLog(
undef
);
my
$log
=
$$self
{
log
};
$$self
{
log
} =
q{}
;
return
$log
; }
1;