You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
643 lines
20 KiB
643 lines
20 KiB
<?php |
|
|
|
/** |
|
* EBSCO EDS API class |
|
* |
|
* PHP version 5 |
|
* |
|
* |
|
* Copyright [2014] [EBSCO Information Services] |
|
* |
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
|
* you may not use this file except in compliance with the License. |
|
* You may obtain a copy of the License at |
|
* |
|
* http://www.apache.org/licenses/LICENSE-2.0 |
|
* |
|
* Unless required by applicable law or agreed to in writing, software |
|
* distributed under the License is distributed on an "AS IS" BASIS, |
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
* See the License for the specific language governing permissions and |
|
* limitations under the License. |
|
*/ |
|
|
|
require_once 'EBSCOConnector.php'; |
|
require_once 'EBSCOResponse.php'; |
|
|
|
|
|
/** |
|
* EBSCO API class |
|
*/ |
|
class EBSCOAPI |
|
{ |
|
/** |
|
* The authentication token used for API transactions |
|
* @global string |
|
*/ |
|
private $authenticationToken; |
|
|
|
|
|
/** |
|
* The session token for API transactions |
|
* @global string |
|
*/ |
|
private $sessionToken; |
|
|
|
|
|
/** |
|
* The EBSCOConnector object used for API transactions |
|
* @global object EBSCOConnector |
|
*/ |
|
private $connector; |
|
|
|
|
|
/** |
|
* Configuration options |
|
*/ |
|
private $config; |
|
|
|
|
|
/** |
|
* VuFind search types mapped to EBSCO search types |
|
* used for urls in search results / detailed result |
|
* @global array |
|
*/ |
|
private static $search_tags = array( |
|
'' => '', |
|
'AllFields' => '', |
|
'Abstract' => 'AB', |
|
'Author' => 'AU', |
|
'Source' => 'SO', |
|
'Subject' => 'SU', |
|
'Title' => 'TI' |
|
); |
|
|
|
|
|
/** |
|
* EBSCO sort options |
|
* @global array |
|
*/ |
|
private static $sort_options = array( |
|
'relevance', |
|
'date', |
|
'date2', |
|
'source' |
|
); |
|
|
|
|
|
/** |
|
* VuFind sort types mapped to EBSCO sort types |
|
* used for urls in Search results / Detailed view |
|
* @global array |
|
*/ |
|
private static $mapped_sort_options = array( |
|
'' => 'relevance', |
|
'relevance' => 'relevance', |
|
'subject' => 'date', |
|
'date' => 'date2', |
|
'date_asc' => 'date2', |
|
'date_desc' => 'date', |
|
'callnumber' => 'date', |
|
'author' => 'author', |
|
'title' => 'date' |
|
); |
|
|
|
|
|
/** |
|
* Constructor |
|
* |
|
* |
|
* @param array config |
|
* |
|
* @access public |
|
*/ |
|
public function __construct($config) |
|
{ |
|
$this->config = $config; |
|
} |
|
|
|
/** |
|
* Setter / Getter for authentication token |
|
* |
|
* @param string The authentication token |
|
* |
|
* @return string or none |
|
* @access public |
|
*/ |
|
public function authenticationToken($token = null) |
|
{ |
|
if (empty($token)) { |
|
$token = $this->readSession('authenticationToken'); |
|
return !empty($token) ? $token : $this->authenticationToken; |
|
} else { |
|
$this->authenticationToken = $token; |
|
$this->writeSession('authenticationToken', $token); |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Setter / Getter for session token |
|
* |
|
* @param string The session token |
|
* |
|
* @return string or none |
|
* @access public |
|
*/ |
|
public function sessionToken($token = null) |
|
{ |
|
if (empty($token)) { |
|
$token = $this->readSession('sessionToken'); |
|
return !empty($token) ? $token : $this->sessionToken; |
|
} else { |
|
$this->sessionToken = $token; |
|
$this->writeSession('sessionToken', $token); |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Getter for isGuest |
|
* |
|
* @param string 'y' or 'n' |
|
* |
|
* @return string or none |
|
* @access public |
|
*/ |
|
public function isGuest($boolean = null) |
|
{ |
|
if (empty($boolean)) { |
|
return $this->readSession('isGuest'); |
|
} else { |
|
$this->writeSession('isGuest', $boolean); |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Create a new EBSCOConnector object or reuse an existing one |
|
* |
|
* @param none |
|
* |
|
* @return EBSCOConnector object |
|
* @access public |
|
*/ |
|
public function connector() |
|
{ |
|
if (empty($this->connector)) { |
|
$this->connector = new EBSCOConnector($this->config); |
|
} |
|
return $this->connector; |
|
} |
|
|
|
|
|
/** |
|
* Create a new EBSCOResponse object |
|
* |
|
* @param object $response |
|
* |
|
* @return EBSCOResponse object |
|
* @access public |
|
*/ |
|
public function response($response) |
|
{ |
|
$responseObj = new EBSCOResponse($response); |
|
return $responseObj; |
|
} |
|
|
|
|
|
/** |
|
* Request authentication and session tokens, then send the API request. |
|
* Retry the request if authentication errors occur |
|
* |
|
* @param string $action The EBSCOConnector method name |
|
* @param array $params The parameters of the HTTP request |
|
* @param integer $attempts The number of retries |
|
* |
|
* @return object SimpleXml DOM or PEAR Error |
|
* @access protected |
|
*/ |
|
protected function request($action, $params = null, $attempts = 5) |
|
{ |
|
$authenticationToken = $this->authenticationToken(); |
|
$sessionToken = $this->sessionToken(); |
|
|
|
// If authentication token is missing then the session token is missing too, so get both tokens |
|
// If session token is missing then the authentication token may be invalid, so get both tokens |
|
if (empty($authenticationToken) || empty($sessionToken)) { |
|
$result = $this->apiAuthenticationAndSessionToken(); |
|
if ($this->isError($result)) { |
|
// Any error should terminate the request immediately |
|
// in order to prevent infinite recursion |
|
return $result; |
|
} |
|
} |
|
|
|
// Any change of the isGuest should request a new session |
|
// (and don't terminate the current request if there was an error during the session request |
|
// since it's not that important) |
|
if ($this->isGuest() != $this->connector()->isGuest()) { |
|
$this->apiSessionToken(); |
|
} |
|
|
|
$headers = array( |
|
'x-authenticationToken' => $this->authenticationToken(), |
|
'x-sessionToken' => $this->sessionToken() |
|
); |
|
|
|
$response = call_user_func_array(array($this->connector(), "request{$action}"), array($params, $headers)); |
|
if ($this->isError($response)) { |
|
// Retry the request if there were authentication errors |
|
$code = $response->getCode(); |
|
switch ($code) { |
|
// If authentication token is invalid then the session token is invalid too, so get both tokens |
|
// If session token is invalid then the authentication token may be invalid too, so get both tokens |
|
case EBSCOConnector::EDS_AUTH_TOKEN_INVALID: |
|
$result = $this->apiAuthenticationToken(); |
|
if ($this->isError($result)) { |
|
// Any error should terminate the request immediately |
|
// in order to prevent infinite recursion |
|
return $result; |
|
} |
|
if ($attempts > 0) { |
|
$result = $this->request($action, $params, --$attempts); |
|
} |
|
break; |
|
case EBSCOConnector::EDS_SESSION_TOKEN_INVALID: |
|
$result = $this->apiAuthenticationAndSessionToken(); |
|
if ($this->isError($result)) { |
|
// Any error should terminate the request immediately |
|
// in order to prevent infinite recursion |
|
return $result; |
|
} |
|
if ($attempts > 0) { |
|
$result = $this->request($action, $params, --$attempts); |
|
} |
|
break; |
|
default: |
|
$result = $this->handleError($response); |
|
break; |
|
} |
|
} else { |
|
$result = $this->response($response)->result(); |
|
} |
|
|
|
return $result; |
|
} |
|
|
|
|
|
/** |
|
* Wrapper for authentication API call |
|
* |
|
* @param none |
|
* |
|
* @access public |
|
*/ |
|
public function apiAuthenticationToken() |
|
{ |
|
$response = $this->connector()->requestAuthenticationToken(); |
|
|
|
if ($this->isError($response)) { |
|
return $response; |
|
} else { |
|
$result = $this->response($response)->result(); |
|
if (isset($result['authenticationToken'])) { |
|
$this->authenticationToken($result['authenticationToken']); |
|
return $result['authenticationToken']; |
|
} else { |
|
return new EBSCOException("No authentication token was found in the response."); |
|
} |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Wrapper for session API call |
|
* |
|
* @param none |
|
* |
|
* @access public |
|
*/ |
|
public function apiSessionToken() |
|
{ |
|
// Add authentication tokens to headers |
|
$headers = array( |
|
'x-authenticationToken' => $this->authenticationToken() |
|
); |
|
|
|
$response = $this->connector()->requestSessionToken($headers); |
|
|
|
// Raise the exception so that any code running this method should exit immediately |
|
if ($this->isError($response)) { |
|
return $response; |
|
} else { |
|
$result = $this->response($response)->result(); |
|
if (is_string($result)) { |
|
$this->sessionToken($result); |
|
return $result; |
|
} else { |
|
return new EBSCOException("No session token was found in the response."); |
|
} |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Initialize the authentication and session tokens |
|
* |
|
* @param none |
|
* |
|
* @access public |
|
*/ |
|
public function apiAuthenticationAndSessionToken() |
|
{ |
|
$authenticationToken = $this->apiAuthenticationToken(); |
|
if ($this->isError($authenticationToken)) { |
|
// An authentication error should terminate the request immediately |
|
return $authenticationToken; |
|
} |
|
|
|
$sessionToken = $this->apiSessionToken(); |
|
if ($this->isError($sessionToken)) { |
|
// A session error should terminate the request immediately |
|
return $sessionToken; |
|
} |
|
|
|
// We don't have to return anything, both tokens can be accessed using the getters |
|
return true; |
|
} |
|
|
|
|
|
/** |
|
* Wrapper for search API call |
|
* |
|
* @param array $search The search terms |
|
* @param array $filters The facet filters |
|
* @param string $start The page to start with |
|
* @param string $limit The number of records to return |
|
* @param string $sortBy The value to be used by for sorting |
|
* @param string $amount The amount of data to be returned |
|
* @param string $mode The search mode |
|
* |
|
* @throws object PEAR Error |
|
* @return array An array of query results |
|
* @access public |
|
*/ |
|
public function apiSearch($search, $filters, |
|
$start = 1, $limit = 10, $sortBy = 'relevance', $amount = 'detailed', $mode = 'all' |
|
) { |
|
$query = array(); |
|
|
|
// Basic search |
|
if(!empty($search['lookfor'])) { |
|
$lookfor = $search['lookfor']; |
|
$type = isset($search['index']) && !empty($search['index']) ? $search['index'] : 'AllFields'; |
|
|
|
// escape some characters from lookfor term |
|
$term = str_replace(array(',', ':', '(', ')'), array('\,', '\:', '\(', '\)'), $lookfor); |
|
// replace multiple consecutive empty spaces with one empty space |
|
$term = preg_replace("/\s+/", ' ', $term); |
|
|
|
// search terms |
|
// Complex search term |
|
if (preg_match('/(.*) (AND|OR) (.*)/i', $term)) { |
|
$query['query'] = $term; |
|
} else { |
|
$tag = self::$search_tags[$type]; |
|
$op = 'AND'; |
|
$query_str = implode(',', array($op, $tag)); |
|
$query_str = implode(($tag ? ':' : ''), array($query_str, $term)); |
|
$query['query-1'] = $query_str; |
|
} |
|
|
|
// Advanced search |
|
} else if(!empty($search['group'])) { |
|
|
|
$counter = 1; |
|
foreach ($search['group'] as $group) { |
|
$type = $group['type']; |
|
if (isset($group['lookfor'])) { |
|
$term = $group['lookfor']; |
|
$op = $group['bool']; |
|
$tag = $type && isset(self::$search_tags[$type]) ? self::$search_tags[$type] : ''; |
|
|
|
// escape some characters from lookfor term |
|
$term = str_replace(array(',', ':', '(', ')'), array('\,', '\:', '\(', '\)'), $term); |
|
// replace multiple consecutive empty spaces with one empty space |
|
$term = preg_replace("/\s+/", ' ', $term); |
|
if (!empty($term)) { |
|
$query_str = implode(',', array($op, $tag)); |
|
$query_str = implode(($tag ? ':' : ''), array($query_str, $term)); |
|
$query["query-$counter"] = $query_str; |
|
$counter++; |
|
} |
|
} |
|
} |
|
|
|
// No search term, return an empty array |
|
} else { |
|
$results = array( |
|
'recordCount' => 0, |
|
'numFound' => 0, |
|
'start' => 0, |
|
'documents' => array(), |
|
'facets' => array() |
|
); |
|
return $results; |
|
} |
|
|
|
// Add filters |
|
$limiters = array(); $expanders = array(); $facets = array(); |
|
foreach ($filters as $filter) { |
|
if (preg_match('/addlimiter/', $filter)) { |
|
list($action, $str) = explode('(', $filter, 2); |
|
$field_and_value = substr($str, 0, -1); // e.g. FT:y or GZ:Student Research, Projects and Publications |
|
list($field, $value) = explode(':', $field_and_value, 2); |
|
$limiters[$field][] = $value; |
|
} else if (preg_match('/addexpander/', $filter)) { |
|
list($action, $str) = explode('(', $filter, 2); |
|
$field = substr($str, 0, -1); // expanders don't have value |
|
$expanders[] = $field; |
|
} else if (preg_match('/addfacetfilter/', $filter)) { |
|
list($action, $str) = explode('(', $filter, 2); |
|
$field_and_value = substr($str, 0, -1); // e.g. ZG:FRANCE |
|
list($field, $value) = explode(':', $field_and_value, 2); |
|
$facets[$field][] = $field_and_value; |
|
} |
|
} |
|
if (!empty($limiters)) { |
|
foreach($limiters as $field => $limiter) { |
|
$query['limiter'][] = $field . ':' . implode(',', $limiter); // e.g. LA99:English,French,German |
|
} |
|
} |
|
if (!empty($expanders)) { |
|
$query['expander'] = implode(',', $expanders); // e.g. fulltext, thesaurus |
|
} |
|
if (!empty($facets)) { |
|
$groupId = 1; |
|
foreach($facets as $field => $facet) { |
|
$query['facetfilter'][] = $groupId . ',' . implode(',', $facet); // e.g. 1,DE:Math,DE:History |
|
$groupId += 1; |
|
} |
|
} |
|
|
|
// Add the sort option |
|
$sortBy = in_array($sortBy, self::$sort_options) ? $sortBy : self::$mapped_sort_options[$sortBy]; |
|
|
|
// Add the HTTP query params |
|
$params = array( |
|
// Specifies the sort. Valid options are: |
|
// relevance, date, date2 |
|
// date = Date descending |
|
// date2 = Date ascending |
|
'sort' => $sortBy, |
|
// Specifies the search mode. Valid options are: |
|
// bool, any, all, smart |
|
'searchmode' => $mode, |
|
// Specifies the amount of data to return with the response |
|
// Valid options are: |
|
// title: Title only |
|
// brief: Title + Source, Subjects |
|
// detailed: Brief + full abstract |
|
'view' => $amount, |
|
/// Specifies whether or not to include facets |
|
'includefacets' => 'y', |
|
'resultsperpage' => $limit, |
|
'pagenumber' => $start, |
|
// Specifies whether or not to include highlighting in the search results |
|
'highlight' => 'y' |
|
); |
|
|
|
$params = array_merge($params, $query); |
|
|
|
$result = $this->request('Search', $params); |
|
return $result; |
|
} |
|
|
|
|
|
/** |
|
* Wrapper for retrieve API call |
|
* |
|
* @param array $an The accession number |
|
* @param string $start The short database name |
|
* |
|
* @throws object PEAR Error |
|
* @return array An associative array of data |
|
* @access public |
|
*/ |
|
public function apiRetrieve($an, $db) |
|
{ |
|
// Add the HTTP query params |
|
$params = array( |
|
'an' => $an, |
|
'dbid' => $db, |
|
'highlight' => 'y' |
|
); |
|
|
|
$result = $this->request('Retrieve', $params); |
|
return $result; |
|
} |
|
|
|
|
|
/** |
|
* Wrapper for info API call |
|
* |
|
* @throws object PEAR Error |
|
* @return array An associative array of data |
|
* @access public |
|
*/ |
|
public function apiInfo() |
|
{ |
|
if ($result = $this->readSession('info')) { |
|
return $result; |
|
} |
|
|
|
$result = $this->request('Info'); |
|
if(!$this->isError($result)) { |
|
$this->writeSession('info', $result); |
|
} |
|
return $result; |
|
} |
|
|
|
|
|
/** |
|
* Handle a PEAR_Error. Return : |
|
* - if the error is critical : an associative array with the current error message |
|
* - if the error is not critical : the error message |
|
* |
|
* @param Pear_Error $exception |
|
* |
|
* @return array or the Pear_Error exception |
|
* @access protected |
|
*/ |
|
private function handleError($error) { |
|
$errorCode = $error->getCode(); |
|
switch($errorCode) { |
|
// This kind of error was generated by user , so display it to user |
|
case EBSCOConnector::EDS_INVALID_ARGUMENT_VALUE: |
|
// Any other errors are system errors, don't display them to user |
|
default: |
|
$errorMessage = 'An error occurred when getting the data.'; |
|
break; |
|
} |
|
$result = array( |
|
'errors' => $errorMessage, |
|
'recordCount' => 0, |
|
'numFound' => 0, |
|
'start' => 0, |
|
'documents' => array(), |
|
'facets' => array() |
|
); |
|
return $result; |
|
} |
|
|
|
|
|
/** |
|
* Store the given object into session |
|
* |
|
* @param string $key The key used for reading the value |
|
* @param object $value The object stored in session |
|
* |
|
* @return none |
|
* @access protected |
|
*/ |
|
protected function writeSession($key, $value) |
|
{ |
|
if(!empty($key) && !empty($value)) { |
|
$_SESSION['EBSCO'][$key] = $value; |
|
} |
|
} |
|
|
|
|
|
/** |
|
* Read from session the object having the given key |
|
* |
|
* @param string $key The key used for reading the object |
|
* |
|
* @return object |
|
* @access protected |
|
*/ |
|
protected function readSession($key) |
|
{ |
|
$value = isset($_SESSION['EBSCO'][$key]) ? $_SESSION['EBSCO'][$key] : ''; |
|
return $value; |
|
} |
|
|
|
|
|
/** |
|
* Check if given object is an EBSCOException object |
|
* |
|
* @param object $object |
|
* |
|
* @return boolean |
|
* @access protected |
|
*/ |
|
protected function isError($object) |
|
{ |
|
return is_a($object, 'EBSCOException'); |
|
} |
|
|
|
} |
|
|
|
|
|
?>
|