/usr/share/perl5/REST/Application.pm is in librest-application-perl 0.992-3.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 | # vi:ai:sm:et:sw=4:ts=4:tw=0
#
# REST::Application - A framework for building RESTful web-applications.
#
# Copyright 2005 Matthew O'Connor <matthew@canonical.org>
package REST::Application;
use strict;
use warnings;
use Carp;
use Tie::IxHash;
use UNIVERSAL;
use CGI;
our $VERSION = '0.992';
####################
# Class Methods
####################
sub new {
my ($proto, %args) = @_;
my $class = ref($proto) ? ref($proto) : $proto;
my $self = bless({ __defaultQuery => CGI->new() }, $class);
$self->setup(%args);
return $self;
}
##################################
# Instance Methods - Object State
##################################
sub query {
my $self = shift;
# Set default value if this method hasn't been called yet.
if (not exists $self->{__query}) {
$self->{__query} = $self->defaultQueryObject();
}
# Set the field if we got any arguments.
$self->{__query} = shift if @_;
return $self->{__query};
}
sub defaultQueryObject {
my $self = shift;
# Set the field if we got any arguments.
if (@_) {
$self->{__defaultQuery} = shift;
}
return $self->{__defaultQuery};
}
sub resourceHooks {
my $self = shift;
# Set default value if this method hasn't been called yet.
if (not exists $self->{__resourceHooks}) {
my %hooks;
tie(%hooks, "Tie::IxHash"); # For keeping hash key order preserved.
$self->{__resourceHooks} = \%hooks;
}
# If we got arguments then they should be an even sized list, otherwise a
# hash reference.
if (@_ and @_%2 == 0) {
%{ $self->{__resourceHooks} } = @_;
} elsif (@_ == 1) {
my $value = shift;
if (ref($value) ne 'HASH') {
croak "Expected hash reference or even-sized list.";
}
%{ $self->{__resourceHooks} } = %$value;
}
return $self->{__resourceHooks};
}
sub extraHandlerArgs {
my $self = shift;
# Set default value for method if it hasn't been called yet.
if (not exists $self->{__extraHandlerArgs}) {
$self->{__extraHandlerArgs} = [];
}
# If got arguments then process them. We expect either a single array
# reference or a list
if (@_) {
if (@_ == 1 and ref($_[0]) eq 'ARRAY') {
$self->{__extraHandlerArgs} = shift;
} else {
$self->{__extraHandlerArgs} = [ @_ ];
}
}
return @{ $self->{__extraHandlerArgs} };
}
##################################
# Instance Methods - Proxies
##################################
sub getPathInfo {
my $self = shift;
return $self->query->path_info();
}
sub getRealRequestMethod {
my $self = shift;
return uc( $self->query->request_method() || "" );
}
sub getRequestMethod {
my $self = shift;
my $real_method = $self->getRealRequestMethod();
my $tunnel_method = uc(
$self->query->http('X-HTTP-Method')
|| $self->query->url_param('http_method')
|| $self->query->param('http_method')
|| $real_method
);
# POST can tunnel any method.
return $tunnel_method if $real_method eq 'POST';
# GET can only tunnel GET/HEAD
if ( $real_method eq 'GET' and $tunnel_method =~ /^(GET|HEAD)$/ ) {
return $tunnel_method;
}
return $real_method;
}
#############################
# Instance Methods - Public
#############################
sub loadResource {
my ($self, $path, @extraArgs) = @_;
$path ||= $self->getMatchText();
my $handler = sub { $self->defaultResourceHandler(@_) };
my $matches = [];
# Loop through the keys of the hash returned by resourceHooks(). Each of
# the keys is a regular expression, see if the current path info matches
# that regular expression. Save the parent matches for passing into the
# handler.
for my $pathRegex (keys %{ $self->resourceHooks() }) {
if ($self->checkMatch($path, $pathRegex)) {
$handler = $self->_getHandlerFromHook($pathRegex);
last;
}
}
return $self->callHandler($handler, @extraArgs);
}
sub getHandlerArgs {
my ($self, @extraArgs) = @_;
my @args = ($self,
$self->_getLastRegexMatches(),
$self->extraHandlerArgs(),
@extraArgs);
# Don't make $self the first argument if the handler is a method on $self,
# because in that case it'd be redundant. Also see _getHandlerFromHook().
shift @args if $self->{__handlerIsOurMethod};
return @args;
}
sub callHandler {
my ($self, $handler, @extraArgs) = @_;
my @args = $self->getHandlerArgs(@extraArgs);
# Call the handler, carp if something goes wrong.
my $result;
eval {
$self->preHandler(\@args); # no-op by default.
$result = $handler->(@args);
$self->postHandler(\$result, \@args); # no-op by default.
};
carp "Handler failed: $@\n" if $@;
# Convert the result to a scalar result if it isn't already.
my $ref = (ref($result) eq 'scalar') ? $result : \$result;
return $ref;
}
sub getMatchText {
my $self = shift;
return $self->getPathInfo();
}
sub checkMatch {
my ($self, $a, $b) = @_;
my $match = 0;
if ($a =~ /$b/) {
$self->_setLastRegexMatches();
$self->{__last_match_path} = $a;
$self->{__last_match_pattern} = $b;
$match = 1;
}
return $match;
}
sub getLastMatchPath {
my $self = shift;
return $self->{__last_match_path};
}
sub getLastMatchPattern {
my $self = shift;
return $self->{__last_match_pattern};
}
sub run {
my $self = shift;
# Get resource.
$self->preRun(); # A no-op by default.
my $repr = $self->loadResource(@_);
$self->postRun($repr); # A no-op by default.
# Get the headers and then add the representation to to the output stream.
my $output = $self->getHeaders();
$self->addRepresentation($repr, \$output);
# Send the output unless we're told not to by the environment.
print $output if not $ENV{REST_APP_RETURN_ONLY};
return $output;
}
sub getHeaders {
my $self = shift;
my $type = $self->headerType() || "";
my $header = "";
if ($type eq 'header') {
$header = $self->query->header($self->header());
} elsif ($type eq 'redirect') {
$header = $self->query->redirect($self->header());
} elsif ($type ne 'none') {
croak "Unexpected header type: \"$type\".";
}
return $header;
}
sub addRepresentation {
my ($self, $repr, $outputRef) = @_;
# Make sure $outputRef is a scalar ref and the scalar it references is
# defined.
return if ref($outputRef) ne 'SCALAR';
return if not defined $$outputRef;
# If we're given a scalar reference then dereference it first, otherwise
# just treat what we got as though it's a string.
if (ref($repr) eq 'SCALAR') {
$$outputRef .= $$repr if defined $$repr;
} else {
$$outputRef .= $repr if defined $repr;
}
}
sub headerType {
my $self = shift;
# Set the default value if this method has not been called yet.
if (not exists $self->{__headerType}) {
$self->{__headerType} = "header";
}
# If an argument was passed in then use them to set the header type.
if (@_) {
my $type = lc(shift || "");
if ($type =~ /^(redirect|header|none)$/) {
$self->{__headerType} = $type;
} else {
croak "Invalid header type specified: \"$type\"";
}
}
return $self->{__headerType};
}
sub header {
my $self = shift;
# Set the default value if this method has not been called yet.
if (not exists $self->{__header}) {
$self->{__header} = {};
}
# If arguments were passed in then use them to set the header type.
# Arguments can be passed in as a hash-ref or as an even sized list.
if (@_) {
if (@_%2 == 0) { # even-sized list, must be hash
%{ $self->{__header} } = ( %{$self->{__header}}, @_ );
} elsif (ref($_[0]) eq 'HASH') { # First item must be a hash reference
%{ $self->{__header} } = ( %{$self->{__header}}, %{$_[0]} );
} else {
croak "Expected even-sized list or hash reference.";
}
}
return %{$self->{__header}};
}
sub resetHeader {
my $self = shift;
my %old = $self->header();
$self->headerType('header');
$self->{__header} = {};
return %old;
}
sub setRedirect {
my ($self, $url) = @_;
$self->headerType('redirect');
$self->header(-url => $url || "");
}
##############################################
# Instance Methods - Intended for Overloading
##############################################
sub setup { return }
sub preRun { return }
sub postRun{ return }
sub preHandler { return }
sub postHandler { return }
sub defaultResourceHandler { return }
#############################
# Instance Methods - Private
#############################
# CodeRef _getHandlerFromHook(String $pathRegex)
#
# This method retrieves a code reference which will yield the resource of the
# given $pathRegex, where $pathRegex is a key into the resource hooks hash (it
# isn't used as a regex in this method, just a hash key).
sub _getHandlerFromHook {
my ($self, $pathRegex) = @_;
my $ref = $self->resourceHooks()->{$pathRegex};
my $refType = ref($ref);
my $handler = sub { $self->defaultResourceHandler(@_) };
my $method = $self->getRequestMethod() || "getResource";
# If we get a hash, then use the request method to narrow down the choice.
# We do this first because we allow the same range of handler types for a
# particular HTTP method that we do for a more generic handler.
if ($refType eq 'HASH') {
%$ref = map { uc($_) => $ref->{$_} } keys %$ref; # Uppercase the keys
my $http_method = $self->getRequestMethod();
if (exists $ref->{$http_method}) {
$ref = $ref->{$http_method}
} elsif (exists $ref->{'*'}) {
$ref = $ref->{'*'};
} else {
return $handler; # Just bail now with the default handler.
}
$refType = ref($ref);
}
# If we still have a hash then assume we're doing Content Negotation
if ($refType eq 'HASH') {
my $type = $self->bestContentType(keys %$ref);
$ref = $ref->{$type};
$refType = ref($ref);
}
# Based on the the reference's type construct the handler.
if ($refType eq "CODE") {
# A code reference
$handler = $ref;
} elsif ($refType eq "ARRAY") {
# Array reference which holds a $object and "method name" pair.
my ($thing, $smethod) = @$ref;
$smethod ||= $method;
if (ref $thing) {
$handler = $self->makeHandlerFromRef($thing, $smethod);
} else {
$handler = $self->makeHandlerFromClass($thing, $smethod);
}
} elsif ($refType) {
# Object with GET, PUT, etc, or getResource method.
$handler = $self->makeHandlerFromRef($ref, $method);
} elsif ($ref) {
# A bare string signifying a method name
$handler = sub { $self->$ref(@_) };
$self->{__handlerIsOurMethod} = 1; # See callHandler().
}
return $handler;
}
sub makeHandlerFromRef {
my ($self, $ref, $method) = @_;
return sub { $ref->$method(@_) };
}
sub makeHandlerFromClass {
my ($self, $class, $method) = @_;
return sub { $class->$method(@_) };
}
sub bestContentType {
my ($self, @types) = @_;
return ($self->simpleContentNegotiation(@types))[0] || '*/*';
}
# We don't support the full range of content negtiation because a) it's
# overkill and b) it makes it hard to specify the hooks cleanly, also see (a).
sub simpleContentNegotiation {
my ($self, @types) = @_;
my @accept_types = $self->getContentPrefs();
my $score = sub { $self->scoreType(shift, @accept_types) };
return sort {$score->($b) <=> $score->($a)} @types;
}
# The pattern matching stuff was stolen from CGI.pm
sub scoreType {
my ($self, $type, @accept_types) = @_;
my $score = scalar(@accept_types);
for my $accept_type (@accept_types) {
return $score if $type eq $accept_type;
my $pat;
($pat = $accept_type) =~ s/([^\w*])/\\$1/g; # escape meta characters
$pat =~ s/\*/.*/g; # turn it into a pattern
return $score if $type =~ /$pat/;
$score--;
}
return 0;
}
# Basic idea stolen from CGI.pm. Its semantics made it hard to pull out the
# information I wanted without a lot of trickery, so I improved upon the
# original. Same with libwww's HTTP::Negotiate algorithim, it's also hard to
# make go with what we want.
sub getContentPrefs {
my $self = shift;
my $default_weight = 1;
my @prefs;
# Parse the Accept header, and save type name, score, and position.
my @accept_types = split /,/, $self->getAcceptHeader();
my $order = 0;
for my $accept_type (@accept_types) {
my ($weight) = ($accept_type =~ /q=(\d\.\d+|\d+)/);
my ($name) = ($accept_type =~ m#(\S+/[^;]+)#);
next unless $name;
push @prefs, { name => $name, order => $order++};
if (defined $weight) {
$prefs[-1]->{score} = $weight;
} else {
$prefs[-1]->{score} = $default_weight;
$default_weight -= 0.001;
}
}
# Sort the types by score, subscore by order, and pull out just the name
@prefs = map {$_->{name}} sort {$b->{score} <=> $a->{score} ||
$a->{order} <=> $b->{order}} @prefs;
return @prefs, '*/*'; # Allows allow for */*
}
sub getAcceptHeader {
my $self = shift;
return $self->query->http('accept') || "";
}
# List _getLastRegexMatches(void)
#
# Returns a list of all the paren matches in the last regular expression who's
# matches were saved with _saveLastRegexMatches().
sub _getLastRegexMatches {
my $self = shift;
my $matches = $self->{__lastRegexMatches} || [];
return @$matches;
}
# ArrayRef _setLastRegexMatches(void)
#
# Grabs the values of $1, $2, etc. as set by the last regular expression to run
# in the current dyanamic scope. This of course exploits that $1, $2, etc. and
# @+ are dynamically scoped. A reference to an array is returned where the
# array values are $1, $2, $3, etc. _getLastRegexMatches() can also be used to
# retrieve the values saved by this method.
sub _setLastRegexMatches {
my $self = shift;
no strict 'refs'; # For the $$_ symbolic reference below.
my @matches = map $$_, (1 .. scalar(@+)-1); # See "perlvar" for @+.
$self->{__lastRegexMatches} = \@matches;
}
1;
__END__
=pod
=head1 NAME
L<REST::Application> - A framework for building RESTful web-applications.
=head1 SYNOPSIS
# MyRESTApp REST::Application instance / mod_perl handler
package MyRESTApp;
use Apache;
use Apache::Constants qw(:common);
sub handler {
__PACKAGE__->new(request => $r)->run();
return OK;
}
sub getMatchText { return Apache->uri }
sub setup {
my $self = shift;
$self->resourceHooks(
qr{/rest/parts/(\d+)} => 'get_part',
# ... other handlers here ...
);
}
sub get_part {
my ($self, $part_num) = @_;
# Business logic to retrieve part num
}
# Apache conf
<Location /rest>
perl-script .cgi
PerlHandler MyRESTApp
</Location>
=head1 DESCRIPTION
This module acts as a base class for applications which implement a RESTful
interface. When an HTTP request is received some dispatching logic in
L<L<REST::Application>> is invoked, calling different handlers based on what the
kind of HTTP request it was (i.e. GET, PUT, etc) and what resource it was
trying to access. This module won't ensure that your API is RESTful but
hopefully it will aid in developing a REST API.
=head1 OVERVIEW
The following list describes the basic way this module is intended to be used.
It does not capture everything the module can do.
=over 8
=item 1. Subclass
Subclass L<REST::Application>, i.e. C<use base 'REST::Application'>.
=item 2. Overload C<setup()>
Overload the C<setup()> method and set up some resource hooks with the
C<resourceHooks()> method. Hooks are mappings of the form:
REGEX => handler
where handler is either a method name, a code reference, an object which
supports a method with the same name as the HTTP method (or C<getResource> if
no such method), or a reference to an array of the form: C<[$objectRef,
"methodName"]> (C<$objectRef> can be a class name instead).
The regular expressions are applied, by default, to the path info of the HTTP
request. Anything captured by parens in the regex will be passed into the
handler as arguments.
For example:
qr{/parts/(\d+)$} => "getPartByNumber",
The above hook will call a method named C<getPartByNumber> on the current
object (i.e. $self, an instance of L<REST::Application>) if the path info of
the requested URI matches the above regular expression. The first argument to
the method will be the part number, since that's the first element captured in
the regular expression.
=item 3. Write code.
Write the code for the handler specified above. So here we'd define the
C<getPartByNumber> method.
=item 4. Create a handler/loader.
Create an Apache handler, for example:
use MyRESTApp;
sub handler {
my $r = shift;
my $app = MyRESTApp->new(request => $r);
$app->run();
}
or a small CGI script with the following code:
#!/usr/bin/perl
use MyRESTApp;
MyRESTApp->new()->run();
In the second case, for a CGI script, you'll probably need to do something
special to get Apache to load up your script unless you give it a .cgi
extension. It would be unRESTful to allow your script to have a .cgi
extension, so you should go the extra mile and configure Apache to run your
script without it. For example, it'd be bad to have your users go to:
http://www.foo.tld/parts.cgi/12345.html
=item 5. Call the C<run()> method.
When the C<run()> method is called the path info is extracted from the HTTP
request. The regexes specified in step 2 are processed, in order, and if one
matches then the handler is called. If the regex had paren. matching then the
matched elements are passed into the handler. A handler is also passed a copy
of the L<REST::Application> object instance (except for the case when the
handler is a method on the L<REST::Application> object, in that case it'd be
redundant). So, when writing a subroutine handler you'd do:
sub rest_handler {
my ($rest, @capturedArgs) = @_;
...
}
=item 6. Return a representation of the resource.
The handler is processed and should return a string or a scalar reference to a
string. Optionally the handler should set any header information via the
C<header()> method on instance object pased in.
=back
=head1 CALLING ORDER
The L<REST::Application> base class provides a good number of methods, each of
which can be overloaded. By default you only need to overload the C<setup()>
method but you may wish to overload others. To help with this the following
outline is the calling order of the various methods in the base class. You can
find detailed descriptions of each method in the METHODS section of this
document.
If a method is followed by the string NOOP then that means it does nothing by
default and it exists only to be overloaded.
new()
setup() - NOOP
run()
preRun() - NOOP
loadResource()
getMatchText()
getPathInfo()
query()
defaultQueryObject()
defaultResourceHandler() - NOOP
resourceHooks()
checkMatch()
_setLastRegexMatches()
_getHandlerFromHook()
resourceHooks()
defaultResourceHandler() - NOOP
getRequestMethod()
query()
defaultQueryObject()
bestContentType()
simpleContentNegotiation
getContentPrefs
getAcceptHeader
scoreType()
callHandler()
getHandlerArgs
_getLastRegexMatches()
extraHandlerArgs()
preHandler() - NOOP
... your handler called here ...
postHandler() - NOOP
postRun() - NOOP
getHeaders()
headerType()
query()
defaultQueryObject()
header()
addRepresentation()
The only methods not called as part of the new() or run() methods are the
helper methods C<resetHeader()> and C<setRedirect()>, both of which call the
C<header()> and C<headerType()> methods.
For example, if you wanted to have your code branch on the entire URI of the
HTTP request rather than just the path info you'd merely overload
C<getMatchText()> to return the URI rather than the path info.
=head1 METHODS
=head2 new(%args)
This method creates a new L<REST::Application> object and returns it. The
arguments passed in via C<%args>, if any, are passed untouched to the
C<setup()> method.
=head2 query([$newCGI])
This accessor/mutator retrieves the current CGI query object or sets it if one
is passed in.
=head2 defaultQueryObject([$newCGI])
This method retrieves/sets the default query object. This method is called if
C<query()> is called for the first time and no query object has been set yet.
=head2 resourceHooks([%hash])
This method is used to set the resource hooks. A L<REST::Application> hook is
a regex to handler mapping. The hooks are passed in as a hash (or a reference
to one) and the keys are treated as regular expressions while the values are
treated as handlers should B<PATH_INFO> match the regex that maps to that
handler.
Handlers can be code references, methods on the current object, methods on
other objects, or class methods. Also, handlers can be differ based on what
the B<REQUEST_METHOD> was (e.g. GET, PUT, POST, DELETE, etc).
The handler's types are as follows:
=over 8
=item string
The handler is considered to be a method on the current L<REST::Application>
instance.
=item code ref
The code ref is considered to be the handler.
=item object ref
The object is considered to have a method the same name as the HTTP method.
That is, if the object is being called because of GET then C<GET()> is called,
if it is called because of a C<DELETE> then C<DELETE()> is called.
C<getResource()> method will be used if C<getRequestMethod()> returns
false.
=item array ref
The array is expected to be two elements long, the first element is a class
name or object instance. The 2nd element is a method name on that
class/instance. IF the 2nd element is ommitted then the method name is assumed
to be the same as the B<REQUEST_METHOD>, e.g. C<GET()>, C<PUT()>, whatever.
=item hash ref
The current B<REQUEST_METHOD> is used as a key to the hash, the value should be
one the four above handler types. In this way you can specify different
handlers for each of the request types. The request method can also be
specified as '*', in which case that is used if a more specific match is not
found.
It is possible for the value of the handler to be another hash ref, rather than
one of the four above types. In this case it is assumed content-negotion is
wanted. The keys of this second hash are MIME types and the values are one of
the four above types. For example:
$self->resourceHooks(
qr{/parts/(\d+)} => {
GET => {
'text/json' => 'get_json',
'text/xml', => 'get_xml',
'text/xml' => 'get_html',
'*/*' => 'get_html',
},
'*' => sub { die "Bad Method!" },
}
);
=back
The return value of the handler is expected to be a string, which
L<REST::Application> will then send to the browser with the
C<sendRepresentation()> method.
If no argument is supplied to C<resourceHooks()> then the current set of hooks
is returned. The returned hash referces is a tied IxHash, so the keys are kept
sorted.
=head2 loadResource([$path])
This method will take the value of B<PATH_INFO>, iterate through the path
regex's set in C<resourceHooks()> and if it finds a match call the associated
handler and return the handler's value, which should be a scalar. If $path is
passed in then that is used instead of B<PATH_INFO>.
=head2 run()
This method calls C<loadResource()> with no arguments and then takes that
output and sends it to the remote client. Headers are sent with
C<sendHeaders()> and the representation is sent with C<sendRepresentation()>.
If the environment variable B<REST_APP_RETURN_ONLY> is set then output isn't
sent to the client. The return value of this method is the text output it
sends (or would've sent).
=head2 sendHeaders()
This method returns the headers as a string.
=head2 sendRepresentation($representation)
This method just returns C<$representation>. It is provided solely for
overloading purposes.
=head2 headerType([$type])
This accessor/mutator controls the type of header to be returned. This method
returns one of "header, redirect, or none." If C<$type> is passed in then that
is used to set the header type.
=head2 header([%args])
This accessor/mutator controls the header values sent. If called without
arguments then it simply returns the current header values as a hash, where the
keys are the header fields and the values are the header field values.
If this method is called multiple times then the values of %args are additive.
So calling C<$self-E<gt>header(-type =E<gt> 'text/html')> and C<$self-E<gt>header(-foo =E<gt>
'bar')> results in both the content-type header being set and the "foo" header
being set.
=head2 resetHeader()
This header causes the current header values to be reset. The previous values
are returned.
=head2 defaultResourceHandler()
This method is called by C<loadResource()> if no regex in C<resourceHooks()>
matches the current B<PATH_INFO>. It returns undef by default, it exists for
overloading.
=head2 bestContentType(@types)
Given a list of MIME types this function returns the best matching type
considering the Accept header of the current request (as returned by
C<getAcceptHeader()>).
=head2 simpleContentNegotiation(@types)
Given a list of MIME types this function returns the same list sorted from best
match to least considering the Accept header as returned by
C<getAcceptHeader()>.
=head2 getContentPrefs()
Returns the list of MIME types in the Accept header form most preferred to
least preferred. Quality weights are taken into account.
=head2 getAcceptHeader()
Returns the value of the Accept header as a single string.
=head2 scoreType($type, @accept_types)
Returns an integer, only good for sorting, for where C<$type> fits among the
C<@accept_types>. This method takes wildcards into account. So C<text/plain>
matches C<text/*>. The integer returned is the position in C<@accept_types> of
the matching MIME type. It assumped @accept_types is already sorted from best
to worst.
=head2 getLastMatchPath()
Returns the last path passed to C<checkMatch()> which successfully matched
against. Unless you're overloading things in funny ways the value returned
will be the path that caused the current handler to be invoked.
=head2 getLastMatchPattern()
Similar to C<getLastMatchPath()> except this is the pattern that was applied to
the path.
=head2 getRequestMethod()
This method tries to be smart and allow tunneling of the other HTTP methods
over GET or PUT. You can tunnel three ways with the higher up taking
precedence:
1) Pass an X-HTTP-Method header
2) Pass the 'http_method' query parameter
3) Pass a parameter via POST
Only POST and GET, being the most common, can be used to tunnel. In an attempt
to prevent people from being bad, GET can only be used to tunnel GET or HEAD.
POST can be used to tunnel anything.
=head1 AUTHORS
Matthew O'Connor E<lt>matthew@canonical.orgE<gt>
=head1 LICENSE
This program is free software. It is subject to the same license as Perl itself.
=head1 SEE ALSO
L<CGI>, L<CGI::Application>, L<Tie::IxHash>, L<CGI::Application::Dispatch>
=cut
|