326 lines
14 KiB
PHP
326 lines
14 KiB
PHP
<?php
|
|
/**
|
|
* Simple OAI-PMH 2.0 Data Provider
|
|
* Copyright (C) 2005 Heinrich Stamerjohanns <stamer@uni-oldenburg.de>
|
|
* Copyright (C) 2011 Jianfeng Li <jianfeng.li@adelaide.edu.au>
|
|
* Copyright (C) 2013 Daniel Neis Araujo <danielneis@gmail.com>
|
|
* Copyright (C) 2017 Sebastian Meyer <sebastian.meyer@opencultureconsulting.com>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
namespace OCC\OAI2;
|
|
|
|
/**
|
|
* This is an implementation of OAI Data Provider version 2.0.
|
|
* @see http://www.openarchives.org/OAI/2.0/openarchivesprotocol.htm
|
|
*/
|
|
class Server {
|
|
|
|
public $errors = [];
|
|
private $args = [];
|
|
private $verb = '';
|
|
private $max_records = 100;
|
|
private $token_prefix = '/tmp/oai2-';
|
|
private $token_valid = 86400;
|
|
private $uri = '';
|
|
private $identifyResponse;
|
|
private $listMetadataFormatsCallback;
|
|
private $listRecordsCallback;
|
|
private $getRecordCallback;
|
|
private Response $response;
|
|
|
|
public function __construct($uri, $args, $identifyResponse, $callbacks, $config) {
|
|
$this->uri = $uri;
|
|
$verbs = ['Identify', 'ListMetadataFormats', 'ListSets', 'ListIdentifiers', 'ListRecords', 'GetRecord'];
|
|
if (empty($args['verb']) || !in_array($args['verb'], $verbs)) {
|
|
$this->errors[] = new Exception('badVerb');
|
|
return;
|
|
}
|
|
$this->verb = $args['verb'];
|
|
unset($args['verb']);
|
|
$this->args = $args;
|
|
$this->identifyResponse = $identifyResponse;
|
|
$this->listMetadataFormatsCallback = $callbacks['ListMetadataFormats'];
|
|
$this->listSetsCallback = $callbacks['ListSets'];
|
|
$this->listRecordsCallback = $callbacks['ListRecords'];
|
|
$this->getRecordCallback = $callbacks['GetRecord'];
|
|
$this->max_records = $config['maxRecords'];
|
|
$this->token_prefix = $config['tokenPrefix'];
|
|
$this->token_valid = $config['tokenValid'];
|
|
$this->response = new Response($this->uri, $this->verb, $this->args);
|
|
call_user_func([$this, $this->verb]);
|
|
}
|
|
|
|
public function response() {
|
|
if (empty($this->errors)) {
|
|
return $this->response->doc;
|
|
}
|
|
$errorResponse = new Response($this->uri, $this->verb, $this->args);
|
|
$oai_node = $errorResponse->doc->documentElement;
|
|
foreach ($this->errors as $e) {
|
|
$node = $errorResponse->addChild($oai_node, 'error', $e->getMessage());
|
|
$node->setAttribute('code', $e->getOAI2Code());
|
|
}
|
|
return $errorResponse->doc;
|
|
}
|
|
|
|
public function Identify() {
|
|
if (count($this->args) > 0) {
|
|
foreach ($this->args as $key => $val) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
}
|
|
} else {
|
|
foreach ($this->identifyResponse as $key => $val) {
|
|
$this->response->addToVerbNode($key, $val);
|
|
}
|
|
}
|
|
}
|
|
|
|
public function ListMetadataFormats() {
|
|
$identifier = '';
|
|
foreach ($this->args as $argument => $value) {
|
|
if ($argument != 'identifier') {
|
|
$this->errors[] = new Exception('badArgument');
|
|
} else {
|
|
$identifier = $value;
|
|
}
|
|
}
|
|
if (empty($this->errors)) {
|
|
try {
|
|
if ($formats = call_user_func($this->listMetadataFormatsCallback, $identifier)) {
|
|
foreach ($formats as $key => $val) {
|
|
$cmf = $this->response->addToVerbNode('metadataFormat');
|
|
$this->response->addChild($cmf, 'metadataPrefix', $key);
|
|
$this->response->addChild($cmf, 'schema', $val['schema']);
|
|
$this->response->addChild($cmf, 'metadataNamespace', $val['namespace']);
|
|
}
|
|
} else {
|
|
$this->errors[] = new Exception('noMetadataFormats');
|
|
}
|
|
} catch (Exception $e) {
|
|
$this->errors[] = $e;
|
|
}
|
|
}
|
|
}
|
|
|
|
public function ListSets() {
|
|
if (isset($this->args['resumptionToken'])) {
|
|
if (count($this->args) > 1) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
} else {
|
|
$this->errors[] = new Exception('badResumptionToken');
|
|
}
|
|
} elseif (empty($this->errors)) {
|
|
try {
|
|
if ($sets = call_user_func($this->listSetsCallback, $identifier)) {
|
|
foreach ($sets as $key => $val) {
|
|
$cmf = $this->response->addToVerbNode('set');
|
|
$this->response->addChild($cmf, 'setSpec', $key);
|
|
$this->response->addChild($cmf, 'setName', $val['name']);
|
|
}
|
|
} else {
|
|
$this->errors[] = new Exception('noSetHierarchy#1');
|
|
}
|
|
} catch (Exception $e) {
|
|
$this->errors[] = $e;
|
|
}
|
|
} else {
|
|
$this->errors[] = new Exception('noSetHierarchy');
|
|
}
|
|
}
|
|
|
|
public function GetRecord() {
|
|
if (!isset($this->args['identifier']) || !isset($this->args['metadataPrefix'])) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
} else {
|
|
$metadataFormats = call_user_func($this->listMetadataFormatsCallback);
|
|
if (!isset($metadataFormats[$this->args['metadataPrefix']])) {
|
|
$this->errors[] = new Exception('cannotDisseminateFormat');
|
|
}
|
|
}
|
|
if (empty($this->errors)) {
|
|
try {
|
|
if ($record = call_user_func($this->getRecordCallback, $this->args['identifier'], $this->args['metadataPrefix'])) {
|
|
$cur_record = $this->response->addToVerbNode('record');
|
|
$this->response->createHeader($record['identifier'], $this->formatDatestamp($record['timestamp']), $record['deleted'], $cur_record);
|
|
if (!$record['deleted']) {
|
|
$this->addMetadata($cur_record, $record['metadata']);
|
|
}
|
|
} else {
|
|
$this->errors[] = new Exception('idDoesNotExist');
|
|
}
|
|
} catch (Exception $e) {
|
|
$this->errors[] = $e;
|
|
}
|
|
}
|
|
}
|
|
|
|
public function ListIdentifiers() {
|
|
$this->ListRecords();
|
|
}
|
|
|
|
public function ListRecords() {
|
|
$maxItems = $this->max_records;
|
|
$deliveredRecords = 0;
|
|
$metadataPrefix = isset($this->args['metadataPrefix']) ? $this->args['metadataPrefix'] : '';
|
|
$from = isset($this->args['from']) ? $this->args['from'] : '';
|
|
$until = isset($this->args['until']) ? $this->args['until'] : '';
|
|
$set = isset($this->args['set']) ? $this->args['set'] : null;
|
|
|
|
if (isset($this->args['resumptionToken'])) {
|
|
if (count($this->args) > 1) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
} else {
|
|
if (!file_exists($this->token_prefix.$this->args['resumptionToken'])) {
|
|
$this->errors[] = new Exception('badResumptionToken');
|
|
} else {
|
|
if (filemtime($this->token_prefix.$this->args['resumptionToken'])+$this->token_valid < time()) {
|
|
$this->errors[] = new Exception('badResumptionToken');
|
|
} else {
|
|
if ($readings = $this->readResumptionToken($this->token_prefix.$this->args['resumptionToken'])) {
|
|
list($deliveredRecords, $metadataPrefix, $from, $until) = $readings;
|
|
} else {
|
|
$this->errors[] = new Exception('badResumptionToken');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
if (!isset($this->args['metadataPrefix'])) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
} else {
|
|
$metadataFormats = call_user_func($this->listMetadataFormatsCallback);
|
|
if (!isset($metadataFormats[$this->args['metadataPrefix']])) {
|
|
$this->errors[] = new Exception('cannotDisseminateFormat');
|
|
}
|
|
}
|
|
if (isset($this->args['from'])) {
|
|
if (!$this->checkDateFormat($this->args['from'])) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
}
|
|
}
|
|
if (isset($this->args['until'])) {
|
|
if (!$this->checkDateFormat($this->args['until'])) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
}
|
|
}
|
|
if (isset($this->args['from']) && isset($this->args['until'])) {
|
|
if (strlen($this->args['from']) !== strlen($this->args['until'])) {
|
|
$this->errors[] = new Exception('badArgument');
|
|
}
|
|
}
|
|
if (isset($this->args['set'])) {
|
|
$ListSets = call_user_func($this->listSetsCallback);
|
|
if (!isset($ListSets[$this->args['set']])) {
|
|
$this->errors[] = new Exception('noSetHierarchy');
|
|
}
|
|
}
|
|
}
|
|
if (empty($this->errors)) {
|
|
try {
|
|
if (!($records_count = call_user_func($this->listRecordsCallback, $metadataPrefix, $this->formatTimestamp($from), $this->formatTimestamp($until), true, 0, 100, $set))) {
|
|
throw new Exception('noRecordsMatch');
|
|
}
|
|
$records = call_user_func($this->listRecordsCallback, $metadataPrefix, $this->formatTimestamp($from), $this->formatTimestamp($until), false, $deliveredRecords, $maxItems, $set);
|
|
foreach ($records as $record) {
|
|
$cur_record = null;
|
|
if ($this->verb == 'ListRecords') { // for ListIdentifiers, only headers will be returned.
|
|
$cur_record = $this->response->addToVerbNode('record');
|
|
}
|
|
$this->response->createHeader($record['identifier'], $this->formatDatestamp($record['timestamp']), $record['deleted'], $cur_record);
|
|
if (!$record['deleted'] && $this->verb == 'ListRecords') { // for ListIdentifiers, only headers will be returned.
|
|
$this->addMetadata($cur_record, $record['metadata']);
|
|
}
|
|
}
|
|
// Will we need a new ResumptionToken?
|
|
if ($records_count - $deliveredRecords > $maxItems) {
|
|
$deliveredRecords += $maxItems;
|
|
$restoken = $this->createResumptionToken($deliveredRecords, $metadataPrefix, $from, $until);
|
|
$expirationDatetime = date('Y-m-d\TH:i:s\Z', time()+$this->token_valid);
|
|
} elseif (isset($this->args['resumptionToken'])) {
|
|
// Last delivery, return empty resumptionToken
|
|
$restoken = null;
|
|
$expirationDatetime = null;
|
|
}
|
|
if (isset($restoken)) {
|
|
$this->response->createResumptionToken($restoken, $expirationDatetime, $records_count, $deliveredRecords-$maxItems);
|
|
}
|
|
} catch (Exception $e) {
|
|
$this->errors[] = $e;
|
|
}
|
|
}
|
|
}
|
|
|
|
private function addMetadata($cur_record, $file) {
|
|
$meta_node = $this->response->addChild($cur_record, 'metadata');
|
|
$fragment = new \DOMDocument();
|
|
$fragment->load($file);
|
|
$this->response->importFragment($meta_node, $fragment);
|
|
}
|
|
|
|
private function createResumptionToken($deliveredRecords, $metadataPrefix, $from, $until) {
|
|
list($usec, $sec) = explode(' ', microtime());
|
|
$token = ((int)($usec*1000) + (int)($sec*1000)).'_'.$metadataPrefix;
|
|
$file = fopen($this->token_prefix.$token, 'w');
|
|
if ($file == false) {
|
|
exit('Cannot write resumption token. Writing permission needs to be changed.');
|
|
}
|
|
fputs($file, $deliveredRecords.'#');
|
|
fputs($file, $metadataPrefix.'#');
|
|
fputs($file, $from.'#');
|
|
fputs($file, $until);
|
|
fclose($file);
|
|
return $token;
|
|
}
|
|
|
|
private function readResumptionToken($resumptionToken) {
|
|
$rtVal = false;
|
|
$file = fopen($resumptionToken, 'r');
|
|
if ($file != false) {
|
|
$filetext = fgets($file, 255);
|
|
$textparts = explode('#', $filetext);
|
|
fclose($file);
|
|
unlink($resumptionToken);
|
|
$rtVal = array_values($textparts);
|
|
}
|
|
return $rtVal;
|
|
}
|
|
|
|
private function formatDatestamp($timestamp) {
|
|
return gmdate('Y-m-d\TH:i:s\Z', $timestamp);
|
|
}
|
|
|
|
private function formatTimestamp($datestamp) {
|
|
$time = date_parse_from_format('Y-m-d\TH:i:s\Z', $datestamp);
|
|
if ($time['error_count'] > 0) {
|
|
$time = date_parse_from_format('Y-m-d', $datestamp);
|
|
}
|
|
if ($time['error_count'] > 0) {
|
|
return null;
|
|
} else {
|
|
return gmmktime($time['hour'], $time['minute'], $time['second'], $time['month'] + 1, $time['day'], $time['year']);
|
|
}
|
|
}
|
|
|
|
private function checkDateFormat($date) {
|
|
$datetime = \DateTime::createFromFormat('Y-m-d\TH:i:s\Z', $date);
|
|
if ($datetime === false) {
|
|
$datetime = \DateTime::createFromFormat('Y-m-d', $date);
|
|
}
|
|
return ($datetime !== false);
|
|
}
|
|
|
|
}
|