my
$IS_WINDOWS
;
BEGIN {
$IS_WINDOWS
= $^O eq
'MSWin32'
;
}
our
@EXPORT
= (
qw(&SetVerbosity)
,
qw(&UseSTDERR &UseLog)
,
qw(&Fatal &Error &Warn &Info)
,
qw(&Note &NoteSTDERR &NoteLog)
,
qw(&ProgressSpinup &ProgressSpindown &ProgressStep)
,
qw(&DebuggableFeature &Debug &CheckDebuggable)
,
qw(&colorizeString)
,
qw(&generateMessage)
,
qw(&MergeStatus)
,
qw(&StartTime &RunTime)
,
);
our
$VERBOSITY
= 0;
our
$IS_TERMINAL
=
undef
;
our
$USE_STDERR
=
undef
;
sub
SetVerbosity {
return
$VERBOSITY
=
$_
[0] || 0; }
our
$DIE_MESSAGE
=
"LaTeXML died!\n"
;
$Term::ANSIColor::AUTORESET
= 1;
sub
UseSTDERR {
if
(
scalar
(
@_
) && !
$_
[0]) {
_spinnerclear()
if
$USE_STDERR
&&
$IS_TERMINAL
;
$USE_STDERR
=
undef
;
$IS_TERMINAL
=
undef
; }
else
{
$USE_STDERR
= 1;
$IS_TERMINAL
= -t STDERR;
binmode
(STDERR,
":encoding(UTF-8)"
);
*STDERR
->autoflush();
if
(
$IS_WINDOWS
&&
$IS_TERMINAL
) {
Win32::Console::OutputCP(65001);
our
$W32_STDERR
= Win32::Console->new(
&Win32::Console::STD_ERROR_HANDLE
());
my
$mode
=
$W32_STDERR
->Mode();
unless
(
$W32_STDERR
->Mode(
$mode
| 0x0004) &&
$W32_STDERR
->Mode() & 0x0004) {
return
; }
our
%color_scheme
= (
details
=>
'bold'
,
success
=>
'green'
,
info
=>
'bright_blue'
,
warning
=>
'yellow'
,
error
=>
'bold red'
,
fatal
=>
'bold red underline'
,
);
sub
colorizeString {
my
(
$string
,
$alias
) =
@_
;
return
(
$IS_TERMINAL
&&
$color_scheme
{
$alias
}
? colored(
$string
,
$color_scheme
{
$alias
})
:
$string
); }
our
$LOG
;
our
$LOG_PATH
;
our
$log_count
= 0;
sub
UseLog {
my
(
$path
,
$append
) =
@_
;
if
(!
$path
) {
$log_count
--;
return
if
!
$LOG
||
$log_count
;
print
$LOG
_freshline(
$LOG
);
close
(
$LOG
) or
die
"Cannot close log file: $!"
;
$LOG
=
undef
; }
else
{
$log_count
++;
return
if
$LOG
or not(
$path
);
pathname_mkdir(pathname_directory(
$path
));
open
(
$LOG
, (
$append
?
'>>'
:
'>'
),
$path
) or
die
"Cannot open log file $path for writing: $!"
;
$LOG_PATH
=
$path
;
binmode
(
$LOG
,
":encoding(UTF-8)"
); }
return
; }
sub
_printline {
my
(
$message
) =
@_
;
return
if
(!
$LOG
&& !(
$USE_STDERR
&& (
$VERBOSITY
>= 0)));
$message
=~ s/^\n+//s;
$message
=~ s/\n+$//s;
if
(
my
$clean_message
= (
$LOG
|| !
$IS_TERMINAL
? strip_ansi(
$message
) :
$message
)) {
$message
=
$clean_message
unless
$IS_TERMINAL
;
if
(
$LOG
) {
print
$LOG
_freshline(
$LOG
),
$clean_message
,
"\n"
; }
if
(
$USE_STDERR
&& (
$VERBOSITY
>= 0)) {
_spinnerclear();
my
$short
=
$message
;
if
(
$short
=~ /^([^\n]*)(:?\n\s*(at\s+[^\n]*))?/s) {
my
(
$first
,
$more
,
$at
) = ($1, $2, $3);
$at
=~ s/\s+-\s+.*$//
if
$at
;
$short
=
$first
;
$short
.=
' '
.
$at
if
$at
; }
print
STDERR _freshline(\
*STDERR
),
$short
,
"\n"
;
_spinnerrestore(); } }
return
; }
sub
_printlines {
my
(
$message
) =
@_
;
return
if
(!
$LOG
&& !(
$USE_STDERR
&& (
$VERBOSITY
>= 0)));
$message
=~ s/^\n+//s;
$message
=~ s/\n+$//s;
if
(
my
$clean_message
= (
$LOG
|| !
$IS_TERMINAL
? strip_ansi(
$message
) :
$message
)) {
$message
=
$clean_message
unless
$IS_TERMINAL
;
if
(
$LOG
) {
print
$LOG
_freshline(
$LOG
),
$clean_message
,
"\n"
; }
if
(
$USE_STDERR
&& (
$VERBOSITY
>= 0)) {
_spinnerclear();
print
STDERR _freshline(\
*STDERR
),
$message
,
"\n"
;
_spinnerrestore(); } }
return
; }
our
%NEEDSFRESHLINE
= ();
sub
_freshline {
my
(
$stream
) =
@_
;
if
(
$stream
&&
$NEEDSFRESHLINE
{
$stream
}) {
$NEEDSFRESHLINE
{
$stream
} = 0;
return
"\n"
; }
return
''
; }
sub
strip_ansi {
my
(
$string
) =
@_
;
$string
=~ s/\e\[[0-9;]*[a-zA-Z]//g;
return
$string
; }
sub
StartTime {
return
[Time::HiRes::gettimeofday]; }
sub
RunTime {
my
(
$starttime
) =
@_
;
my
$s
= Time::HiRes::tv_interval(
$starttime
, [Time::HiRes::gettimeofday]);
my
(
$h
,
$m
);
$m
=
int
(
$s
/ 60);
$s
-= 60 *
$m
;
$h
=
int
(
$m
/ 60);
$m
-= 60 *
$h
;
return
(
$h
?
$h
.
'h '
:
''
) . (
$m
?
$m
.
'm '
:
''
) .
sprintf
(
"%.2fs"
,
$s
); }
our
@spinnerstack
= ();
our
@spinnerchar
=
map
{ colored(
$_
,
"bold red"
); } (
'-'
,
'\\'
,
'|'
,
'/'
);
our
$spinnerpos
= 0;
our
$spinnerpre
=
"\x1b[1G\x1b[?7l"
;
our
$spinnerpost
=
"\x1b[?7h"
;
sub
_spinnerclear {
if
(
$USE_STDERR
&&
$IS_TERMINAL
&& (
$VERBOSITY
>= 0) &&
@spinnerstack
) {
print
STDERR
"\x1b[1G\x1b[0K"
; }
return
; }
sub
_spinnerrestore {
if
(
$USE_STDERR
&&
$IS_TERMINAL
&& (
$VERBOSITY
>= 0) &&
@spinnerstack
) {
my
(
$stage
,
$short
,
$start
) = @{
$spinnerstack
[-1] };
print
STDERR
join
(
' '
,
$spinnerpre
,
$spinnerchar
[
$spinnerpos
],
(
map
{
$$_
[1]; }
@spinnerstack
[0 ..
$#spinnerstack
- 1]),
$stage
),
$spinnerpost
; }
return
; }
sub
_spinnerstep {
my
(
$note
) =
@_
;
if
(
$USE_STDERR
&&
$IS_TERMINAL
&& (
$VERBOSITY
>= 0) &&
@spinnerstack
) {
my
(
$stage
,
$short
,
$start
) = @{
$spinnerstack
[-1] };
$spinnerpos
= (
$spinnerpos
+ 1) % 4;
if
(
$note
) {
print
STDERR
join
(
' '
,
$spinnerpre
,
$spinnerchar
[
$spinnerpos
],
(
map
{
$$_
[1]; }
@spinnerstack
),
$note
,
"\x1b[0K"
),
$spinnerpost
; }
else
{
print
STDERR
$spinnerpre
.
' '
,
$spinnerchar
[
$spinnerpos
],
$spinnerpost
; } }
return
; }
sub
_spinnerpush {
my
(
$stage
) =
@_
;
my
$short
= (
$stage
=~ /^(\w+)\s+(.*)$/ && $2 ?
"$1 >"
:
$stage
);
push
(
@spinnerstack
, [
$stage
,
$short
, [Time::HiRes::gettimeofday]]);
return
; }
sub
_spinnerpop {
my
(
$stage
) =
@_
;
if
(
@spinnerstack
&& (
$stage
eq
$spinnerstack
[-1][0])) {
my
(
$xstage
,
$short
,
$start
) = @{
pop
(
@spinnerstack
) };
return
Time::HiRes::tv_interval(
$start
, [Time::HiRes::gettimeofday]); }
elsif
(
$USE_STDERR
&& (
$VERBOSITY
>= 0)) {
print
STDERR
"SPINNER is "
. ((
@spinnerstack
&&
$spinnerstack
[-1][0]) ||
'undef'
) .
" not $stage\n"
; }
return
; }
sub
Fatal {
my
(
$category
,
$object
,
$where
,
$message
,
@details
) =
@_
;
if
(((
$category
eq
'internal'
) && (
$object
eq
'<recursion>'
)) ||
(
$category
eq
'too_many_errors'
) ||
(
$object
eq
'deep_recursion'
) || (
$object
eq
'die'
)) {
$LaTeXML::UNSAFE_FATAL
= 1; }
die
$DIE_MESSAGE
if
$LaTeXML::IGNORE_ERRORS
|| ((
$SIG
{__DIE__} eq
'DEFAULT'
) && $^S);
my
$inhandler
= !
$SIG
{__DIE__};
my
$ineval
= 0;
local
$SIG
{__DIE__} =
'DEFAULT'
;
my
$state
=
$STATE
;
if
(!
$inhandler
) {
local
$LaTeXML::BAILOUT
=
$LaTeXML::BAILOUT
;
if
(checkRecursiveError()) {
$LaTeXML::BAILOUT
= 1;
push
(
@details
,
"Recursive Error!"
); }
$state
->noteStatus(
'fatal'
)
if
$state
&& !
$ineval
;
my
$detail_level
= ((
$VERBOSITY
<= 1) && (
$category
=~ /^(?:timeout|too_many_errors)$/)) ? 0 : 2;
$message
= generateMessage(colorizeString(
"Fatal:"
.
$category
.
":"
. ToString(
$object
),
'fatal'
),
$where
,
$message
,
$detail_level
,
@details
);
}
else
{
$message
=
$details
[0]
if
$details
[0]; }
_printlines(
$message
);
hardYankProcessing();
$LaTeXML::IGNORE_ERRORS
= 1;
die
$DIE_MESSAGE
; }
sub
hardYankProcessing {
my
$state
=
$STATE
;
return
unless
$state
;
my
$stomach
=
$$state
{stomach};
my
$gullet
=
$$stomach
{gullet};
$$stomach
{token_stack} = [];
my
$relax_def
=
$$state
{meaning}{
"\\relax"
}[0];
$state
->assignMeaning(
$LaTeXML::CURRENT_TOKEN
,
$relax_def
,
'global'
)
if
$LaTeXML::CURRENT_TOKEN
;
for
my
$token
(@{
$$gullet
{pushback} }) {
$state
->assignMeaning(
$token
,
$relax_def
,
'global'
); }
if
(
@LaTeXML::LIST
) {
$$stomach
{rescued_boxes} = [
@LaTeXML::LIST
];
@LaTeXML::LIST
= ();
}
if
(
$LaTeXML::DOCUMENT
) {
$$state
{rescued_document} =
$LaTeXML::DOCUMENT
; }
$state
->assignValue(
'current_environment'
,
undef
,
'global'
);
$$gullet
{pushback} = [];
$$gullet
{mouthstack} = [];
$$gullet
{pending_comments} = [];
$$gullet
{mouth} = LaTeXML::Core::Mouth->new();
return
; }
sub
checkRecursiveError {
my
@caller
;
for
(
my
$frame
= 2 ;
@caller
=
caller
(
$frame
) ;
$frame
++) {
if
(
$caller
[3] =~ /^LaTeXML::(Global::ToString|Global::Stringify)$/) {
return
1; } }
return
; }
sub
Error {
my
(
$category
,
$object
,
$where
,
$message
,
@details
) =
@_
;
return
if
$LaTeXML::IGNORE_ERRORS
;
my
$state
=
$STATE
;
$state
&&
$state
->noteStatus(
'error'
);
if
(
$state
&&
$state
->lookupValue(
'STRICT'
)) {
Fatal(
$category
,
$object
,
$where
,
$message
,
@details
); }
else
{
my
$formatted
= generateMessage(
"Error:"
.
$category
.
":"
. ToString(
$object
),
$where
,
$message
, 1,
@details
);
_printline(
$formatted
); }
my
$maxerrors
= (
$state
?
$state
->lookupValue(
'MAX_ERRORS'
) : 100);
if
(
$state
&& (
defined
$maxerrors
) && ((
$state
->getStatus(
'error'
) || 0) >
$maxerrors
)) {
Fatal(
'too_many_errors'
,
$maxerrors
,
$where
,
"Too many errors (> $maxerrors)!"
); }
return
; }
sub
Warn {
my
(
$category
,
$object
,
$where
,
$message
,
@details
) =
@_
;
return
if
$LaTeXML::IGNORE_ERRORS
;
my
$state
=
$STATE
;
$state
&&
$state
->noteStatus(
'warning'
);
my
$formatted
= generateMessage(
"Warning:"
.
$category
.
":"
. ToString(
$object
),
$where
,
$message
, 0,
@details
);
_printline(
$formatted
);
return
; }
sub
Info {
my
(
$category
,
$object
,
$where
,
$message
,
@details
) =
@_
;
return
if
$LaTeXML::IGNORE_ERRORS
;
my
$state
=
$STATE
;
$state
&&
$state
->noteStatus(
'info'
);
my
$formatted
= generateMessage(
"Info:"
.
$category
.
":"
. ToString(
$object
),
$where
,
$message
, -1,
@details
);
_printline(
$formatted
);
return
; }
sub
Note {
my
(
$message
) =
@_
;
_printline(
$message
);
return
; }
sub
NoteSTDERR {
my
(
$message
) =
@_
;
if
(
$USE_STDERR
&& (
$VERBOSITY
>= 0)) {
_spinnerclear();
print
STDERR _freshline(\
*STDERR
),
$message
,
"\n"
;
_spinnerrestore(); }
return
; }
sub
NoteLog {
my
(
$message
) =
@_
;
print
$LOG
_freshline(
$LOG
), strip_ansi(
$message
),
"\n"
if
$LOG
;
return
; }
sub
ProgressStep {
my
(
$note
) =
@_
;
_spinnerstep(
$note
);
return
; }
sub
ProgressSpinup {
my
(
$stage
) =
@_
;
if
(
$LOG
|| (
$USE_STDERR
&& (
$VERBOSITY
>= 0))) {
my
$message
=
"($stage..."
;
_spinnerclear();
_spinnerpush(
$stage
);
_spinnerrestore();
if
(
$LOG
) {
print
$LOG
_freshline(
$LOG
),
$message
;
$NEEDSFRESHLINE
{
$LOG
} = 1
if
$LOG
; }
if
(
$USE_STDERR
&& (
$VERBOSITY
>= 0) && !
$IS_TERMINAL
) {
print
STDERR _freshline(\
*STDERR
),
$message
;
$NEEDSFRESHLINE
{ \
*STDERR
} = 1; } }
return
; }
sub
ProgressSpindown {
my
(
$stage
) =
@_
;
if
(
$LOG
|| (
$USE_STDERR
&& (
$VERBOSITY
>= 0))) {
_spinnerclear();
my
$elapsed
= _spinnerpop(
$stage
);
_spinnerrestore();
my
$message
= (
$elapsed
?
sprintf
(
" %.2f sec)"
,
$elapsed
) :
'?'
);
print
$LOG
$message
if
$LOG
;
$NEEDSFRESHLINE
{
$LOG
} = 1
if
$LOG
;
if
(
$USE_STDERR
&& (
$VERBOSITY
>= 0) && !
$IS_TERMINAL
) {
print
STDERR
$message
;
$NEEDSFRESHLINE
{ \
*STDERR
} = 1; } }
return
; }
our
%Debugbable
= ();
sub
DebuggableFeature {
my
(
$feature
,
$description
) =
@_
;
$LaTeXML::Debuggable
{
$feature
} =
$description
;
return
; }
sub
Debug {
my
(
$message
) =
@_
;
_printlines(
$message
);
return
; }
sub
CheckDebuggable {
my
%unknown
= ();
foreach
my
$feature
(
keys
%LaTeXML::DEBUG
) {
$unknown
{
$feature
} = 1
unless
$LaTeXML::Debuggable
{
$feature
}; }
if
(
keys
%unknown
) {
print
STDERR _freshline(\
*STDERR
),
"The debugging feature(s) "
.
join
(
', '
,
sort
keys
%unknown
) .
" were never declared\n"
;
print
STDERR _freshline(\
*STDERR
),
"Known debugging features: "
.
join
(
', '
,
sort
keys
%LaTeXML::Debuggable
) .
"\n"
; }
return
; }
my
$quoted_re
=
qr/\"([^\"]*)\"/
;
my
$cantcall_re
=
qr/Can't call method/
;
my
$cantlocate_re
=
qr/Can't locate object method/
;
my
$undef_re
=
qr/Undefined subroutine/
;
my
$noself_re
=
qr/on an undefined value|without a package or object reference/
;
my
$via_re
=
qr/via package/
;
my
$at_re
=
qr/(at .*)/
;
sub
perl_die_handler {
my
(
@line
) =
@_
;
if
(
$LaTeXML::IGNORE_ERRORS
|| (colorstrip(
$line
[0]) =~ /^\s
*Fatal
:/)) {
local
$SIG
{__DIE__} =
undef
;
die
@line
; }
if
(
$line
[0] =~ /^
$cantcall_re
\s+
$quoted_re
\s+(
$noself_re
)\s+
$at_re
$/) {
my
(
$method
,
$kind
,
$where
) = ($1, $2, $3);
Fatal(
'misdefined'
, callerName(1),
$where
,
"Can't call method '$method' $kind"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^
$undef_re
\s+(\S+)\s+called
$at_re
$/) {
my
(
$function
,
$where
) = ($1, $2);
Fatal(
'misdefined'
, callerName(1),
$where
,
"Undefined subroutine '$function' called"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^
$cantlocate_re
\s+
$quoted_re
\s+
$via_re
\s+
$quoted_re
\s+\(.*\)\s+
$at_re
/) {
my
(
$method
,
$class
,
$where
) = ($1, $2, $3);
Fatal(
'misdefined'
, callerName(1),
$where
,
"Can't locate method '$method' via '$class'"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^Can't locate \S* in \
@INC
\(you may need to install the (\S*) module\) \(\
@INC
contains: ([^\)]*)\)
$at_re
$/) {
my
(
$class
,
$inc
,
$where
) = ($1, $2);
Fatal(
'misdefined'
, callerName(1),
$where
,
"Can't locate class '$class' (not installed or misspelled?)"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^Can't
use
\s+(\w*)\s+\([^\)]*\) as (.*?)
ref
(?:\s+
while
"strict refs"
in
use
)? at (.*)$/) {
my
(
$gottype
,
$wanttype
,
$where
) = ($1, $2, $3);
Fatal(
'misdefined'
, callerName(1),
$where
,
"Can't use $gottype as $wanttype reference"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^File (.*?) had an error:/) {
my
(
$file
) = ($1);
Fatal(
'misdefined'
,
$file
,
undef
,
@line
); }
else
{
Fatal(
'perl'
,
'die'
,
undef
,
"Perl died"
,
@line
); }
return
; }
sub
perl_warn_handler {
my
(
@line
) =
@_
;
return
if
$LaTeXML::IGNORE_ERRORS
;
if
(
$line
[0] =~ /^Use of uninitialized value (.*?)(\s?+in .*?)\s+(at\s+.*?\s+line\s+\d+)\.$/) {
my
(
$what
,
$how
,
$where
) = ($1 ||
'value'
, $2, $3);
Warn(
'uninitialized'
,
$what
,
$where
,
"Use of uninitialized value $what $how"
,
@line
[1 ..
$#line
]); }
elsif
(
$line
[0] =~ /^Deep recursion on/) {
Fatal(
'perl'
,
'deep_recursion'
,
undef
,
$line
[0]); }
elsif
(
$line
[0] =~ /^(.*?)\s+(at\s+.*?\s+line\s+\d+)\.$/) {
my
(
$warning
,
$where
) = ($1, $2);
Warn(
'perl'
,
'warn'
,
undef
,
$warning
,
$where
,
@line
[1 ..
$#line
]); }
else
{
Warn(
'perl'
,
'warn'
,
undef
,
@line
); }
return
; }
sub
perl_interrupt_handler {
my
(
@line
) =
@_
;
$LaTeXML::IGNORE_ERRORS
= 0;
$LaTeXML::UNSAFE_FATAL
= 1;
Fatal(
'interrupt'
,
'interrupted'
,
undef
,
"LaTeXML was interrupted"
,
@line
);
return
; }
sub
perl_timeout_handler {
my
(
@line
) =
@_
;
$LaTeXML::IGNORE_ERRORS
= 0;
$LaTeXML::UNSAFE_FATAL
= 1;
Fatal(
'timeout'
,
'timedout'
,
undef
,
"Conversion timed out"
,
@line
);
return
; }
sub
perl_terminate_handler {
my
(
@line
) =
@_
;
$LaTeXML::IGNORE_ERRORS
= 0;
$LaTeXML::UNSAFE_FATAL
= 1;
Fatal(
'terminate'
,
'terminated'
,
undef
,
"Conversion was terminated"
,
@line
);
return
; }
sub
generateMessage {
my
(
$errorcode
,
$where
,
$message
,
$detail
,
@extra
) =
@_
;
if
(
$USE_STDERR
&&
$IS_TERMINAL
&& (
$VERBOSITY
>= 0)) {
$errorcode
=~ /^(\w+)\:/;
my
$errorkind
= $1 &&
lc
($1);
$errorcode
= colorizeString(
$errorcode
,
$errorkind
)
if
$errorkind
; }
my
$docloc
= getLocation(
$where
);
$docloc
=
defined
$docloc
? ToString(
$docloc
) :
''
;
@extra
=
grep
{
$_
ne
''
}
map
{
split
(
"\n"
,
$_
) }
grep
{
defined
$_
}
$message
,
@extra
;
$message
=
shift
(
@extra
);
$message
=
''
unless
defined
$message
;
my
@lines
= (
$errorcode
.
' '
.
$message
,
(
$docloc
? (
'at '
.
$docloc
) : ()),
@extra
);
$detail
= 0
unless
defined
$detail
;
if
((
$detail
> -1) && (
$VERBOSITY
> 0)) {
$detail
= 0
if
defined
$VERBOSITY
&&
$VERBOSITY
< -1;
$detail
++
if
defined
$VERBOSITY
&&
$VERBOSITY
> +1; }
my
$wheretype
=
ref
$where
;
if
(
$detail
<= 0) { }
elsif
(
$wheretype
=~ /^XML::LibXML/) {
push
(
@lines
,
"Node is "
. Stringify(
$where
)); }
elsif
(
$wheretype
=~
'LaTeXML::Core::Gullet'
) {
push
(
@lines
,
$where
->showUnexpected); }
elsif
(
$wheretype
=~
'LaTeXML::Core::Stomach'
) {
push
(
@lines
,
"Recently digested: "
.
join
(
' '
,
map
{ Stringify(
$_
) }
@LaTeXML::LIST
))
if
$VERBOSITY
> 1; }
if
((
$detail
> 1) && (
$VERBOSITY
> 0)) {
push
(
@lines
,
"Stack Trace:"
, stacktrace()); }
elsif
(
$detail
> -1) {
my
$nstack
= (
$detail
> 1 ?
undef
: (
$detail
> 0 ? 4 : 1));
if
(
my
@objects
= objectStack(
$nstack
)) {
my
$top
=
shift
(
@objects
);
push
(
@lines
,
"In "
. trim(Stringify(
$$top
[0])) .
' '
. Stringify(
$$top
[1]));
push
(
@objects
, [
'...'
])
if
@objects
&&
defined
$nstack
;
push
(
@lines
,
join
(
''
, (
map
{
' <= '
. trim(Stringify(
$$_
[0])) }
@objects
)))
if
@objects
;
} }
return
join
(
"\n\t"
,
@lines
); }
sub
MergeStatus {
my
(
$external_state
) =
@_
;
my
$state
=
$STATE
;
return
unless
$state
&&
$external_state
;
my
$status
=
$$state
{status};
my
$external_status
=
$$external_state
{status};
foreach
my
$type
(
keys
%$external_status
) {
if
(
$type
eq
'undefined'
or
$type
eq
'missing'
) {
my
$table
=
$$external_status
{
$type
};
foreach
my
$subtype
(
keys
%$table
) {
$$status
{
$type
}{
$subtype
} +=
$$table
{
$subtype
};
}
}
else
{
$$status
{
$type
} +=
$$external_status
{
$type
};
}
}
return
; }
sub
Locator {
my
(
$object
) =
@_
;
return
(
$object
&&
$object
->can(
'getLocator'
) ?
$object
->getLocator :
undef
); }
sub
getLocation {
my
(
$where
) =
@_
;
my
$wheretype
=
ref
$where
;
if
(
$wheretype
&& (
$wheretype
=~ /^XML::LibXML/)) {
my
$box
=
$LaTeXML::DOCUMENT
&&
$LaTeXML::DOCUMENT
->getNodeBox(
$where
);
return
Locator(
$box
)
if
$box
; }
elsif
(
$wheretype
&&
$where
->can(
'getLocator'
)) {
return
$where
->getLocator; }
elsif
(
defined
$where
) {
return
$where
; }
elsif
(
$LaTeXML::DOCUMENT
) {
my
$node
=
$LaTeXML::DOCUMENT
->getNode;
my
$box
=
$LaTeXML::DOCUMENT
->getNodeBox(
$node
);
return
Locator(
$box
)
if
$box
; }
if
(
$LaTeXML::BOX
) {
return
Locator(
$LaTeXML::BOX
); }
if
(
$STATE
&&
$STATE
->getStomach) {
my
$gullet
=
$STATE
->getStomach->getGullet;
return
Locator(
$gullet
); }
return
; }
sub
callerName {
my
(
$frame
) =
@_
;
my
%info
= caller_info((
$frame
|| 0) + 2);
return
$info
{
sub
}; }
sub
callerInfo {
my
(
$frame
) =
@_
;
my
%info
= caller_info((
$frame
|| 0) + 2);
return
"$info{call} @ $info{file} line $info{line}"
; }
my
$MAXARGS
= 8;
my
$MAXLEN
= 40;
sub
trim {
my
(
$string
) =
@_
;
return
$string
unless
defined
$string
;
$string
=
substr
(
$string
, 0,
$MAXLEN
- 3) .
"..."
if
(
length
(
$string
) >
$MAXLEN
);
$string
=~ s/\n/\x{240D}/gs;
return
$string
; }
sub
caller_info {
my
(
$i
) =
@_
;
my
(
%info
,
@args
);
@info
{
qw(package file line sub has_args wantarray evaltext is_require)
}
=
caller
(
$i
);
@args
=
@DB::args
; }
return
()
unless
defined
$info
{
package
};
my
$call
=
''
;
if
(
defined
$info
{evaltext}) {
my
$eval
=
$info
{evaltext};
if
(
$info
{is_require}) {
$call
=
"require $eval"
; }
else
{
$eval
=~ s/([\\\'])/\\$1/g;
$call
=
"eval '"
. trim(
$eval
) .
"'"
; } }
elsif
(
$info
{
sub
} eq
'(eval)'
) {
$call
=
"eval {...}"
; }
else
{
$call
=
$info
{
sub
};
my
$method
=
$call
;
$method
=~ s/^.*:://;
if
(
$info
{has_args} &&
@args
&&
ref
$args
[0] && ((
ref
$args
[0]) !~ /^(?:SCALAR|ARRAY|HASH|CODE|REF|GLOB|LVALUE)$/)
&&
$args
[0]->can(
$method
)) {
$call
= format_arg(
shift
(
@args
)) .
"->"
.
$method
; } }
if
(
$info
{has_args}) {
@args
=
map
{ format_arg(
$_
) }
@args
;
if
(
@args
>
$MAXARGS
) {
$#args
=
$MAXARGS
;
push
(
@args
,
'...'
); }
$call
.=
'('
.
join
(
','
,
@args
) .
')'
; }
$info
{call} =
$call
;
return
%info
; }
sub
format_arg {
my
(
$arg
) =
@_
;
if
(not
defined
$arg
) {
$arg
=
'undef'
; }
elsif
(
ref
$arg
) {
$arg
= Stringify(
$arg
); }
elsif
(
$arg
=~ /^-?[\d.]+\z/) { }
else
{
$arg
=~ s/
'/\\'
/g;
$arg
=~ s/([[:cntrl:]])/
"\\"
.
chr
(
ord
($1)+
ord
(
'A'
))/ge;
$arg
=
"'$arg'"
}
return
trim(
$arg
); }
sub
stacktrace {
my
$frame
= 0;
my
$trace
=
""
;
while
(
my
%info
= caller_info(
$frame
++)) {
next
if
$info
{
sub
} =~ /^LaTeXML::Common::Error/;
$trace
.=
"\t$info{call} @ $info{file} line $info{line}\n"
; }
return
$trace
; }
sub
objectStack {
my
(
$maxdepth
) =
@_
;
my
$frame
= 0;
my
@objects
= ();
while
(1) {
my
(
%info
,
@args
);
@info
{
qw(package file line sub has_args wantarray evaltext is_require)
} =
caller
(
$frame
++);
@args
=
@DB::args
; }
last
unless
defined
$info
{
package
};
next
if
(
$info
{
sub
} eq
'(eval)'
) || !
$info
{has_args} || !
@args
;
my
$self
=
$args
[0];
if
((
ref
$self
) && ((
ref
$self
) !~ /^(?:SCALAR|ARRAY|HASH|CODE|REF|GLOB|LVALUE)$/)) {
my
$method
=
$info
{
sub
};
$method
=~ s/^.*:://;
if
(
$self
->can(
$method
)) {
next
if
@objects
&& (
$self
eq
$objects
[-1][0]);
if
(
$self
->can(
'getLocator'
)) {
push
(
@objects
, [
$self
, Locator(
$self
)]); }
elsif
(
$self
->isa(
'LaTeXML::Post::Processor'
) ||
$self
->isa(
'LaTeXML::Post::Document'
)) {
push
(
@objects
, [
$self
,
'->'
.
$method
]); }
last
if
$maxdepth
&& (
scalar
(
@objects
) >=
$maxdepth
); } } }
return
@objects
; }
1;