X-Git-Url: http://git.freeside.biz/gitweb/?a=blobdiff_plain;f=rt%2Flib%2FRT%2FInterface%2FREST.pm;h=ae1d6ad1e48ee3af255b47f10f9f4078e1b6e3fe;hb=fc6209f398899f0211cfcedeb81a3cd65e04a941;hp=4eeb468c438c39d79e5ca3c96a2c54e160dc101a;hpb=9c68254528b6f2c7d8c1921b452fa56064783782;p=freeside.git diff --git a/rt/lib/RT/Interface/REST.pm b/rt/lib/RT/Interface/REST.pm index 4eeb468c4..ae1d6ad1e 100644 --- a/rt/lib/RT/Interface/REST.pm +++ b/rt/lib/RT/Interface/REST.pm @@ -1,38 +1,40 @@ # BEGIN BPS TAGGED BLOCK {{{ -# +# # COPYRIGHT: -# -# This software is Copyright (c) 1996-2005 Best Practical Solutions, LLC -# -# +# +# This software is Copyright (c) 1996-2011 Best Practical Solutions, LLC +# +# # (Except where explicitly superseded by other copyright notices) -# -# +# +# # LICENSE: -# +# # This work is made available to you under the terms of Version 2 of # the GNU General Public License. A copy of that license should have # been provided with this software, but in any event can be snarfed # from www.gnu.org. -# +# # This work is distributed in the hope that it will be useful, but # WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # General Public License for more details. -# +# # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software -# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. -# -# +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA +# 02110-1301 or visit their web page on the internet at +# http://www.gnu.org/licenses/old-licenses/gpl-2.0.html. +# +# # CONTRIBUTION SUBMISSION POLICY: -# +# # (The following paragraph is not intended to limit the rights granted # to you to modify and distribute this software under the terms of # the GNU General Public License and is only of importance to you if # you choose to contribute your changes and enhancements to the # community by submitting them to Best Practical Solutions, LLC.) -# +# # By intentionally submitting any modifications, corrections or # derivatives to this work, or any other work intended for use with # Request Tracker, to Best Practical Solutions, LLC, you confirm that @@ -41,8 +43,9 @@ # royalty-free, perpetual, license to use, copy, create derivative # works based on those contributions, and sublicense and distribute # those contributions and any derivatives thereof. -# +# # END BPS TAGGED BLOCK }}} + # lib/RT/Interface/REST.pm # @@ -51,28 +54,63 @@ use strict; use RT; BEGIN { - use Exporter (); - use vars qw($VERSION @ISA @EXPORT); + use base 'Exporter'; + use vars qw($VERSION @EXPORT); - $VERSION = do { my @r = (q$Revision: 1.1.1.5 $ =~ /\d+/g); sprintf "%d."."%02d"x$#r, @r }; + $VERSION = do { my @r = (q$Revision: 1.1.1.9 $ =~ /\d+/g); sprintf "%d."."%02d"x$#r, @r }; - @ISA = qw(Exporter); @EXPORT = qw(expand_list form_parse form_compose vpush vsplit); } -my $field = '[a-zA-Z][a-zA-Z0-9_-]*'; +sub custom_field_spec { + my $self = shift; + my $capture = shift; + my $CF_char = '[\sa-z0-9_ :()/-]'; + my $CF_name = $CF_char . '+'; + $CF_name = '(' . $CF_name . ')' if $capture; + + my $new_style = 'CF\.\{'.$CF_name.'\}'; + my $old_style = 'C(?:ustom)?F(?:ield)?-'.$CF_name; + + return '(?i:' . join('|', $new_style, $old_style) . ')'; +} + +sub field_spec { + my $self = shift; + my $capture = shift; + + my $field = '[a-z][a-z0-9_-]*'; + $field = '(' . $field . ')' if $capture; + + my $custom_field = __PACKAGE__->custom_field_spec($capture); + + return '(?i:' . join('|', $field, $custom_field) . ')'; +} + +# WARN: this code is duplicated in bin/rt.in, +# change both functions at once sub expand_list { my ($list) = @_; - my ($elt, @elts, %elts); - foreach $elt (split /,/, $list) { - if ($elt =~ /^(\d+)-(\d+)$/) { push @elts, ($1..$2) } - else { push @elts, $elt } + my @elts; + foreach (split /\s*,\s*/, $list) { + push @elts, /^(\d+)-(\d+)$/? ($1..$2): $_; } - @elts{@elts}=(); - return sort {$a<=>$b} keys %elts; + return map $_->[0], # schwartzian transform + sort { + defined $a->[1] && defined $b->[1]? + # both numbers + $a->[1] <=> $b->[1] + :!defined $a->[1] && !defined $b->[1]? + # both letters + $a->[2] cmp $b->[2] + # mix, number must be first + :defined $a->[1]? -1: 1 + } + map [ $_, (defined( /^(\d+)$/ )? $1: undef), lc($_) ], + @elts; } # Returns a reference to an array of parsed forms. @@ -81,6 +119,7 @@ sub form_parse { my @forms = (); my @lines = split /\n/, $_[0]; my ($c, $o, $k, $e) = ("", [], {}, ""); + my $field = __PACKAGE__->field_spec; LINE: while (@lines) { @@ -108,10 +147,11 @@ sub form_parse { } $c .= "\n"; } - elsif ($state <= 1 && $line =~ /^($field):(?:\s+(.*))?$/) { + elsif ($state <= 1 && $line =~ /^($field):(?:\s+(.*))?$/i) { # Read a field: value specification. my $f = $1; - my @v = ($2 || ()); + my @v = ($2); + $v[0] = '' unless defined $v[0]; # Read continuation lines, if any. while (@lines && ($lines[0] eq '' || $lines[0] =~ /^\s+/)) { @@ -126,12 +166,14 @@ sub form_parse { } s/^$ws// foreach @v; + shift @v while (@v && $v[0] eq ''); + push(@$o, $f) unless exists $k->{$f}; vpush($k, $f, join("\n", @v)); $state = 1; } - elsif ($line !~ /^#/) { + elsif ($line =~ /^#/) { # We've found a syntax error, so we'll reconstruct the # form parsed thus far, and add an error marker. (>>) $state = -1; @@ -184,7 +226,8 @@ sub form_compose { $sp = " "x4 if length($sp) > 16; foreach $v (@values) { - if ($v =~ /\n/) { + $v = '' unless defined $v; + if ( $v =~ /\n/) { $v =~ s/^/$sp/gm; $v =~ s/^$sp//; @@ -252,9 +295,9 @@ sub vsplit { my ($val) = @_; my ($line, $word, @words); - foreach $line (map {split /\n/} (ref $val eq 'ARRAY') ? @$val : $val) + foreach $line (map {split /\n/} (ref $val eq 'ARRAY') ? @$val : ($val||'')) { - # XXX: This should become a real parser, à la Text::ParseWords. + # XXX: This should become a real parser, ? la Text::ParseWords. $line =~ s/^\s+//; $line =~ s/\s+$//; push @words, split /\s*,\s*/, $line; @@ -263,6 +306,16 @@ sub vsplit { return \@words; } +eval "require RT::Interface::REST_Vendor"; +if ($@ && $@ !~ qr{^Can't locate RT/Interface/REST_Vendor.pm}) { + die $@; +}; + +eval "require RT::Interface::REST_Local"; +if ($@ && $@ !~ qr{^Can't locate RT/Interface/REST_Local.pm}) { + die $@; +}; + 1; =head1 NAME