=head1 NAME CGI::Deurl.pm - a CGI parameter decoding package version 1.08 =head1 SYNOPSIS use CGI::Deurl as => 'q'; ... print "$q{ParamName}\n"; =head1 DESCRIPTION This is a little module made for CGI scripting. It decodes the parameters passed to the CGI. It does nothing more, so it's much smaller and loads more quickly than CGI.pm. Since version 0.04 it also exports the C, C and C functions so that you are able to decode not only the parameters your CGI got, but also an arbitrary string. =head2 Reading CGI query The module can take the arguments from several sources. CGI::Deurl tests the environmental variable 'REQUEST_METHOD' to find the arguments. =over 4 =item REQUEST_METHOD=POST CGI::Deurl slurps in it's STDIN and parses it, then it tests $ENV{QUERY_STRING} and if defined parses the data as well. =item REQUEST_METHOD=GET CGI::Deurl reads $ENV{QUERY_STRING} and parses the contents. =item REQUEST_METHOD not defined CGI::Deurl tests ARGV and parses all it's arguments. If it didn't find any args it reads it's STDIN line by line until EOF and parses all the lines. This is handy if you want to test a CGI script from command prompt. You may specify the parameters either on the command line or enter them after the script starts to run. =back If you C the module doesn't look for the parameters and just exports the functions. This is handy if you use CGI::Deurl.pm in a script that is not a CGI. =head2 Using query variables The data are stored in a hash C<%CGI::Deurl::query>, but CGI::Deurl provides two ways to specify otherwise. They may be stored either in a hash you specify or be exported into a package. =head2 "use" statement options =over 4 =item lc If you use the option 'lc' all names of the parameters are converted to lowercase. This way you may get case insensitive parameters. =item uc This option is similar to 'lc'. It converts the names to uppercase. =item as variable If you use CGI::Deurl qw(as variablename); CGI::Deurl uses variable %variablename from package main to store the data (%CGI::Deurl::query works as well, this is just a nickname). =item use CGI::Deurl export; All the query variables are exported into package CGI::Deurl. That means that if the query was "?x=5&y=24", you can use $CGI::Deurl::x + $CGI::Deurl::y. Again %CGI::Deurl::query works as usual. =item use CGI::Deurl export Package; All the query values are exported into Package. That means that if the query was "?x=5&y=24", you can use $Package::x + $Package::y. Again %CGI::Deurl::query works as usual. =item use CGI::Deurl NOTCGI; Do not read any query. This option should be first or directly after the &=.... =item use CGI::Deurl '&=.' You may change the character used to separate the parameters. Use any character or string you want. This must be the first option if present! The parameter separator is stored in variable $CGI::Deurl::ParamSeparator. You may change it any time you want. =item use CGI::Deurl 'JOIN' => ';'; =item use CGI::Deurl 'JOIN' , 'file' => ';', '-all' => ','; This option will cause a call to joinquery(), all folowing arguments will be passed to this function, so this switch has to be the last one! =back =head2 Parsing query If the argument in the query in in the form "name=value". $CGI::Deurl::query{name} is set to value. If it is just "value" (say myscript.pl?one&two), $CGI::Deurl::query{0}='one' and $CGI::Deurl::query{1}='two'. These kinds of parameters can be intermixed. If there is more than one occurence of a variable, $CGI::Deurl::query{name} contains a refference to an array containing all the values. Ex. ?x=one&y=two&x=three => $CGI::Deurl::query{x}=['one','three']; $CGI::Deurl::query{y}='two'; !!! If you 'export' such a variable it's not exported as a refference but as a real array!!! That is if you use CGI::Deurl qw(export CGI::Deurl) you will get : @CGI::Deurl::x = ('one','three'); $CGI::Deurl::y = 'two'; ! All changes made to $CGI::Deurl::variable are visible in $CGI::Deurl::query{variable} and vice versa. =head2 Functions =over 4 =item deurl =item deurl $string, \%hash Decodes the string as if it was an ordinary CGI query. The %hash then contains all CGI parameters specified there. Ex. deurl('a=5&b=13',\%query); leads to : $query{a} = 5; $query{b} = 13; =item deurlstr =item $decodedstring = deurlstr $string Decodes the string as if it was a CGI parameter value. That is ist translates all '+' to ' ' and all %xx to the corresponding character. It doesn't care about '&' nor '='. Ex. $result = deurlstr 'How+are+you%3F'; leads to: $result = 'How are you?' !!! but notice that !!! $result = deurlstr('a=5&b=13%25'); gives: $result = 'a=5&b=13%' !!!!!! =item deurlarr =item @result = deurlarr $string; Decodes the string as a sequence of unnamed CGI parameters, that is it supposes that the $string looks somehow like this : 'param1¶m2&par%61m3'. It doesn't care about '=' Ex. @result = deurlarr 'How&are+you%3f'; leads to @result = ( 'How', 'are you?'); @result = deurlstr('a=5&b=13%25'); gives: @result = ( 'a=5', 'b=13%'); which may but may not be what you want. =item CGI::Deurl::load Instructs CGI::Deurl to load the CGI data from QUERY_STRING, @ARGV or . The parameters are the same as for the C Usefull only if you C, but later on you find out you want the CGI parameters. =item joinquery %query, $delimiter =item joinquery %query, $key => $delimiter [, ...] If the query contains several values for a singe variable, these values are stored as an array in the hash. This function joins them using the delimiter you specify. You may either join all the keys using the same delimiter, or use different delimiters for each key. You may even leave some values intact. Ex.: joinquery %query, $delimiter it will join all multivalues it finds using the $delimiter. joinquery %query, 'key' => $delimiter it will join only the multivalue for 'key'. All other values will remain the same. joinquery %query, 'key' => ';', '-all' => ' ' it will join the multivalue for 'key' by semicolons. All other values will be joined using spaces. You may call this function from the "use" statement. =item $CGI::Deurl::offline If the script was called from the command line instead of as a CGI, this variable contains 1. Otherwise it's undefined. =back =cut package CGI::Deurl; require Exporter; @EXPORT=qw(deurl deurlstr deurlarr joinquery); use vars qw'$VERSION $query $string'; $VERSION='1.08'; $ParamSeparator = '&' unless $ParamSeparator; sub joinquery (\%@); sub export { my $pkg=$_[0]; $pkg.='::' if $pkg; my $key; foreach $key (keys %CGI::Deurl::query) { if (ref ${CGI::Deurl::query{$key}}) { *{$pkg.$key} = ${CGI::Deurl::query{$key}}; } else { *{$pkg.$key} = \${CGI::Deurl::query{$key}}; } } } sub as { my $name = $_[0]; package main; *{$name} = *CGI::Deurl::query; package CGI::Deurl; } sub import { my $caller_pack = caller; my( $pkg )= shift; Exporter::export( $pkg, $caller_pack, qw(deurl deurlstr deurlarr joinquery)); if (defined $_[0]) { if ($_[0] =~ /^&=(.+)$/) {shift; $ParamSeparator="\Q$1"} return if ($_[0] eq 'NOTCGI' or $_[0] eq 'NOCGI'); if ($_[0] =~ /^&=(.+)$/) {shift; $ParamSeparator="\Q$1"} } &CGI::Deurl::load(@_); } sub load { my $data; if (defined $ENV{REQUEST_METHOD} and $ENV{REQUEST_METHOD} eq "POST") { read STDIN , $data , $ENV{CONTENT_LENGTH} ,0; if ($ENV{QUERY_STRING}) { $data .= $ParamSeparator . $ENV{QUERY_STRING}; } } elsif (defined $ENV{REQUEST_METHOD} and $ENV{REQUEST_METHOD} eq "GET") { $data=$ENV{QUERY_STRING}; } elsif (defined $ENV{REQUEST_METHOD}) { print "Status: 405 Method Not Allowed\r\n\r\n"; exit; } elsif ($#ARGV >= 0) { $CGI::Deurl::offline = 1; $data=join $ParamSeparator, @ARGV; } else { print STDERR "\t\n"; $CGI::Deurl::offline = 1; my @lines=; chomp @lines; $data=join $ParamSeparator, @lines; } return unless (defined $data and $data ne ''); deurl($data,\%CGI::Deurl::query); my $i; for($i=0;$i<=$#_;$i++) { if (lc $_[$i] eq 'lc') { my (%hash,$key,$value); while (($key,$value) = each %CGI::Deurl::query) { $hash{lc $key} = $value; } %CGI::Deurl::query = %hash; } elsif (lc $_[$i] eq 'uc') { my (%hash,$key,$value); while (($key,$value) = each %CGI::Deurl::query) { $hash{uc $key} = $value; } %CGI::Deurl::query = %hash; } elsif (lc $_[$i] eq 'as') { $i++; CGI::Deurl::as $_[$i]; } elsif (lc $_[$i] eq 'export') { CGI::Deurl::export($_[++$i]); } elsif ($_[$i] =~ /^join$/i) { if (ref $_[++$i] eq 'ARRAY') { joinquery(%CGI::Deurl::query, @{$_[$i]}); } else { joinquery(%CGI::Deurl::query, $_[$i]); } } else { die "Unknown export directive $_[$i] in CGI::Deurl.pm!\n" } } } sub deurl ($$) { my ($data,$hash)=@_; die "deurl: ussage deurl($string, \%hash)\n" unless ($data and ref $hash eq 'HASH'); $data=~s/\?$//; my $i=0; my @items = grep {!/^$/} (split /$ParamSeparator/o, $data); my $thing; foreach $thing (@items) { my @res = $thing=~/^(.*?)=(.*)$/; my ($name,$value,@value); if ($#res<=0) { $name = $i++; $value = $thing; } else { ($name,$value) = @res; } next unless $value ne ''; $name=~tr/+/ /; $name =~ s/%(\w\w)/chr(hex $1)/ge; $value=~tr/+/ /; $value =~ s/%(\w\w)/chr(hex $1)/ge; if ($hash->{$name}) { if (ref $hash->{$name}) { push @{$hash->{$name}},$value; } else { $hash->{$name} = [ $hash->{$name}, $value]; } } else { $hash->{$name} = $value; } } 1; } sub deurlstr ($) { my $value=$_[0]; $value=~tr/+/ /; $value =~ s/%(\w\w)/chr(hex $1)/ge; $value; } sub deurlarr ($) { my @value = split /$ParamSeparator/o, $_[0]; foreach (@value) { tr/+/ /; s/%(\w\w)/chr(hex $1)/ge; } return @value; } sub joinquery (\%@) { my $query = shift; #{ #local $"=", "; #print "(@_)\n"; #} if (@_ == 1) { @_ = ('-all',$_[0]); } my %joins = @_; while (($key,$value) = each %$query) { if (ref $value eq 'ARRAY') { my $join; if ($join = $joins{$key} or $join = $joins{'-all'}) { $query->{$key} = join $join, @$value; } } } } 1; __END__ =head2 AUTHOR Jenda@Krynicky.cz =head2 COPYRIGHT Copyright (c) 1997 Jan Krynicky . All rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. =cut