22 |
|
|
23 |
## Namespace support |
## Namespace support |
24 |
our %NS_phname2uri; ## PH-namespace name -> namespace URI |
our %NS_phname2uri; ## PH-namespace name -> namespace URI |
25 |
our %NS_uri2phpackage; ## namespace URI -> PH-package name |
our %NS_uri2package; ## namespace URI -> Package name |
26 |
|
our %NS_uri2phpackage; ## namespace URI -> PH-Package name |
27 |
require Message::Header::Default; ## Default namespace |
require Message::Header::Default; ## Default namespace |
28 |
|
|
29 |
## Initialize of this class -- called by constructors |
## Initialize of this class -- called by constructors |
41 |
-format => 'mail-rfc2822', |
-format => 'mail-rfc2822', |
42 |
-header_default_charset => 'iso-2022-int-1', |
-header_default_charset => 'iso-2022-int-1', |
43 |
-header_default_charset_input => 'iso-2022-int-1', |
-header_default_charset_input => 'iso-2022-int-1', |
44 |
|
-hook_init_fill_options => sub {}, |
45 |
|
-hook_stringify_fill_fields => sub {}, |
46 |
-linebreak_strict => 0, |
-linebreak_strict => 0, |
47 |
-line_length_max => 60, ## For folding |
-line_length_max => 60, ## For folding |
48 |
#ns_default_phuri |
#ns_default_phuri |
118 |
push @new_fields, ($name => $options{$name}); |
push @new_fields, ($name => $options{$name}); |
119 |
} |
} |
120 |
} |
} |
121 |
|
&{ $self->{option}->{hook_init_fill_options} } ($self, $self->{option}); |
122 |
$self->_init_by_format ($self->{option}->{format}, $self->{option}); |
$self->_init_by_format ($self->{option}->{format}, $self->{option}); |
123 |
# Make alternative representations of @header_order. This is used |
# Make alternative representations of @header_order. This is used |
124 |
# for sorting. |
# for sorting. |
134 |
sub _init_by_format ($$\%) { |
sub _init_by_format ($$\%) { |
135 |
my $self = shift; |
my $self = shift; |
136 |
my ($format, $option) = @_; |
my ($format, $option) = @_; |
137 |
|
return if $format eq $option->{format}; |
138 |
if ($format =~ /http/) { |
if ($format =~ /http/) { |
139 |
$option->{ns_default_phuri} = $self->{ns}->{phname2uri}->{'x-http'}; |
$option->{ns_default_phuri} = $self->{ns}->{phname2uri}->{'x-http'}; |
140 |
if ($format =~ /cgi/) { |
if ($format =~ /cgi/) { |
144 |
} else { |
} else { |
145 |
$option->{field_sort} = 'good-practice'; |
$option->{field_sort} = 'good-practice'; |
146 |
} |
} |
147 |
} else { ## RFC 822 |
} elsif ($format =~ /mail|news/) { ## RFC 822 |
148 |
$option->{ns_default_phuri} = $self->{ns}->{phname2uri}->{'x-rfc822'}; |
$option->{ns_default_phuri} = $self->{ns}->{phname2uri}->{'x-rfc822'}; |
149 |
} |
} |
150 |
if ($format =~ /uri-url-mailto/) { |
if ($format =~ /uri-url-mailto/) { |
193 |
} else { |
} else { |
194 |
$header =~ s/\x0D?\x0A$REG{WSP}/\x20/gos if $self->{option}->{use_folding}; |
$header =~ s/\x0D?\x0A$REG{WSP}/\x20/gos if $self->{option}->{use_folding}; |
195 |
} |
} |
196 |
|
my %option = (%{ $self->{option} }); |
197 |
|
$option{parse_all} = 0; |
198 |
for my $field (split /\x0D?\x0A/, $header) { |
for my $field (split /\x0D?\x0A/, $header) { |
199 |
if ($field =~ /$REG{M_fromline}/) { |
if ($field =~ /$REG{M_fromline}/) { |
200 |
my ($s,undef,$value) = $self->_value_to_arrayitem |
my ($s,undef,$value) = $self->_value_to_arrayitem |
201 |
('mail-from' => $1, $self->{option}); |
('mail-from' => $1, \%option); |
202 |
push @{$self->{value}}, $value if $s; |
push @{$self->{value}}, $value if $s; |
203 |
} elsif ($field =~ /$REG{M_field}/) { |
} elsif ($field =~ /$REG{M_field}/) { |
204 |
my ($name, $body) = ($1, $2); |
my ($name, $body) = ($1, $2); |
205 |
$body =~ s/$REG{WSP}+$//; |
$body =~ s/$REG{WSP}+$//; |
206 |
my ($s,undef,$value) = $self->_value_to_arrayitem |
my ($s,undef,$value) = $self->_value_to_arrayitem |
207 |
($name => $body, $self->{option}); |
($name => $body, \%option); |
208 |
push @{$self->{value}}, $value if $s; |
push @{$self->{value}}, $value if $s; |
209 |
} elsif (length $field) { |
} elsif (length $field) { |
210 |
my ($s,undef,$value) = $self->_value_to_arrayitem |
my ($s,undef,$value) = $self->_value_to_arrayitem |
211 |
('x-unknown' => $field, $self->{option}); |
('x-unknown' => $field, \%option); |
212 |
push @{$self->{value}}, $value if $s; |
push @{$self->{value}}, $value if $s; |
213 |
} |
} |
214 |
} |
} |
215 |
$self->_ns_associate_numerical_prefix; ## RFC 2774 namespace |
$self->_ns_associate_numerical_prefix; ## RFC 2774 namespace |
216 |
|
for (@{ $self->{value} }) { |
217 |
|
no strict 'refs'; |
218 |
|
$_->{name} |
219 |
|
= &{ ${ &_NS_uri2package ($_->{ns}).'::OPTION' }{n11n_name} } |
220 |
|
($self, &_NS_uri2package ($_->{ns}), $_->{name}); |
221 |
|
$_->{body} = $self->_parse_value ($_->{name} => $_->{body}, ns => $_->{ns}) |
222 |
|
if $self->{option}->{parse_all}; |
223 |
|
} |
224 |
$self; |
$self; |
225 |
} |
} |
226 |
|
|
302 |
my %o = %$option; #$o{parse} = 0; |
my %o = %$option; #$o{parse} = 0; |
303 |
my %l; |
my %l; |
304 |
for (keys %$list) { |
for (keys %$list) { |
305 |
my ($s, undef, $v) = $self->_value_to_arrayitem ($_, '', %o); |
my ($s, undef, $v) = $self->_value_to_arrayitem ($_, '', \%o); |
306 |
if ($s) { |
if ($s) { |
307 |
$l{$v->{name} . ':' . ( $option->{ns} || $v->{ns} ) } = 1; |
$l{$v->{name} . ':' . ( $option->{ns} || $v->{ns} ) } = 1; |
308 |
} else { |
} else { |
366 |
my $value = shift; return $value if ref $value; |
my $value = shift; return $value if ref $value; |
367 |
my %option = @_; |
my %option = @_; |
368 |
my $vtype; { no strict 'refs'; |
my $vtype; { no strict 'refs'; |
369 |
my $vt = ${&_NS_uri2phpackage ($option{ns}).'::OPTION'}{value_type}; |
my $vt = ${&_NS_uri2package ($option{ns}).'::OPTION'}{value_type}; |
370 |
if (ref $vt) { |
if (ref $vt) { |
371 |
$vtype = $vt->{$name} || $vt->{$self->{option}->{_VALTYPE_DEFAULT}}; |
$vtype = $vt->{$name} || $vt->{$self->{option}->{_VALTYPE_DEFAULT}}; |
372 |
} |
} |
430 |
if (defined $_[0]) { |
if (defined $_[0]) { |
431 |
no strict 'refs'; |
no strict 'refs'; |
432 |
$self->{option}->{ns_default_phuri} = $_[0]; |
$self->{option}->{ns_default_phuri} = $_[0]; |
433 |
$self->_ns_load_ph (${&_NS_uri2phpackage ($self->{option}->{ns_default_phuri}).'::OPTION'}{namespace_phname}); |
$self->_ns_load_ph (${&_NS_uri2package ($self->{option}->{ns_default_phuri}).'::OPTION'}{namespace_phname}); |
434 |
} |
} |
435 |
$self->{option}->{ns_default_phuri}; |
$self->{option}->{ns_default_phuri}; |
436 |
} |
} |
471 |
if (ref $value eq 'ARRAY') { |
if (ref $value eq 'ARRAY') { |
472 |
($value, %$value_option) = @$value; |
($value, %$value_option) = @$value; |
473 |
} |
} |
474 |
my $nsuri = $self->{option}->{ns_default_phuri}; |
my $default_ns = $option->{ns_default_phuri}; |
475 |
|
my $nsuri = $default_ns; |
476 |
$name =~ s/^$REG{WSP}+//; $name =~ s/$REG{WSP}+$//; |
$name =~ s/^$REG{WSP}+//; $name =~ s/$REG{WSP}+$//; |
477 |
|
|
478 |
no strict 'refs'; |
no strict 'refs'; |
480 |
$nsuri = $value_option->{ns}; |
$nsuri = $value_option->{ns}; |
481 |
} elsif ($option->{ns}) { |
} elsif ($option->{ns}) { |
482 |
$nsuri = $option->{ns}; |
$nsuri = $option->{ns}; |
483 |
} elsif (($option->{ns_default_phuri} eq $self->{ns}->{uri2phname}->{'x-http'} |
} elsif (($default_ns eq $self->{ns}->{uri2phname}->{'x-http'} |
484 |
&& $name =~ s/^([0-9]+)-//) |
&& $name =~ s/^([0-9]+)-//) |
485 |
|| ($name =~ s/^x-http-([0-9]+)-//i)) { ## Numric namespace prefix, RFC 2774 |
|| ($name =~ s/^x-http-([0-9]+)-//i)) { ## Numric namespace prefix, RFC 2774 |
486 |
my $prefix = 0+$1; |
my $prefix = 0+$1; |
489 |
$self->{ns}->{number2uri}->{ $prefix } = 'urn:x-suika-fam-cx:msgpm:header:x-temp:'.$prefix; |
$self->{ns}->{number2uri}->{ $prefix } = 'urn:x-suika-fam-cx:msgpm:header:x-temp:'.$prefix; |
490 |
$nsuri = $self->{ns}->{number2uri}->{ $prefix }; |
$nsuri = $self->{ns}->{number2uri}->{ $prefix }; |
491 |
} |
} |
492 |
} elsif (($name =~ s/^([Xx]-[A-Za-z]+|[A-YZa-yz][A-Za-z]*)- |
} elsif ( |
493 |
([Xx]-[A-Za-z]+|[A-YZa-yz][A-Za-z]*)-//x) |
${ &_NS_uri2package ($default_ns).'::OPTION' }{use_ph_namespace} |
494 |
|| $name =~ s/^([Xx]-[A-Za-z]+|[A-Za-z]+)-//) { |
&& ( |
495 |
|
($name =~ s/^([Xx]-[A-Za-z0-9]+|[A-Za-z]*[A-WYZa-wyz0-9][A-Za-z0-9]*)- |
496 |
|
([Xx]-[A-Za-z0-9]+|[A-Za-z0-9]*[A-WYZa-wyz0-9][A-Za-z0-9]*)-//x) |
497 |
|
|| $name =~ s/^([Xx]-[A-Za-z0-9]+|[A-Za-z0-9]*[A-WYZa-wyz0-9][A-Za-z0-9]*)-// |
498 |
|
)) { |
499 |
my ($prefix1, $prefix2) = ($1, $2); |
my ($prefix1, $prefix2) = ($1, $2); |
500 |
my $original_prefix = $&; my $one_prefix = 0; |
my $original_prefix = $&; my $one_prefix = 0; |
501 |
unless ($prefix2) { |
unless ($prefix2) { |
502 |
$prefix2 = $prefix1; |
$prefix2 = $prefix1; |
503 |
$prefix1 = $self->{ns}->{uri2phname}->{ $option->{ns_default_phuri} }; |
$prefix1 = $self->{ns}->{uri2phname}->{ $default_ns }; |
504 |
$one_prefix = 1; |
$one_prefix = 1; |
505 |
} |
} |
506 |
my $prefix |
my $prefix |
507 |
= &{ ${ &_NS_uri2phpackage ($nsuri).'::OPTION' }{n11n_prefix} } |
= &{ ${ &_NS_uri2package ($nsuri).'::OPTION' }{n11n_prefix} } |
508 |
($self, &_NS_uri2phpackage ($nsuri), $prefix1.'-'.$prefix2); |
($self, &_NS_uri2package ($nsuri), $prefix1.'-'.$prefix2); |
509 |
$self->_ns_load_ph ($prefix); |
$self->_ns_load_ph ($prefix); |
510 |
$nsuri = $self->{ns}->{phname2uri}->{ $prefix }; |
$nsuri = $self->{ns}->{phname2uri}->{ $prefix }; |
511 |
unless ($nsuri) { |
unless ($nsuri) { |
512 |
$nsuri = $self->{option}->{ns_default_phuri}; |
$nsuri = $default_ns; |
513 |
$prefix |
$prefix |
514 |
= &{ ${ &_NS_uri2phpackage ($nsuri).'::OPTION' }{n11n_prefix} } |
= &{ ${ &_NS_uri2package ($nsuri).'::OPTION' }{n11n_prefix} } |
515 |
($self, &_NS_uri2phpackage ($nsuri), $one_prefix? $prefix2: $prefix1); |
($self, &_NS_uri2package ($nsuri), $one_prefix? $prefix2: $prefix1); |
516 |
$self->_ns_load_ph ($prefix); |
$self->_ns_load_ph ($prefix); |
517 |
$nsuri = $self->{ns}->{phname2uri}->{ $prefix }; |
$nsuri = $self->{ns}->{phname2uri}->{ $prefix }; |
518 |
if ($nsuri) { |
if ($nsuri) { |
519 |
$name = $prefix2 . '-' . $name unless $one_prefix; |
$name = $prefix2 . '-' . $name unless $one_prefix; |
520 |
} else { |
} else { |
521 |
$name = $original_prefix . $name; |
$name = $original_prefix . $name; |
522 |
$nsuri = $self->{option}->{ns_default_phuri}; |
$nsuri = $default_ns; |
523 |
} |
} |
524 |
} |
} |
525 |
} |
} |
526 |
$name |
$name |
527 |
= &{${&_NS_uri2phpackage ($nsuri).'::OPTION'}{n11n_name}} |
= &{ ${ &_NS_uri2package ($nsuri).'::OPTION' }{n11n_name} } |
528 |
($self, &_NS_uri2phpackage ($nsuri), $name); |
($self, &_NS_uri2package ($nsuri), $name); |
529 |
Carp::croak "$name: invalid field-name" |
Carp::croak "$name: invalid field-name" |
530 |
if $option->{field_name_validation} |
if $option->{field_name_validation} |
531 |
&& $name =~ /$REG{$option->{field_name_unsafe_rule}}/; |
&& $name =~ /$REG{ $option->{field_name_unsafe_rule} }/; |
532 |
$value = $self->_parse_value ($name => $value, ns => $nsuri) |
$value = $self->_parse_value ($name => $value, ns => $nsuri) |
533 |
if $$option{parse} || $$option{parse_all}; |
if $option->{parse} || $option->{parse_all}; |
534 |
$$option{parse} = 0; |
$option->{parse} = 0; |
535 |
(1, $name.':'.$nsuri => {name => $name, body => $value, ns => $nsuri}); |
(1, $name.':'.$nsuri => {name => $name, body => $value, ns => $nsuri}); |
536 |
} |
} |
537 |
*_add_hash_check = \&_value_to_arrayitem; |
*_add_hash_check = \&_value_to_arrayitem; |
642 |
my $self = shift; |
my $self = shift; |
643 |
my $s = shift; |
my $s = shift; |
644 |
$s =~ s/^$REG{WSP}+//; $s =~ s/$REG{WSP}+$//; |
$s =~ s/^$REG{WSP}+//; $s =~ s/$REG{WSP}+$//; |
645 |
$s = lc $s unless ${&_NS_uri2phpackage ($self->{option}->{ns_default_phuri}).'::OPTION'}{case_sensible}; |
$s = lc $s unless ${&_NS_uri2package ($self->{option}->{ns_default_phuri}).'::OPTION'}{case_sensible}; |
646 |
$s; |
$s; |
647 |
} |
} |
648 |
|
|
674 |
$option{format} = $params{-format} if $params{-format}; |
$option{format} = $params{-format} if $params{-format}; |
675 |
$self->_init_by_format ($option{format}, \%option); |
$self->_init_by_format ($option{format}, \%option); |
676 |
for (grep {/^-/} keys %params) {$option{substr ($_, 1)} = $params{$_}} |
for (grep {/^-/} keys %params) {$option{substr ($_, 1)} = $params{$_}} |
677 |
|
## Fill required fields |
678 |
|
my %exist; |
679 |
|
for ($self->field_name_list) {$exist{$_} = 1} |
680 |
|
&{ $option{hook_stringify_fill_fields} } ($self, \%exist, \%option); |
681 |
my @ret; |
my @ret; |
682 |
## RFC 2774 numerical field name prefix |
## RFC 2774 numerical field name prefix |
683 |
my %nprefix; |
my %nprefix; |
685 |
%nprefix = reverse %{ $self->{ns}->{number2uri} }; |
%nprefix = reverse %{ $self->{ns}->{number2uri} }; |
686 |
my $i = (sort { $a <=> $b } keys %{ $self->{ns}->{number2uri} })[-1] + 1; |
my $i = (sort { $a <=> $b } keys %{ $self->{ns}->{number2uri} })[-1] + 1; |
687 |
$i = 10 if $i < 10; |
$i = 10 if $i < 10; |
688 |
my $hprefix = ${ &_NS_uri2phpackage |
my $hprefix = ${ &_NS_uri2package |
689 |
($self->{ns}->{phname2uri}->{'x-http'}) |
($self->{ns}->{phname2uri}->{'x-http'}) |
690 |
.'::OPTION' } {namespace_phname_goodcase}; |
.'::OPTION' } {namespace_phname_goodcase}; |
691 |
for my $uri (keys %nprefix) { |
for my $uri (keys %nprefix) { |
709 |
return unless length $name; |
return unless length $name; |
710 |
return if $option{output_mail_from} && $name eq 'mail-from'; |
return if $option{output_mail_from} && $name eq 'mail-from'; |
711 |
$body = '' if !$option{output_bcc} && $name eq 'bcc'; |
$body = '' if !$option{output_bcc} && $name eq 'bcc'; |
712 |
my $nspackage = &_NS_uri2phpackage ($nsuri); |
my $nspackage = &_NS_uri2package ($nsuri); |
713 |
my $oname; ## Outputed field-name |
my $oname; ## Outputed field-name |
714 |
my $prefix = $nprefix{ $nsuri } |
my $prefix = $nprefix{ $nsuri } |
715 |
|| ${$nspackage.'::OPTION'} {namespace_phname_goodcase} |
|| ${$nspackage.'::OPTION'} {namespace_phname_goodcase} |
716 |
|| $self->{ns}->{uri2phname}->{ $nsuri }; |
|| $self->{ns}->{uri2phname}->{ $nsuri }; |
717 |
my $default_prefix = ${ &_NS_uri2phpackage ($option{ns_default_phuri}) |
my $default_prefix = ${ &_NS_uri2package ($option{ns_default_phuri}) |
718 |
.'::OPTION'} {namespace_phname_goodcase}; |
.'::OPTION'} {namespace_phname_goodcase}; |
719 |
$prefix = '' if $prefix eq $default_prefix; |
$prefix = '' if $prefix eq $default_prefix; |
720 |
$prefix =~ s/^\Q$default_prefix\E-//; |
$prefix =~ s/^\Q$default_prefix\E-//; |
|
#$prefix = undef if $nsuri eq $option{ns_default_phuri}; |
|
|
#if ($prefix && $prefix eq $nprefix{ $nsuri }) { ## RFC 2774 prefix |
|
|
# $prefix = ${ &_NS_uri2phpackage ($self->{ns}->{phname2uri}->{'x-http'}) |
|
|
# .'::OPTION' } {namespace_phname_goodcase} . '-' . $prefix |
|
|
# unless $option{ns_default_phuri} eq $self->{ns}->{phname2uri}->{'x-http'}; |
|
|
#} |
|
721 |
my $gc = ${$nspackage.'::OPTION'} {to_be_goodcase}; |
my $gc = ${$nspackage.'::OPTION'} {to_be_goodcase}; |
722 |
if (ref $gc) { $oname = &$gc ($self, $nspackage, $name, \%option) } |
if (ref $gc) { $oname = &$gc ($self, $nspackage, $name, \%option) } |
723 |
else { $oname = $name } |
else { $oname = $name } |
883 |
$NS_uri2phpackage{$_[0]} |
$NS_uri2phpackage{$_[0]} |
884 |
|| $NS_uri2phpackage{$Message::Header::Default::OPTION{namespace_uri}}; |
|| $NS_uri2phpackage{$Message::Header::Default::OPTION{namespace_uri}}; |
885 |
} |
} |
886 |
|
sub _NS_uri2package ($) { |
887 |
|
$NS_uri2package{$_[0]} |
888 |
|
|| $NS_uri2phpackage{$_[0]} |
889 |
|
|| $NS_uri2phpackage{$Message::Header::Default::OPTION{namespace_uri}}; |
890 |
|
} |
891 |
|
|
892 |
=head2 $self->clone () |
=head2 $self->clone () |
893 |
|
|