2014-01-30 47 views
9

我用这段代码加载以下.yaml文件的Perl YAML模块不能识别的标量类型的

foo : bar 
s : 1 
hx: 0x34 

use YAML qw(LoadFile Dump); 
use Data::Dumper; 
my $d=LoadFile("test.yaml"); 
print Dumper($d); 
print "x  =", $d->{hx},"\n"; 
print "x/2  =", $d->{hx}/2,"\n"; 
print "hex(x)/2 =", hex($d->{hx})/2,"\n"; 

并且输出是

Output: 
> ./yaml.pl 
$VAR1 = { 
      'foo' => 'bar', 
      'hx' => '0x34', 
      's' => '1' 
     }; 
x  =0x34 
x/2  =0 
hex(x)/2 =26 

这意味着它将所有标量视为字符串,这是我没有想到的。从CPAN http://metacpan.org/pod/YAML的YAML文档中,我认为这是通过隐式标记处理的,并且模块会识别十六进制数。

有谁知道如何做到这一点?

+4

这可以说是一个bug,取决于YAML处理器使用哪个[schema](http://www.yaml.org/spec/1.2/spec.html#Schema)(您的程序行为与Failsafe或JSON一致,但与Core不同)。您可以查看该模块的[问题](https://rt.cpan.org/Public/Dist/Display.html?Name=YAML),然后查看之前是否报告过这种差异,或者在那里发现新问题。 – amon

+0

您是否可以安装并尝试使用YAML :: XS?当我使用XS版本测试你的yaml输入时,1没有被引用。但是,hx值仍被视为字符串。 – user3183018

+0

我如何测试'/ usr/bin/perl -MYAML :: Any = LoadFile -MData :: Dumper -e'$ y = LoadFile($ ARGV [0]);打印Dumper($ y)。 “\ n”'$ YAMLFILE' – user3183018

回答

2

如果您检查加载器类,您会发现_parse_inline子对序列,映射,单引号和双引号字符串以及其他一些情况都有几个分支。您只需为0x开始的值添加分支。我写了那个分支和一个处理YAML 0.900.0的十六进制值的子。我尝试了一些基本的输入,它似乎做你想要的。如果它也适用于您,请考虑提交修补程序。

dump.pl

use YAML qw(LoadFile Dump); 
use Data::Dumper; 
my $d=LoadFile("test.yaml"); 
$d=LoadFile("test.yaml"); 
print Dumper($d); 
print "x  =", $d->{hx},"\n"; 
print "x/2  =", $d->{hx}/2,"\n"; 
print "hex(x)/2 =", hex($d->{hx})/2,"\n"; 



$d=LoadFile("sym1.yaml"); 
print Dumper($d); 

print "dodo[4] =", $d->{dodo}->{doto}[4], "\n"; 
print "dodo[3] =", $d->{dodo}->{doto}[3], "\n"; 

sym1.yaml

"symfony 1.0": 
    PHP: 5.0 
    Propel: 1.2 
"symfony 1.2": 
    PHP: 5.2 
    Propel: 1.3 

dodo: 
    doto: [1,23,4, 0x34, 0x16 ] 
    dozo: [1,23,4, 0x12, 0x11 ] 
    dofo: { a: 2, 358: 0x166, 255: 0xff, 255: 0xFF} 
ffa: 0xFfA 

$ perl的dump.pl

$VAR1 = { 
      'foo' => 'bar', 
      'hx' => 52, 
      's' => '1' 
     }; 
x  =52 
x/2  =26 
hex(x)/2 =41 
$VAR1 = { 
      'ffa' => 4090, 
      'symfony 1.2' => { 
          'PHP' => '5.2', 
          'Propel' => '1.3' 
         }, 
      'dodo' => { 
        'dofo' => { 
           'a' => '2', 
           '255' => 255, 
           '358' => 358 
          }, 
        'dozo' => [ '1','23', '4', 18, 17 ], 
        'doto' => [ '1', '23', '4', 52, 22 ] 
        }, 
      'symfony 1.0' => { 
          'PHP' => '5.0', 
          'Propel' => '1.2' 
         } 
     }; 

doto[4] =22 
doto[3] =52 

补丁YAML 0.900.0

diff --git a/Loader.pm b/Loader.pm 
index 3bf20c7..d7096df 100644 
--- a/Loader.pm 
+++ b/Loader.pm 
@@ -437,6 +437,10 @@ sub _parse_inline { 
     $node = $self->_parse_inline_single_quoted(); 
     $node = $self->_parse_implicit($node) if $implicit; 
    } 
+ elsif ($self->inline =~ /^0x/) { 
+  $node = $self->_parse_inline_single_hex(); 
+  # do something if implicit? 
+ } 
    else { 
     if ($top) { 
      $node = $self->inline; 
@@ -541,6 +545,21 @@ sub _parse_inline_single_quoted { 
    return $node; 
} 

+# Parse the inline hex value 
+sub _parse_inline_single_hex { 
+ my $self = shift; 
+ my $node; 
+ if ($self->inline =~ /^0x([A-Fa-f0-9]+)(,?.*)?$/) { 
+  $node = hex($1); 
+  $self->inline($2); 
+  $node =~ s/''/'/g; 
+ } 
+ else { 
+  $self->die('YAML_PARSE_ERR_BAD_HEX'); 
+ } 
+ return $node; 
+} 
+ 
# Parse the inline unquoted string and do implicit typing. 
sub _parse_inline_simple { 
    my $self = shift;