package Google::Search; use warnings; use strict; =head1 NAME Google::Search - Interface to the Google AJAX Search API =head1 VERSION Version 0.023 =cut our $VERSION = '0.023'; =head1 SYNOPSIS my $key = ... # This should be a valid API key, gotten from: # http://code.google.com/apis/ajaxsearch/signup.html my $referer = "http://www.mysite.com/index.html" # This should be a valid referer for the above key my $search = Google::Search->Web(q => "rock", key => $key, referer => $referer); my $result = $search->first; while ($result) { print $result->number, " ", $result->uri, "\n"; $result = $result->next; } $search = Google::Search->Local(..., q => "rock"); $search = Google::Search->Video(..., q => "rock"); $search = Google::Search->Blog(..., q => "rock"); $search = Google::Search->News(..., q => "rock"); $search = Google::Search->Book(..., q => "rock"); $search = Google::Search->Image(..., q => "rock"); # You can also take advantage of each service's specialized interface # The search below specifies the latitude and longitude: $search = Google::Search->Local(..., q => { q => "rock", sll => "33.823230,-116.512110" }, ...); =head1 DESCRIPTION Google::Search is an interface to the Google AJAX Search API (L). Currently, their API looks like it will fetch you the top 28 results for your search query. According to the Terms of Service, you need to sign up for an API key here: L =cut use Moose; use Google::Search::Carp; use Google::Search::Response; use Google::Search::Page; use Google::Search::Result; use Google::Search::Error; use LWP::UserAgent; use constant SERVICE_URI => { web => 'http://ajax.googleapis.com/ajax/services/search/web', local => 'http://ajax.googleapis.com/ajax/services/search/local', video => 'http://ajax.googleapis.com/ajax/services/search/video', blog => 'http://ajax.googleapis.com/ajax/services/search/blogs', blogs => 'http://ajax.googleapis.com/ajax/services/search/blogs', news => 'http://ajax.googleapis.com/ajax/services/search/news', book => 'http://ajax.googleapis.com/ajax/services/search/books', books => 'http://ajax.googleapis.com/ajax/services/search/books', image => 'http://ajax.googleapis.com/ajax/services/search/images', images => 'http://ajax.googleapis.com/ajax/services/search/images', }; =head1 METHODS =head2 Google::Search->Web(...) Create a new web search. See C for more information. =head2 Google::Search->Local(...) Create a new local search. See C for more information. =head2 Google::Search->Video(...) Create a new video search. See C for more information. =head2 Google::Search->Blog(...) =head2 Google::Search->Blogs(...) Create a new blog search. See C for more information. =head2 Google::Search->News(...) Create a new news search. See C for more information. =head2 Google::Search->Book(...) =head2 Google::Search->Books(...) Create a new book search. See C for more information. =head2 Google::Search->Image(...) =head2 Google::Search->Images(...) Create a new book search. See C for more information. =head2 Google::Search->new(...) Create and return a new Google::Search object You can configure the search by passing the following to C: q The search phrase to submit to Google Optionally, this can also be a hash of parameters to submit. You can use the hash form to take advantage of each service's varying interface. Make sure to at least include a C parameter with your search. key Your Google AJAX Search API key (see Description) referer A referer that is valid for the above key service The service to search under. This can be any of: web, local, video, blog, news, book, or image. =cut for my $service (keys %{ SERVICE_URI() }) { no strict 'refs'; my $method = ucfirst $service; *$method = sub { my $class = shift; return $class->new(service => $service, @_); }; } has agent => qw/is ro required 1 lazy 1 isa LWP::UserAgent/, default => sub { my $self = shift; my $agent = LWP::UserAgent->new; $agent->env_proxy; return $agent; }; has service => qw/is ro lazy 1/, default => "web"; has uri => qw/is ro required 1 lazy 1 isa URI/, default => sub { my $self = shift; return URI->new($self->uri_for_service($self->service)); }; has q => qw/is ro required 1/; has v => qw/is ro required 1 isa Str/, default => sub { "1.0" }; has referer => qw/is ro isa Str/; has key => qw/is ro isa Str/; has rsz => qw/is ro required 1 isa Str default small/; has rsz_number => qw/is ro required 1 isa Int/, default => sub { my $self = shift; my $rsz = $self->rsz; return 4 if $rsz eq "small"; return 8 if $rsz eq "large"; croak "Don't understand rsz ($rsz)"; }; has _page => qw/is ro required 1/, default => sub { [] }; has _result => qw/is ro required 1/, default => sub { [] }; has current => qw/is ro required 1 lazy 1/, default => sub { return shift->first; }; has error => qw/is rw/; sub uri_for_service { my $self = shift; my $service = shift; return unless $service; $service = lc $service; $service =~ s/^\s*//g; $service =~ s/\s*$//g; return SERVICE_URI->{$service}; } sub request { my $self = shift; my @parameters; my @headers; my $referer = $self->referer; my $key = $self->key; push @headers, Referer => $referer if $referer; push @parameters, key => $key if $key; my $q = $self->q; if (ref $q eq "HASH") { push @parameters, %$q; } else { push @parameters, q => $q; } my $uri = $self->uri->clone; $uri->query_form({ v => $self->v, rsz => $self->rsz, @parameters, @_ }); return unless my $http_response = $self->agent->get($uri, @headers); return Google::Search::Response->new(http_response => $http_response); } sub page { my $self = shift; my $number = shift; $self->error(undef); my $page = $self->_page->[$number] ||= Google::Search::Page->new(search => $self, number => $number); $self->error($page->error) if $page->error; return $page; } =head2 $search->first Returns a L representing the first result in the search, if any. Returns undef if nothing was found =cut sub first { my $self = shift; return $self->result(0); } =head2 $search->next An iterator for $search. Will the return the next result each time it is called, and undef when there are no more results. Returns a L Returns undef if nothing was found =cut sub next { my $self = shift; return $self->current unless $self->{current}; return $self->{current} = $self->current->next; } =head2 $search->result( ) Returns a L corresponding to the result at These are equivalent: $search->result(0) $search->first =cut sub result { my $self = shift; my $number = shift; $self->error(undef); return $self->_result->[$number] if $self->_result->[$number]; my $result = do { my $result_number = $number % $self->rsz_number; my $page_number = int($number / $self->rsz_number); my $page = $self->page($page_number); my $content = $page->result($result_number); if ($content) { Google::Search::Result->parse($content, page => $page, search => $self, number => $number); } else { undef; } }; return undef unless $result; return $self->_result->[$number] = $result; } =head2 $search->all Returns L list which includes every result Google has returned for the query In scalar context an array reference is returned, a list otherwise An empty list is returned if nothing was found =cut sub all { my $self = shift; my $result = $self->first; 1 while $result && ($result = $result->next); # Fetch everything if ($self->error) { die $self->error->reason unless $self->error->message eq "out of range start"; } my @results = @{ $self->_result }; return wantarray ? @results : \@results; } =head2 $search->match( ) Returns a L list This method will iterate through each result in the search, passing the result to as the first argument. If returns true, then the result will be included in the returned list In scalar context this method returns the number of matches =cut sub match { my $self = shift; my $matcher = shift; my @matched; my $result = $self->first; while ($result) { push @matched, $result if $matcher->($result); $result = $result->next; } if ($self->error) { die $self->error->reason unless $self->error->message eq "out of range start"; } return @matched; } =head2 $search->first_match( ) Returns a L that is the first to match This method will iterate through each result in the search, passing the result to as the first argument. If returns true, then the result will be returned and iteration will stop. =cut sub first_match { my $self = shift; my $matcher = shift; my $result = $self->first; while ($result) { return $result if $matcher->($result); $result = $result->next; } if ($self->error) { die $self->error->reason unless $self->error->message eq "out of range start"; } return undef; } $_->meta->make_immutable for qw/ Google::Search Google::Search::Response Google::Search::Page Google::Search::Result Google::Search::Error /; =head2 $search->error Returns a L if there was an error with the last search If you receive undef from a result access then you can use this routine to see if there was a problem warn $search->error->reason; warn $search->error->http_response->as_string; # Etc, etc. This will return undef if no error was encountered =cut =head1 AUTHOR Robert Krimen, C<< >> =head1 SEE ALSO L =head1 BUGS Please report any bugs or feature requests to C, or through the web interface at L. I will be notified, and then you'll automatically be notified of progress on your bug as I make changes. =head1 SUPPORT You can find documentation for this module with the perldoc command. perldoc Google::Search You can also look for information at: =over 4 =item * RT: CPAN's request tracker L =item * AnnoCPAN: Annotated CPAN documentation L =item * CPAN Ratings L =item * Search CPAN L =back =head1 ACKNOWLEDGEMENTS =head1 COPYRIGHT & LICENSE Copyright 2008 Robert Krimen, all rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. =cut 1; # End of Google::Search