Difference between revisions of "Contribute: Plugins/ImportPluginsAWS"
(→Import Plugin Tutorial 2: Amazon Web Services) |
m (Code formatting) |
||
(25 intermediate revisions by 2 users not shown) | |||
Line 1: | Line 1: | ||
+ | [[Category:Contribute]] | ||
+ | [[Category:Plugins]] | ||
= Import Plugin Tutorial 2: Amazon Web Services = | = Import Plugin Tutorial 2: Amazon Web Services = | ||
− | In the last tutorial we created an import plugin that took data which needed very little modification to import into the respository. The column names in the CSV file matched the names of metadata fields present in the repository. In this tutorial we'll look at importing data that needs some modification to be imported, needs more error checking and is obtained in a different way. | + | In the [[Contribute:_Plugins/ImportPluginsCSV|last tutorial]] we created an import plugin that took data which needed very little modification to import into the respository. The column names in the [http://en.wikipedia.org/wiki/Comma-Separated_Values CSV] file matched the names of metadata fields present in the repository. In this tutorial we'll look at importing data that needs some modification to be imported, needs more error checking and is obtained in a different way. |
− | We | + | We will be using Amazon's E-Commerce Webservice to import books from their website into our respository given a list of [http://en.wikipedia.org/wiki/ASIN ASIN]. |
+ | |||
+ | We will be accessing the service using a [http://en.wikipedia.org/wiki/REST REST] approach, communicating with the server using URL parameters and retrieving an XML document in response to our request. It is also possible to access their services using [http://en.wikipedia.org/wiki/SOAP SOAP], but that will not be discussed here. | ||
= Before You Start = | = Before You Start = | ||
+ | |||
+ | == Amazon Web Services == | ||
+ | To use Amazon's web services you must first signup for an account [http://aws.amazon.com here]. Their site has extensive documentation on the services that they offer as well as example programs including some written in Perl. | ||
+ | |||
+ | == Required Modules == | ||
+ | To prepare for this tutorial you should make sure the [http://search.cpan.org/~gaas/libwww-perl-5.805/lib/LWP/UserAgent.pm LWP::UserAgent] module is installed. The following command as root, or using sudo should work. | ||
+ | |||
+ | <pre> | ||
+ | cpan LWP::UserAgent | ||
+ | </pre> | ||
+ | |||
= AWS.pm = | = AWS.pm = | ||
− | < | + | The code in the section below should be placed in a file called CSV.pm in the directory created previously, and MyPlugins should be changed to the name of that directory. |
+ | |||
+ | <syntaxhighlight lang="perl"> | ||
package EPrints::Plugin::Import::MyPlugins::AWS; | package EPrints::Plugin::Import::MyPlugins::AWS; | ||
Line 16: | Line 33: | ||
our @ISA = ('EPrints::Plugin::Import::TextFile'); | our @ISA = ('EPrints::Plugin::Import::TextFile'); | ||
− | my $endpoint = | + | my $endpoint = 'http://ecs.amazonaws.co.uk/onca/xml'; |
my $accesskey = '<YOURAMAZONWSKEY>'; | my $accesskey = '<YOURAMAZONWSKEY>'; | ||
− | my $service = | + | my $service = 'AWSECommerceService'; |
− | my $operation = | + | my $operation = 'ItemLookup'; |
− | my $version = | + | my $version = '2007-07-16'; |
+ | my $responsegroup = 'Large'; | ||
sub new | sub new | ||
Line 35: | Line 53: | ||
{ | { | ||
$self->{visible} = ''; | $self->{visible} = ''; | ||
− | $self->{error} = ' | + | $self->{error} = 'Unable to load required module LWP::UserAgent'; |
} | } | ||
Line 71: | Line 89: | ||
my %output = (); | my %output = (); | ||
− | $input =~ m/([ | + | $input =~ m/([A-Za-z0-9]+)/; |
$input = $1; | $input = $1; | ||
Line 81: | Line 99: | ||
"ItemId=$input&". | "ItemId=$input&". | ||
"Version=$version&". | "Version=$version&". | ||
− | "ResponseGroup= | + | "ResponseGroup=$responsegroup"; |
my $ua = LWP::UserAgent->new; | my $ua = LWP::UserAgent->new; | ||
Line 90: | Line 108: | ||
my $rep = | my $rep = | ||
− | $dom->getElementsByTagName( | + | $dom->getElementsByTagName('Items')->item(0)-> |
− | getElementsByTagName( | + | getElementsByTagName('Request')->item(0); |
my $reptext = | my $reptext = | ||
− | EPrints::Utils::tree_to_utf8($rep->getElementsByTagName( | + | EPrints::Utils::tree_to_utf8($rep->getElementsByTagName('IsValid')->item(0)); |
unless ($reptext eq 'True') | unless ($reptext eq 'True') | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('Invalid AWS Request'); |
return undef; | return undef; | ||
} | } | ||
Line 104: | Line 122: | ||
#Get Item Object | #Get Item Object | ||
my $item = | my $item = | ||
− | $dom->getElementsByTagName( | + | $dom->getElementsByTagName('Items')->item(0)-> |
− | getElementsByTagName( | + | getElementsByTagName('Item')->item(0); |
unless (defined $item) | unless (defined $item) | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('No Item element found'); |
return undef; | return undef; | ||
} | } | ||
− | my $attr = $item->getElementsByTagName( | + | my $attr = $item->getElementsByTagName('ItemAttributes')->item(0); |
− | my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName( | + | my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName('ProductGroup')->item(0)); |
unless ($pg eq 'Book') | unless ($pg eq 'Book') | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('Product is not a book.'); |
return undef; | return undef; | ||
} | } | ||
− | $output{type} = | + | $output{type} = 'book'; |
− | $output{refereed} = | + | $output{refereed} = 'FALSE'; |
− | $output{ispublished} = | + | $output{ispublished} = 'pub'; |
− | my $title = $attr->getElementsByTagName( | + | my $title = $attr->getElementsByTagName('Title')->item(0); |
$output{title} = EPrints::Utils::tree_to_utf8($title); | $output{title} = EPrints::Utils::tree_to_utf8($title); | ||
− | my $url = $item->getElementsByTagName( | + | my $url = $item->getElementsByTagName('DetailPageURL')->item(0); |
$output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url)); | $output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url)); | ||
− | my $isbn = $attr->getElementsByTagName( | + | my $isbn = $attr->getElementsByTagName('ISBN')->item(0); |
if (defined $isbn) | if (defined $isbn) | ||
{ | { | ||
Line 139: | Line 157: | ||
} | } | ||
− | my $pages = $attr->getElementsByTagName( | + | my $pages = $attr->getElementsByTagName('NumberOfPages')->item(0); |
if (defined $pages) | if (defined $pages) | ||
{ | { | ||
Line 145: | Line 163: | ||
} | } | ||
− | my $publisher = $attr->getElementsByTagName( | + | my $publisher = $attr->getElementsByTagName('Publisher')->item(0); |
if (defined $publisher) | if (defined $publisher) | ||
{ | { | ||
Line 151: | Line 169: | ||
} | } | ||
− | my $pubdate = $attr->getElementsByTagName( | + | my $pubdate = $attr->getElementsByTagName('PublicationDate')->item(0); |
if (defined $pubdate) | if (defined $pubdate) | ||
{ | { | ||
Line 161: | Line 179: | ||
1; | 1; | ||
− | </ | + | </syntaxhighlight> |
= In More Detail = | = In More Detail = | ||
+ | We will use the URI::Escape module in this plugin. As it is included with EPrints we don't need to check if it exists first. | ||
<pre> | <pre> | ||
use URI::Escape; | use URI::Escape; | ||
</pre> | </pre> | ||
+ | |||
+ | Here we setup a number of values for parameters that will be part of our web service requests. The endpoint variable determines which server will be sent the request. Here we have used the UK server, but by changing the [http://en.wikipedia.org/wiki/TLD TLD] we can use the US, Canadian, German or French servers. | ||
+ | |||
+ | The accesskey variable stores the access key you will have gained from signing up to Amazon earlier. You should use the normal access key and not the secret one. | ||
+ | |||
+ | Here we use the ItemLookup operation of the AWSECommerceService with the 2007-07-16 version of the API. Other operations allow searching for items, but here we want to look up specific products. Finally the variable responsegroup determines the amount and nature of the information returned, we select "Large" in this case, which gives us more information about the item. | ||
+ | |||
<pre> | <pre> | ||
− | my $endpoint = | + | my $endpoint = 'http://ecs.amazonaws.co.uk/onca/xml'; |
my $accesskey = '<YOURAMAZONWSKEY>'; | my $accesskey = '<YOURAMAZONWSKEY>'; | ||
− | my $service = | + | my $service = 'AWSECommerceService'; |
− | my $operation = | + | my $operation = 'ItemLookup'; |
− | my $version = | + | my $version = '2007-07-16'; |
+ | my $responsegroup = 'Large'; | ||
</pre> | </pre> | ||
== Constructor == | == Constructor == | ||
+ | The constructor is similar to the one used for the [Contribute:_Plugins/ImportPluginsCSV CSV plugin], except this one will import individual eprints, given an [http://en.wikipedia.org/wiki/ASIN ASIN]. | ||
<pre> | <pre> | ||
$self->{produce} = [ 'list/eprint' , 'dataobj/eprint']; | $self->{produce} = [ 'list/eprint' , 'dataobj/eprint']; | ||
</pre> | </pre> | ||
+ | Like we imported Text::CSV in the [Contribute:_Plugins/ImportPluginsCSV last tutorial], here we import LWP::UserAgent which will be used for making requests to the web service. | ||
<pre> | <pre> | ||
my $rc = EPrints::Utils::require_if_exists('LWP::UserAgent'); | my $rc = EPrints::Utils::require_if_exists('LWP::UserAgent'); | ||
Line 185: | Line 214: | ||
{ | { | ||
$self->{visible} = ''; | $self->{visible} = ''; | ||
− | $self->{error} = ' | + | $self->{error} = 'Unable to load required module LWP::UserAgent'; |
} | } | ||
</pre> | </pre> | ||
Line 191: | Line 220: | ||
== Input == | == Input == | ||
=== input_fh === | === input_fh === | ||
+ | This method is similar to the one used in the [Contribute:_Plugins/ImportPluginsCSV CSV plugin], but doesn't have to do quite so much work. | ||
+ | |||
+ | First we create the array to hold our imported eprint ids. | ||
<pre> | <pre> | ||
my @ids; | my @ids; | ||
</pre> | </pre> | ||
+ | |||
+ | Next we read all the lines in the supplied file handle into our records array. | ||
<pre> | <pre> | ||
my $fh = $opts{fh}; | my $fh = $opts{fh}; | ||
Line 200: | Line 234: | ||
</pre> | </pre> | ||
+ | Then we iterate over each record, running convert_input on it, importing it into our repository and adding the id to our array. | ||
<pre> | <pre> | ||
foreach my $input_data (@records) | foreach my $input_data (@records) | ||
Line 214: | Line 249: | ||
</pre> | </pre> | ||
+ | Then we return a List object of the items imported. | ||
<pre> | <pre> | ||
return EPrints::List->new( | return EPrints::List->new( | ||
Line 222: | Line 258: | ||
=== convert_input === | === convert_input === | ||
+ | |||
+ | [http://en.wikipedia.org/wiki/ASIN ASINs] are strings which identify a product. Here we remove any non-alphanumerical characters which are surrounding the [http://en.wikipedia.org/wiki/ASIN ASIN]. | ||
<pre> | <pre> | ||
− | $input =~ m/([ | + | $input =~ m/([A-Za-z0-9]+)/; |
$input = $1; | $input = $1; | ||
</pre> | </pre> | ||
+ | We form the request from the variables we created earlier and the [http://en.wikipedia.org/wiki/ASIN ASIN] we have just obtained. | ||
<pre> | <pre> | ||
− | |||
my $request = | my $request = | ||
"$endpoint?". | "$endpoint?". | ||
Line 236: | Line 274: | ||
"ItemId=$input&". | "ItemId=$input&". | ||
"Version=$version&". | "Version=$version&". | ||
− | "ResponseGroup= | + | "ResponseGroup=$responsegroup"; |
</pre> | </pre> | ||
+ | We now send the request, by creating a new LWP::UserAgent object, setting its timeout to 30 seconds and then performing the request using HTTP GET. | ||
<pre> | <pre> | ||
− | |||
my $ua = LWP::UserAgent->new; | my $ua = LWP::UserAgent->new; | ||
$ua->timeout(30); | $ua->timeout(30); | ||
Line 246: | Line 284: | ||
</pre> | </pre> | ||
+ | We then create a [http://en.wikipedia.org/wiki/Document_Object_Model DOM] object from the XML document returned. | ||
<pre> | <pre> | ||
− | |||
my $dom = EPrints::XML::parse_xml_string($response->content); | my $dom = EPrints::XML::parse_xml_string($response->content); | ||
</pre> | </pre> | ||
+ | Each request contains an Items element within the root element of the document which contains a Request element. This element contains an element IsValid. This element will contain the value True or False depending on whether a valid request was made or not. | ||
+ | |||
+ | Here we obtain the Request element and check that the IsValid element within it contains the value True. If it doesn't we call the error method and return undef. | ||
<pre> | <pre> | ||
− | |||
my $rep = | my $rep = | ||
$dom->getElementsByTagName("Items")->item(0)-> | $dom->getElementsByTagName("Items")->item(0)-> | ||
− | getElementsByTagName( | + | getElementsByTagName('Request')->item(0); |
my $reptext = | my $reptext = | ||
− | EPrints::Utils::tree_to_utf8($rep->getElementsByTagName( | + | EPrints::Utils::tree_to_utf8($rep->getElementsByTagName('IsValid')->item(0)); |
unless ($reptext eq 'True') | unless ($reptext eq 'True') | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('Invalid AWS Request'); |
return undef; | return undef; | ||
} | } | ||
</pre> | </pre> | ||
+ | The product found with the ItemLookup method is contained within an Item element within the Items element. Here we attempt to get that element and raise the error and return undef if we can't. | ||
<pre> | <pre> | ||
− | |||
my $item = | my $item = | ||
− | $dom->getElementsByTagName( | + | $dom->getElementsByTagName('Items')->item(0)-> |
− | getElementsByTagName( | + | getElementsByTagName('Item')->item(0); |
unless (defined $item) | unless (defined $item) | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('No Item element found'); |
return undef; | return undef; | ||
} | } | ||
</pre> | </pre> | ||
+ | Each item contains an ItemAttributes element which contains most of the metadata about an item. | ||
<pre> | <pre> | ||
− | + | my $attr = $item->getElementsByTagName('ItemAttributes')->item(0); | |
− | my $attr = $item->getElementsByTagName( | ||
</pre> | </pre> | ||
+ | For some specialised repositories it might make sense to import DVDs, computer games and electronic equipment, but we're just going to deal with books. The ProductGroup element within the ItemAttributes element tells you what sort of item we're dealing with. We're looking for the value 'Book'. | ||
<pre> | <pre> | ||
− | + | my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName('ProductGroup')->item(0)); | |
− | my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName( | ||
unless ($pg eq 'Book') | unless ($pg eq 'Book') | ||
{ | { | ||
− | $plugin->error( | + | $plugin->error('Product is not a book.'); |
return undef; | return undef; | ||
} | } | ||
</pre> | </pre> | ||
+ | Here we set a few fields without consulting the imported data. We know this is a book, so we set the type. We assume that it has not been refereed. We also assume it has been published, because we can buy it. | ||
<pre> | <pre> | ||
− | + | $output{type} = 'book'; | |
− | $output{type} = | + | $output{refereed} = 'FALSE'; |
− | $output{refereed} = | + | $output{ispublished} = 'pub'; |
− | $output{ispublished} = | ||
</pre> | </pre> | ||
+ | We get and set the title. | ||
<pre> | <pre> | ||
− | + | my $title = $attr->getElementsByTagName('Title')->item(0); | |
− | my $title = $attr->getElementsByTagName( | ||
$output{title} = EPrints::Utils::tree_to_utf8($title); | $output{title} = EPrints::Utils::tree_to_utf8($title); | ||
</pre> | </pre> | ||
+ | Here we set the official URL to the Amazon product page. We have to do a bit of extra work using the uri_unescape method from the URI::Escape package to convert URI escape codes into characters. | ||
<pre> | <pre> | ||
− | + | my $url = $item->getElementsByTagName('DetailPageURL')->item(0); | |
− | my $url = $item->getElementsByTagName( | ||
$output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url)); | $output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url)); | ||
</pre> | </pre> | ||
+ | We can set the [http://en.wikipedia.org/wiki/ISBN ISBN]. Note that the [http://en.wikipedia.org/wiki/ISBN ISBN] is often the same as the [http://en.wikipedia.org/wiki/ASIN ASIN]. | ||
<pre> | <pre> | ||
− | + | my $isbn = $attr->getElementsByTagName('ISBN')->item(0); | |
− | my $isbn = $attr->getElementsByTagName( | ||
if (defined $isbn) | if (defined $isbn) | ||
{ | { | ||
Line 324: | Line 364: | ||
</pre> | </pre> | ||
+ | We can set the number of pages. | ||
<pre> | <pre> | ||
− | + | my $pages = $attr->getElementsByTagName('NumberOfPages')->item(0); | |
− | my $pages = $attr->getElementsByTagName( | ||
if (defined $pages) | if (defined $pages) | ||
{ | { | ||
Line 333: | Line 373: | ||
</pre> | </pre> | ||
+ | We can set the publisher. | ||
<pre> | <pre> | ||
− | + | my $publisher = $attr->getElementsByTagName('Publisher')->item(0); | |
− | my $publisher = $attr->getElementsByTagName( | ||
if (defined $publisher) | if (defined $publisher) | ||
{ | { | ||
Line 342: | Line 382: | ||
</pre> | </pre> | ||
+ | We can set the publication date and finally return our output hash. | ||
<pre> | <pre> | ||
− | my $pubdate = $attr->getElementsByTagName( | + | my $pubdate = $attr->getElementsByTagName('PublicationDate')->item(0); |
if (defined $pubdate) | if (defined $pubdate) | ||
{ | { | ||
Line 353: | Line 394: | ||
= Testing Your Plugin = | = Testing Your Plugin = | ||
+ | After restarting your webserver go to the Import Items screen from the Manage Deposits screen. If you can't find this, make sure you're logged in. | ||
+ | |||
+ | We'll start by collecting a few ASINS. Go to [http://www.amazon.co.uk Amazon] and pick a few books. The URL for each project page is in the form http://www.amazon.co.uk/Combination-of-title-an-author/dp/ASIN... Collect a few different ASINS. | ||
+ | |||
+ | Now we'll demonstrate importing from Amazon with a few sample ASINs. | ||
+ | Type this into the "Cut and Paste Records" box: | ||
+ | <pre> | ||
+ | 0946719616 | ||
+ | 0297843877 | ||
+ | </pre> | ||
+ | |||
+ | Select "AWS" from the Select import format drop down menu and click "Test Run + Import". You should end up at the Manage Deposits screen with the following message being displayed "Import completed: 2 item(s) imported.". |
Latest revision as of 11:13, 24 May 2021
Contents
Import Plugin Tutorial 2: Amazon Web Services
In the last tutorial we created an import plugin that took data which needed very little modification to import into the respository. The column names in the CSV file matched the names of metadata fields present in the repository. In this tutorial we'll look at importing data that needs some modification to be imported, needs more error checking and is obtained in a different way.
We will be using Amazon's E-Commerce Webservice to import books from their website into our respository given a list of ASIN.
We will be accessing the service using a REST approach, communicating with the server using URL parameters and retrieving an XML document in response to our request. It is also possible to access their services using SOAP, but that will not be discussed here.
Before You Start
Amazon Web Services
To use Amazon's web services you must first signup for an account here. Their site has extensive documentation on the services that they offer as well as example programs including some written in Perl.
Required Modules
To prepare for this tutorial you should make sure the LWP::UserAgent module is installed. The following command as root, or using sudo should work.
cpan LWP::UserAgent
AWS.pm
The code in the section below should be placed in a file called CSV.pm in the directory created previously, and MyPlugins should be changed to the name of that directory.
package EPrints::Plugin::Import::MyPlugins::AWS;
use EPrints::Plugin::Import::TextFile;
use strict;
use URI::Escape;
our @ISA = ('EPrints::Plugin::Import::TextFile');
my $endpoint = 'http://ecs.amazonaws.co.uk/onca/xml';
my $accesskey = '<YOURAMAZONWSKEY>';
my $service = 'AWSECommerceService';
my $operation = 'ItemLookup';
my $version = '2007-07-16';
my $responsegroup = 'Large';
sub new
{
my( $class, %params ) = @_;
my $self = $class->SUPER::new( %params );
$self->{name} = 'AWS';
$self->{visible} = 'all';
$self->{produce} = [ 'list/eprint' , 'dataobj/eprint'];
my $rc = EPrints::Utils::require_if_exists('LWP::UserAgent');
unless ($rc)
{
$self->{visible} = '';
$self->{error} = 'Unable to load required module LWP::UserAgent';
}
return $self;
}
sub input_fh
{
my( $plugin, %opts ) = @_;
my @ids;
my $fh = $opts{fh};
my @records = <$fh>;
foreach my $input_data (@records)
{
my $epdata = $plugin->convert_input($input_data);
next unless defined $epdata;
my $dataobj = $plugin->epdata_to_dataobj($opts{dataset},$epdata);
if( defined $dataobj )
{
push @ids, $dataobj->get_id;
}
}
return EPrints::List->new(
dataset => $opts{dataset},
session => $plugin->{session},
ids=>\@ids );
}
sub convert_input
{
my ($plugin, $input) = @_;
my %output = ();
$input =~ m/([A-Za-z0-9]+)/;
$input = $1;
my $request =
"$endpoint?".
"Service=$service&".
"AWSAccessKeyId=$accesskey&".
"Operation=$operation&".
"ItemId=$input&".
"Version=$version&".
"ResponseGroup=$responsegroup";
my $ua = LWP::UserAgent->new;
$ua->timeout(30);
my $response = $ua->get($request);
my $dom = EPrints::XML::parse_xml_string($response->content);
my $rep =
$dom->getElementsByTagName('Items')->item(0)->
getElementsByTagName('Request')->item(0);
my $reptext =
EPrints::Utils::tree_to_utf8($rep->getElementsByTagName('IsValid')->item(0));
unless ($reptext eq 'True')
{
$plugin->error('Invalid AWS Request');
return undef;
}
#Get Item Object
my $item =
$dom->getElementsByTagName('Items')->item(0)->
getElementsByTagName('Item')->item(0);
unless (defined $item)
{
$plugin->error('No Item element found');
return undef;
}
my $attr = $item->getElementsByTagName('ItemAttributes')->item(0);
my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName('ProductGroup')->item(0));
unless ($pg eq 'Book')
{
$plugin->error('Product is not a book.');
return undef;
}
$output{type} = 'book';
$output{refereed} = 'FALSE';
$output{ispublished} = 'pub';
my $title = $attr->getElementsByTagName('Title')->item(0);
$output{title} = EPrints::Utils::tree_to_utf8($title);
my $url = $item->getElementsByTagName('DetailPageURL')->item(0);
$output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url));
my $isbn = $attr->getElementsByTagName('ISBN')->item(0);
if (defined $isbn)
{
$output{isbn} = EPrints::Utils::tree_to_utf8($isbn);
}
my $pages = $attr->getElementsByTagName('NumberOfPages')->item(0);
if (defined $pages)
{
$output{pages} = EPrints::Utils::tree_to_utf8($pages);
}
my $publisher = $attr->getElementsByTagName('Publisher')->item(0);
if (defined $publisher)
{
$output{publisher} = EPrints::Utils::tree_to_utf8($publisher);
}
my $pubdate = $attr->getElementsByTagName('PublicationDate')->item(0);
if (defined $pubdate)
{
$output{date} = EPrints::Utils::tree_to_utf8($pubdate);
}
return \%output;
}
1;
In More Detail
We will use the URI::Escape module in this plugin. As it is included with EPrints we don't need to check if it exists first.
use URI::Escape;
Here we setup a number of values for parameters that will be part of our web service requests. The endpoint variable determines which server will be sent the request. Here we have used the UK server, but by changing the TLD we can use the US, Canadian, German or French servers.
The accesskey variable stores the access key you will have gained from signing up to Amazon earlier. You should use the normal access key and not the secret one.
Here we use the ItemLookup operation of the AWSECommerceService with the 2007-07-16 version of the API. Other operations allow searching for items, but here we want to look up specific products. Finally the variable responsegroup determines the amount and nature of the information returned, we select "Large" in this case, which gives us more information about the item.
my $endpoint = 'http://ecs.amazonaws.co.uk/onca/xml'; my $accesskey = '<YOURAMAZONWSKEY>'; my $service = 'AWSECommerceService'; my $operation = 'ItemLookup'; my $version = '2007-07-16'; my $responsegroup = 'Large';
Constructor
The constructor is similar to the one used for the [Contribute:_Plugins/ImportPluginsCSV CSV plugin], except this one will import individual eprints, given an ASIN.
$self->{produce} = [ 'list/eprint' , 'dataobj/eprint'];
Like we imported Text::CSV in the [Contribute:_Plugins/ImportPluginsCSV last tutorial], here we import LWP::UserAgent which will be used for making requests to the web service.
my $rc = EPrints::Utils::require_if_exists('LWP::UserAgent'); unless ($rc) { $self->{visible} = ''; $self->{error} = 'Unable to load required module LWP::UserAgent'; }
Input
input_fh
This method is similar to the one used in the [Contribute:_Plugins/ImportPluginsCSV CSV plugin], but doesn't have to do quite so much work.
First we create the array to hold our imported eprint ids.
my @ids;
Next we read all the lines in the supplied file handle into our records array.
my $fh = $opts{fh}; my @records = <$fh>;
Then we iterate over each record, running convert_input on it, importing it into our repository and adding the id to our array.
foreach my $input_data (@records) { my $epdata = $plugin->convert_input($input_data); next unless defined $epdata; my $dataobj = $plugin->epdata_to_dataobj($opts{dataset},$epdata); if( defined $dataobj ) { push @ids, $dataobj->get_id; } }
Then we return a List object of the items imported.
return EPrints::List->new( dataset => $opts{dataset}, session => $plugin->{session}, ids=>\@ids );
convert_input
ASINs are strings which identify a product. Here we remove any non-alphanumerical characters which are surrounding the ASIN.
$input =~ m/([A-Za-z0-9]+)/; $input = $1;
We form the request from the variables we created earlier and the ASIN we have just obtained.
my $request = "$endpoint?". "Service=$service&". "AWSAccessKeyId=$accesskey&". "Operation=$operation&". "ItemId=$input&". "Version=$version&". "ResponseGroup=$responsegroup";
We now send the request, by creating a new LWP::UserAgent object, setting its timeout to 30 seconds and then performing the request using HTTP GET.
my $ua = LWP::UserAgent->new; $ua->timeout(30); my $response = $ua->get($request);
We then create a DOM object from the XML document returned.
my $dom = EPrints::XML::parse_xml_string($response->content);
Each request contains an Items element within the root element of the document which contains a Request element. This element contains an element IsValid. This element will contain the value True or False depending on whether a valid request was made or not.
Here we obtain the Request element and check that the IsValid element within it contains the value True. If it doesn't we call the error method and return undef.
my $rep = $dom->getElementsByTagName("Items")->item(0)-> getElementsByTagName('Request')->item(0); my $reptext = EPrints::Utils::tree_to_utf8($rep->getElementsByTagName('IsValid')->item(0)); unless ($reptext eq 'True') { $plugin->error('Invalid AWS Request'); return undef; }
The product found with the ItemLookup method is contained within an Item element within the Items element. Here we attempt to get that element and raise the error and return undef if we can't.
my $item = $dom->getElementsByTagName('Items')->item(0)-> getElementsByTagName('Item')->item(0); unless (defined $item) { $plugin->error('No Item element found'); return undef; }
Each item contains an ItemAttributes element which contains most of the metadata about an item.
my $attr = $item->getElementsByTagName('ItemAttributes')->item(0);
For some specialised repositories it might make sense to import DVDs, computer games and electronic equipment, but we're just going to deal with books. The ProductGroup element within the ItemAttributes element tells you what sort of item we're dealing with. We're looking for the value 'Book'.
my $pg = EPrints::Utils::tree_to_utf8($attr->getElementsByTagName('ProductGroup')->item(0)); unless ($pg eq 'Book') { $plugin->error('Product is not a book.'); return undef; }
Here we set a few fields without consulting the imported data. We know this is a book, so we set the type. We assume that it has not been refereed. We also assume it has been published, because we can buy it.
$output{type} = 'book'; $output{refereed} = 'FALSE'; $output{ispublished} = 'pub';
We get and set the title.
my $title = $attr->getElementsByTagName('Title')->item(0); $output{title} = EPrints::Utils::tree_to_utf8($title);
Here we set the official URL to the Amazon product page. We have to do a bit of extra work using the uri_unescape method from the URI::Escape package to convert URI escape codes into characters.
my $url = $item->getElementsByTagName('DetailPageURL')->item(0); $output{official_url} = uri_unescape(EPrints::Utils::tree_to_utf8($url));
We can set the ISBN. Note that the ISBN is often the same as the ASIN.
my $isbn = $attr->getElementsByTagName('ISBN')->item(0); if (defined $isbn) { $output{isbn} = EPrints::Utils::tree_to_utf8($isbn); }
We can set the number of pages.
my $pages = $attr->getElementsByTagName('NumberOfPages')->item(0); if (defined $pages) { $output{pages} = EPrints::Utils::tree_to_utf8($pages); }
We can set the publisher.
my $publisher = $attr->getElementsByTagName('Publisher')->item(0); if (defined $publisher) { $output{publisher} = EPrints::Utils::tree_to_utf8($publisher); }
We can set the publication date and finally return our output hash.
my $pubdate = $attr->getElementsByTagName('PublicationDate')->item(0); if (defined $pubdate) { $output{date} = EPrints::Utils::tree_to_utf8($pubdate); } return \%output;
Testing Your Plugin
After restarting your webserver go to the Import Items screen from the Manage Deposits screen. If you can't find this, make sure you're logged in.
We'll start by collecting a few ASINS. Go to Amazon and pick a few books. The URL for each project page is in the form http://www.amazon.co.uk/Combination-of-title-an-author/dp/ASIN... Collect a few different ASINS.
Now we'll demonstrate importing from Amazon with a few sample ASINs. Type this into the "Cut and Paste Records" box:
0946719616 0297843877
Select "AWS" from the Select import format drop down menu and click "Test Run + Import". You should end up at the Manage Deposits screen with the following message being displayed "Import completed: 2 item(s) imported.".