Files
scripts/Perl OTRS/Kernel/System/YAML.pm
2024-10-14 00:08:40 +02:00

216 lines
5.1 KiB
Perl

# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --
package Kernel::System::YAML;
use strict;
use warnings;
use Encode qw();
use YAML::Any qw();
use YAML qw();
our @ObjectDependencies = (
'Kernel::System::Log',
);
=head1 NAME
Kernel::System::YAML - YAML wrapper functions
=head1 DESCRIPTION
Functions for YAML serialization / deserialization.
=head2 new()
create a YAML object. Do not use it directly, instead use:
my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');
=cut
sub new {
my ( $Type, %Param ) = @_;
# allocate new hash for object
my $Self = {};
bless( $Self, $Type );
return $Self;
}
=head2 Dump()
Dump a perl data structure to a YAML string.
my $YAMLString = $YAMLObject->Dump(
Data => $Data,
);
=cut
sub Dump {
my ( $Self, %Param ) = @_;
# check for needed data
if ( !defined $Param{Data} ) {
$Kernel::OM->Get('Kernel::System::Log')->Log(
Priority => 'error',
Message => 'Need Data!',
);
return;
}
my $Result = YAML::Any::Dump( $Param{Data} ) || "--- ''\n";
# Make sure the resulting string has the UTF-8 flag.
Encode::_utf8_on($Result);
return $Result;
}
=head2 Load()
Load a YAML string to a perl data structure.
This string must be a encoded in UTF8.
my $PerlStructureScalar = $YAMLObject->Load(
Data => $YAMLString,
);
=cut
sub Load {
my ( $Self, %Param ) = @_;
# check for needed data
return if !defined $Param{Data};
if ( Encode::is_utf8( $Param{Data} ) ) {
Encode::_utf8_off( $Param{Data} );
}
# There is a currently a problem with YAML loader it cant load YAML representations of:
# 0, '0' or ''.
# This workaround looks for this special cases and returns the correct value without using the
# loader
if ( $Param{Data} =~ m{\A---[ ](?: '0' | 0 )\n\z}msx ) {
return 0;
}
elsif ( $Param{Data} eq "--- ''\n" ) {
return '';
}
my $Result;
# get used YAML implementation
my $YAMLImplementation = YAML::Any->implementation();
if ( !eval { $Result = YAML::Any::Load( $Param{Data} ) } ) {
$Kernel::OM->Get('Kernel::System::Log')->Log(
Priority => 'error',
Message => 'Loading the YAML string failed: ' . $@,
);
my $DumpString = $Param{Data};
if ( length $DumpString > 1000 ) {
$DumpString = substr( $DumpString, 0, 1000 ) . '[...]';
}
$Kernel::OM->Get('Kernel::System::Log')->Log(
Priority => 'error',
Message => 'YAML data was: "' . $DumpString . '"',
);
# if used implementation is pure perl YAML there is nothing to do, but exit with error
return if $YAMLImplementation eq 'YAML';
# otherwise use pure-perl YAML as fallback if YAML::XS or other can't parse the data
# structure correctly
if ( !eval { $Result = YAML::Load( $Param{Data} ) } ) {
$Kernel::OM->Get('Kernel::System::Log')->Log(
Priority => 'error',
Message => 'YAML data was not readable even by pure-perl YAML module',
);
return;
}
$Kernel::OM->Get('Kernel::System::Log')->Log(
Priority => 'error',
Message => 'Data was only readable pure-perl YAML module, please contact the'
. ' System Administrator to update this record, as the stored data is still in a'
. ' wrong format!',
);
}
# YAML does not set the UTF8 flag on strings that need it, do that manually now.
if ( $YAMLImplementation eq 'YAML' && defined $Result ) {
_AddUTF8Flag( \$Result );
}
return $Result;
}
=begin Internal:
=head2 _AddUTF8Flag()
adds the UTF8 flag to all elements in a complex data structure.
=cut
sub _AddUTF8Flag {
my ($Data) = @_;
if ( !ref ${$Data} ) {
Encode::_utf8_on( ${$Data} );
return;
}
if ( ref ${$Data} eq 'SCALAR' ) {
return _AddUTF8Flag( ${$Data} );
}
if ( ref ${$Data} eq 'HASH' ) {
KEY:
for my $Key ( sort keys %{ ${$Data} } ) {
next KEY if !defined ${$Data}->{$Key};
_AddUTF8Flag( \${$Data}->{$Key} );
}
return;
}
if ( ref ${$Data} eq 'ARRAY' ) {
KEY:
for my $Key ( 0 .. $#{ ${$Data} } ) {
next KEY if !defined ${$Data}->[$Key];
_AddUTF8Flag( \${$Data}->[$Key] );
}
return;
}
if ( ref ${$Data} eq 'REF' ) {
return _AddUTF8Flag( ${$Data} );
}
return;
}
1;
=end Internal:
=head1 TERMS AND CONDITIONS
This software is part of the OTRS project (L<https://otrs.org/>).
This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.
=cut