Clean up
This commit is contained in:
parent
6120d7990f
commit
588aa878d1
|
@ -23,32 +23,32 @@
|
||||||
class OAI2Exception extends Exception {
|
class OAI2Exception extends Exception {
|
||||||
|
|
||||||
public function __construct($code) {
|
public function __construct($code) {
|
||||||
$this->errorTable = array(
|
$this->errorTable = [
|
||||||
'badArgument' => array(
|
'badArgument' => [
|
||||||
'text' => 'The request includes illegal arguments, is missing required arguments, includes a repeated argument, or values for arguments have an illegal syntax.',
|
'text' => 'The request includes illegal arguments, is missing required arguments, includes a repeated argument, or values for arguments have an illegal syntax.',
|
||||||
),
|
],
|
||||||
'badResumptionToken' => array(
|
'badResumptionToken' => [
|
||||||
'text' => 'The value of the resumptionToken argument is invalid or expired.',
|
'text' => 'The value of the resumptionToken argument is invalid or expired.',
|
||||||
),
|
],
|
||||||
'badVerb' => array(
|
'badVerb' => [
|
||||||
'text' => 'Value of the verb argument is not a legal OAI-PMH verb, the verb argument is missing, or the verb argument is repeated.',
|
'text' => 'Value of the verb argument is not a legal OAI-PMH verb, the verb argument is missing, or the verb argument is repeated.',
|
||||||
),
|
],
|
||||||
'cannotDisseminateFormat' => array(
|
'cannotDisseminateFormat' => [
|
||||||
'text' => 'The metadata format identified by the value given for the metadataPrefix argument is not supported by the item or by the repository.',
|
'text' => 'The metadata format identified by the value given for the metadataPrefix argument is not supported by the item or by the repository.',
|
||||||
),
|
],
|
||||||
'idDoesNotExist' => array(
|
'idDoesNotExist' => [
|
||||||
'text' => 'The value of the identifier argument is unknown or illegal in this repository.',
|
'text' => 'The value of the identifier argument is unknown or illegal in this repository.',
|
||||||
),
|
],
|
||||||
'noRecordsMatch' => array(
|
'noRecordsMatch' => [
|
||||||
'text' => 'The combination of the values of the from, until, set and metadataPrefix arguments results in an empty list.',
|
'text' => 'The combination of the values of the from, until, set and metadataPrefix arguments results in an empty list.',
|
||||||
),
|
],
|
||||||
'noMetadataFormats' => array(
|
'noMetadataFormats' => [
|
||||||
'text' => 'There are no metadata formats available for the specified item.',
|
'text' => 'There are no metadata formats available for the specified item.',
|
||||||
),
|
],
|
||||||
'noSetHierarchy' => array(
|
'noSetHierarchy' => [
|
||||||
'text' => 'The repository does not support sets.',
|
'text' => 'The repository does not support sets.',
|
||||||
),
|
],
|
||||||
);
|
];
|
||||||
parent::__construct($this->errorTable[$code]['text']);
|
parent::__construct($this->errorTable[$code]['text']);
|
||||||
$this->code = $code;
|
$this->code = $code;
|
||||||
}
|
}
|
|
@ -0,0 +1,137 @@
|
||||||
|
<?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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class OAI2Response {
|
||||||
|
|
||||||
|
public $doc; // DOMDocument. Handle of current XML Document object
|
||||||
|
|
||||||
|
public function __construct($uri, $verb, $request_args) {
|
||||||
|
if (substr($uri, -1, 1) == '/') {
|
||||||
|
$stylesheet = $uri.'Resources/Stylesheet.xsl';
|
||||||
|
} else {
|
||||||
|
$stylesheet = (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') ? 'https://' : 'http://';
|
||||||
|
$stylesheet .= $_SERVER['HTTP_HOST'].pathinfo(parse_url($_SERVER['REQUEST_URI'], PHP_URL_PATH), PATHINFO_DIRNAME).'/Resources/Stylesheet.xsl';
|
||||||
|
}
|
||||||
|
$this->verb = $verb;
|
||||||
|
$this->doc = new DOMDocument('1.0', 'UTF-8');
|
||||||
|
$this->doc->appendChild($this->doc->createProcessingInstruction('xml-stylesheet', 'type="text/xsl" href="'.$stylesheet.'"'));
|
||||||
|
$oai_node = $this->doc->createElement('OAI-PMH');
|
||||||
|
$oai_node->setAttribute('xmlns', 'http://www.openarchives.org/OAI/2.0/');
|
||||||
|
$oai_node->setAttribute('xmlns:xsi', 'http://www.w3.org/2001/XMLSchema-instance');
|
||||||
|
$oai_node->setAttribute('xsi:schemaLocation', 'http://www.openarchives.org/OAI/2.0/ http://www.openarchives.org/OAI/2.0/OAI-PMH.xsd');
|
||||||
|
$this->addChild($oai_node, 'responseDate', gmdate('Y-m-d\TH:i:s\Z'));
|
||||||
|
$this->doc->appendChild($oai_node);
|
||||||
|
$request = $this->addChild($this->doc->documentElement, 'request', $uri);
|
||||||
|
if (!empty($this->verb)) {
|
||||||
|
$request->setAttribute('verb', $this->verb);
|
||||||
|
}
|
||||||
|
foreach($request_args as $key => $value) {
|
||||||
|
$request->setAttribute($key, $value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a child node to a parent node on a XML Doc: a worker function.
|
||||||
|
*
|
||||||
|
* @param DOMNode $mom_node The target node.
|
||||||
|
* @param string $name The name of child node is being added.
|
||||||
|
* @param string $value Text for the adding node if it is a text node.
|
||||||
|
*
|
||||||
|
* @return DOMElement $added_node * The newly created node
|
||||||
|
*/
|
||||||
|
public function addChild($mom_node, $name, $value = '') {
|
||||||
|
$added_node = $this->doc->createElement($name, $value);
|
||||||
|
$added_node = $mom_node->appendChild($added_node);
|
||||||
|
return $added_node;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add direct child nodes to verb node (OAI-PMH), e.g. response to ListMetadataFormats.
|
||||||
|
* Different verbs can have different required child nodes.
|
||||||
|
* @see createHeader, importFragment
|
||||||
|
*
|
||||||
|
* @param string $nodeName The name of appending node.
|
||||||
|
* @param string $value The content of appending node.
|
||||||
|
*/
|
||||||
|
public function addToVerbNode($nodeName, $value = null) {
|
||||||
|
if (!isset($this->verbNode) && !empty($this->verb)) {
|
||||||
|
$this->verbNode = $this->addChild($this->doc->documentElement, $this->verb);
|
||||||
|
}
|
||||||
|
return $this->addChild($this->verbNode, $nodeName, $value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Headers are enclosed inside of <record> to the query of ListRecords, ListIdentifiers and etc.
|
||||||
|
*
|
||||||
|
* @param string $identifier The identifier string for node <identifier>.
|
||||||
|
* @param string $timestamp Timestamp in UTC format for node <datastamp>.
|
||||||
|
* @param boolean $deleted Deleted status for the record.
|
||||||
|
* @param DOMElement $add_to_node Default value is null.
|
||||||
|
* In normal cases, $add_to_node is the <record> node created previously.
|
||||||
|
* When it is null, the newly created header node is attatched to $this->verbNode.
|
||||||
|
* Otherwise it will be attached to the desired node defined in $add_to_node.
|
||||||
|
*/
|
||||||
|
public function createHeader($identifier, $timestamp, $deleted = false, $add_to_node = null) {
|
||||||
|
if(is_null($add_to_node)) {
|
||||||
|
$header_node = $this->addToVerbNode('header');
|
||||||
|
} else {
|
||||||
|
$header_node = $this->addChild($add_to_node, 'header');
|
||||||
|
}
|
||||||
|
$this->addChild($header_node, 'identifier', $identifier);
|
||||||
|
$this->addChild($header_node, 'datestamp', $timestamp);
|
||||||
|
if($deleted) {
|
||||||
|
$header_node->setAttribute('status', 'deleted');
|
||||||
|
}
|
||||||
|
return $header_node;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If there are too many records request could not finished a resumpToken is generated to let harvester know
|
||||||
|
*
|
||||||
|
* @param string $token A random number created somewhere?
|
||||||
|
* @param string $expirationdatetime A string representing time.
|
||||||
|
* @param integer $num_rows Number of records retrieved.
|
||||||
|
* @param string $cursor Cursor can be used for database to retrieve next time.
|
||||||
|
*/
|
||||||
|
public function createResumptionToken($token, $expirationdatetime, $num_rows, $cursor = null) {
|
||||||
|
$resump_node = $this->addChild($this->verbNode, 'resumptionToken', $token);
|
||||||
|
if(isset($expirationdatetime)) {
|
||||||
|
$resump_node->setAttribute('expirationDate', $expirationdatetime);
|
||||||
|
}
|
||||||
|
$resump_node->setAttribute('completeListSize', $num_rows);
|
||||||
|
$resump_node->setAttribute('cursor', $cursor);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Imports a XML fragment into a parent node on a XML Doc: a worker function.
|
||||||
|
*
|
||||||
|
* @param DOMNode $mom_node The target node.
|
||||||
|
* @param DOMDocument $fragment The XML fragment is being added.
|
||||||
|
*
|
||||||
|
* @return DOMElement $added_node * The newly created node
|
||||||
|
*/
|
||||||
|
public function importFragment($mom_node, $fragment) {
|
||||||
|
$added_node = $mom_node->appendChild($this->doc->importNode($fragment->documentElement, true));
|
||||||
|
return $added_node;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -20,8 +20,8 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
require_once('oai2exception.php');
|
require_once './OAI2Exception.php';
|
||||||
require_once('oai2xml.php');
|
require_once './OAI2Response.php';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is an implementation of OAI Data Provider version 2.0.
|
* This is an implementation of OAI Data Provider version 2.0.
|
||||||
|
@ -29,17 +29,16 @@ require_once('oai2xml.php');
|
||||||
*/
|
*/
|
||||||
class OAI2Server {
|
class OAI2Server {
|
||||||
|
|
||||||
public $errors = array();
|
public $errors = [];
|
||||||
private $args = array();
|
private $args = [];
|
||||||
private $verb = '';
|
private $verb = '';
|
||||||
private $deleted_record = 'transient';
|
|
||||||
private $max_records = 100;
|
private $max_records = 100;
|
||||||
private $token_prefix = '/tmp/oai2-';
|
private $token_prefix = '/tmp/oai2-';
|
||||||
private $token_valid = 86400;
|
private $token_valid = 86400;
|
||||||
|
|
||||||
public function __construct($uri, $args, $identifyResponse, $callbacks, $config) {
|
public function __construct($uri, $args, $identifyResponse, $callbacks, $config) {
|
||||||
$this->uri = $uri;
|
$this->uri = $uri;
|
||||||
$verbs = array('Identify', 'ListMetadataFormats', 'ListSets', 'ListIdentifiers', 'ListRecords', 'GetRecord');
|
$verbs = ['Identify', 'ListMetadataFormats', 'ListSets', 'ListIdentifiers', 'ListRecords', 'GetRecord'];
|
||||||
if (empty($args['verb']) || !in_array($args['verb'], $verbs)) {
|
if (empty($args['verb']) || !in_array($args['verb'], $verbs)) {
|
||||||
$this->errors[] = new OAI2Exception('badVerb');
|
$this->errors[] = new OAI2Exception('badVerb');
|
||||||
return;
|
return;
|
||||||
|
@ -51,19 +50,18 @@ class OAI2Server {
|
||||||
$this->listMetadataFormatsCallback = $callbacks['ListMetadataFormats'];
|
$this->listMetadataFormatsCallback = $callbacks['ListMetadataFormats'];
|
||||||
$this->listRecordsCallback = $callbacks['ListRecords'];
|
$this->listRecordsCallback = $callbacks['ListRecords'];
|
||||||
$this->getRecordCallback = $callbacks['GetRecord'];
|
$this->getRecordCallback = $callbacks['GetRecord'];
|
||||||
$this->deleted_record = $config['deletedRecord'];
|
|
||||||
$this->max_records = $config['maxRecords'];
|
$this->max_records = $config['maxRecords'];
|
||||||
$this->token_prefix = $config['tokenPrefix'];
|
$this->token_prefix = $config['tokenPrefix'];
|
||||||
$this->token_valid = $config['tokenValid'];
|
$this->token_valid = $config['tokenValid'];
|
||||||
$this->response = new OAI2XMLResponse($this->uri, $this->verb, $this->args);
|
$this->response = new OAI2Response($this->uri, $this->verb, $this->args);
|
||||||
call_user_func(array($this, $this->verb));
|
call_user_func([$this, $this->verb]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function response() {
|
public function response() {
|
||||||
if (empty($this->errors)) {
|
if (empty($this->errors)) {
|
||||||
return $this->response->doc;
|
return $this->response->doc;
|
||||||
}
|
}
|
||||||
$errorResponse = new OAI2XMLResponse($this->uri, $this->verb, $this->args);
|
$errorResponse = new OAI2Response($this->uri, $this->verb, $this->args);
|
||||||
$oai_node = $errorResponse->doc->documentElement;
|
$oai_node = $errorResponse->doc->documentElement;
|
||||||
foreach($this->errors as $e) {
|
foreach($this->errors as $e) {
|
||||||
$node = $errorResponse->addChild($oai_node, 'error', $e->getMessage());
|
$node = $errorResponse->addChild($oai_node, 'error', $e->getMessage());
|
||||||
|
@ -222,7 +220,7 @@ class OAI2Server {
|
||||||
$restoken = $this->createResumptionToken($deliveredRecords, $metadataPrefix, $from, $until);
|
$restoken = $this->createResumptionToken($deliveredRecords, $metadataPrefix, $from, $until);
|
||||||
$expirationDatetime = gmstrftime('%Y-%m-%dT%TZ', time()+$this->token_valid);
|
$expirationDatetime = gmstrftime('%Y-%m-%dT%TZ', time()+$this->token_valid);
|
||||||
} elseif (isset($this->args['resumptionToken'])) {
|
} elseif (isset($this->args['resumptionToken'])) {
|
||||||
// Last delivery, return empty ResumptionToken
|
// Last delivery, return empty resumptionToken
|
||||||
$restoken = null;
|
$restoken = null;
|
||||||
$expirationDatetime = null;
|
$expirationDatetime = null;
|
||||||
}
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
<?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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This file contains all configuration you need to change according to your preferences
|
||||||
|
* @see http://www.openarchives.org/OAI/2.0/openarchivesprotocol.htm for further explanation
|
||||||
|
*/
|
||||||
|
|
||||||
|
$config = [
|
||||||
|
|
||||||
|
// A human readable name for the repository
|
||||||
|
'repositoryName' => 'Simple OAI 2.0 Data Provider',
|
||||||
|
|
||||||
|
// Email address for contacting the repository owner
|
||||||
|
'adminEmail' => 'admin@example.org',
|
||||||
|
|
||||||
|
// Do you provide 0-byte files for deleted records?
|
||||||
|
//
|
||||||
|
// Possible values:
|
||||||
|
// "no" -> the repository does not maintain information about deletions
|
||||||
|
// "transient" -> the repository maintains information about deletions, but
|
||||||
|
// does not guarantee them to be persistent (default)
|
||||||
|
// "persistent" -> the repository maintains information about deletions with
|
||||||
|
// no time limit
|
||||||
|
'deletedRecord' => 'transient',
|
||||||
|
|
||||||
|
// Metadata formats, schemas and namespaces of your records
|
||||||
|
//
|
||||||
|
// The default is 'oai_dc' which is also required by the OAI-PMH specification,
|
||||||
|
// but technically you can deliver any XML based data format you want. Just add
|
||||||
|
// another entry with the 'metadataPrefix' as key and schema/namespace URIs as
|
||||||
|
// array values or replace the default 'oai_dc' entry (not recommended).
|
||||||
|
'metadataPrefix' => [
|
||||||
|
'oai_dc' => [
|
||||||
|
'schema' => 'http://www.openarchives.org/OAI/2.0/oai_dc.xsd',
|
||||||
|
'namespace' => 'http://www.openarchives.org/OAI/2.0/oai_dc/',
|
||||||
|
],
|
||||||
|
],
|
||||||
|
|
||||||
|
// Directory containing the records
|
||||||
|
//
|
||||||
|
// Make sure the given path is readable and there is a subdirectory for every
|
||||||
|
// 'metadataPrefix' you specified above.
|
||||||
|
'dataDirectory' => './Data/',
|
||||||
|
|
||||||
|
// Maximum number of records to return before giving a resumption token
|
||||||
|
'maxRecords' => 100,
|
||||||
|
|
||||||
|
// Absolute path and filename prefix for saving resumption tokens
|
||||||
|
//
|
||||||
|
// Make sure the given path is writable.
|
||||||
|
'tokenPrefix' => '/tmp/oai2-',
|
||||||
|
|
||||||
|
// Number of seconds a resumption token should be valid
|
||||||
|
'tokenValid' => 86400, // 24 hours
|
||||||
|
|
||||||
|
];
|
|
@ -12,7 +12,7 @@ A demo installation can be found [here](https://demo.opencultureconsulting.com/o
|
||||||
|
|
||||||
1. Deploy all the files to a webserver.
|
1. Deploy all the files to a webserver.
|
||||||
|
|
||||||
2. Edit `config.php` and adjust the settings according to your preferences.
|
2. Edit `Configuration/Main.php` and adjust the settings according to your preferences.
|
||||||
|
|
||||||
3. Create a subdirectory inside the specified data directory for every format (i. e. `metadataPrefix`) you want to provide.
|
3. Create a subdirectory inside the specified data directory for every format (i. e. `metadataPrefix`) you want to provide.
|
||||||
|
|
||||||
|
|
|
@ -164,7 +164,7 @@
|
||||||
<xsl:apply-templates select="/oai:OAI-PMH"/>
|
<xsl:apply-templates select="/oai:OAI-PMH"/>
|
||||||
<xsl:call-template name="quicklinks"/>
|
<xsl:call-template name="quicklinks"/>
|
||||||
<p class="info">You are viewing an HTML version of the XML OAI-PMH response. To see the underlying XML as it appears to any OAI-PMH harvester use your web browser's <em>view source</em> option or disable XSLT processing.</p>
|
<p class="info">You are viewing an HTML version of the XML OAI-PMH response. To see the underlying XML as it appears to any OAI-PMH harvester use your web browser's <em>view source</em> option or disable XSLT processing.</p>
|
||||||
<p class="info">This XSL script was originally written by Christopher Gutteridge at <a href="https://www.southampton.ac.uk/">University of Southampton</a> for the <a href="http://www.eprints.org/">EPrints</a> project and was later adapted by Sebastian Meyer at <a href="http://www.opencultureconsulting.com/">Open Culture Consulting</a> to be more generally applicable to other OAI-PMH interfaces. It is available on <a href="https://github.com/opencultureconsulting/oai_pmh">GitHub</a> for free!</p>
|
<p class="info">This XSL script was originally written by Christopher Gutteridge at <a href="https://www.southampton.ac.uk/">University of Southampton</a> for the <a href="https://www.eprints.org/">EPrints</a> project and was later adapted by Sebastian Meyer at <a href="https://www.opencultureconsulting.com/">Open Culture Consulting</a> to be more generally applicable to other OAI-PMH interfaces. It is available on <a href="https://github.com/opencultureconsulting/oai_pmh">GitHub</a> for free!</p>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
</xsl:template>
|
</xsl:template>
|
74
config.php
74
config.php
|
@ -1,74 +0,0 @@
|
||||||
<?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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This file contains all configuration you need to change according to your preferences
|
|
||||||
* @see http://www.openarchives.org/OAI/2.0/openarchivesprotocol.htm for further explanation
|
|
||||||
*/
|
|
||||||
|
|
||||||
$config = array();
|
|
||||||
|
|
||||||
// A human readable name for the repository
|
|
||||||
$config['repositoryName'] = 'Simple OAI 2.0 Data Provider';
|
|
||||||
|
|
||||||
// Email address for contacting the repository owner
|
|
||||||
$config['adminEmail'] = 'admin@example.org';
|
|
||||||
|
|
||||||
// Do you provide 0-byte files for deleted records?
|
|
||||||
//
|
|
||||||
// Possible values:
|
|
||||||
// "no" -> the repository does not maintain information about deletions
|
|
||||||
// "transient" -> the repository maintains information about deletions, but
|
|
||||||
// does not guarantee them to be persistent (default)
|
|
||||||
// "persistent" -> the repository maintains information about deletions with
|
|
||||||
// no time limit
|
|
||||||
$config['deletedRecord'] = 'transient';
|
|
||||||
|
|
||||||
// Metadata formats, schemas and namespaces of your records
|
|
||||||
//
|
|
||||||
// The default is 'oai_dc' which is also required by the OAI-PMH specification,
|
|
||||||
// but technically you can deliver any XML based data format you want. Just add
|
|
||||||
// another entry with the 'metadataPrefix' as key and schema/namespace URIs as
|
|
||||||
// array values or replace the default 'oai_dc' entry (not recommended).
|
|
||||||
$config['metadataPrefix'] = array(
|
|
||||||
'oai_dc' => array(
|
|
||||||
'schema' => 'http://www.openarchives.org/OAI/2.0/oai_dc.xsd',
|
|
||||||
'namespace' => 'http://www.openarchives.org/OAI/2.0/oai_dc/',
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
// Directory containing the records
|
|
||||||
//
|
|
||||||
// Make sure the given path is readable and there is a subdirectory for every
|
|
||||||
// 'metadataPrefix' you specified above.
|
|
||||||
$config['dataDirectory'] = 'data/';
|
|
||||||
|
|
||||||
// Maximum number of records to return before giving a resumption token
|
|
||||||
$config['maxRecords'] = 100;
|
|
||||||
|
|
||||||
// Path and prefix for saving resumption tokens
|
|
||||||
//
|
|
||||||
// Make sure the given path is writable.
|
|
||||||
$config['tokenPrefix'] = '/tmp/oai2-';
|
|
||||||
|
|
||||||
// Number of seconds a resumption token should be valid
|
|
||||||
$config['tokenValid'] = 86400; // 24 hours
|
|
51
index.php
51
index.php
|
@ -20,18 +20,18 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
require_once('config.php');
|
require_once './Configuration/Main.php';
|
||||||
require_once('oai2server.php');
|
require_once './Classes/OAI2Server.php';
|
||||||
|
|
||||||
// Get all available records and their respective status and timestamps
|
// Get all available records and their respective status and timestamps
|
||||||
$records = array();
|
$records = [];
|
||||||
$deleted = array();
|
$deleted = [];
|
||||||
$timestamps = array();
|
$timestamps = [];
|
||||||
$earliest = time();
|
$earliest = time();
|
||||||
|
|
||||||
foreach($config['metadataPrefix'] as $prefix => $uris) {
|
foreach ($config['metadataPrefix'] as $prefix => $uris) {
|
||||||
$files = glob(rtrim($config['dataDirectory'], '/').'/'.$prefix.'/*.xml');
|
$files = glob(rtrim($config['dataDirectory'], '/').'/'.$prefix.'/*.xml');
|
||||||
foreach($files as $file) {
|
foreach ($files as $file) {
|
||||||
$records[$prefix][pathinfo($file, PATHINFO_FILENAME)] = $file;
|
$records[$prefix][pathinfo($file, PATHINFO_FILENAME)] = $file;
|
||||||
$deleted[$prefix][pathinfo($file, PATHINFO_FILENAME)] = !filesize($file);
|
$deleted[$prefix][pathinfo($file, PATHINFO_FILENAME)] = !filesize($file);
|
||||||
$timestamps[$prefix][filemtime($file)][] = pathinfo($file, PATHINFO_FILENAME);
|
$timestamps[$prefix][filemtime($file)][] = pathinfo($file, PATHINFO_FILENAME);
|
||||||
|
@ -54,7 +54,7 @@ if (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build the Identify response
|
// Build the Identify response
|
||||||
$identifyResponse = array(
|
$identifyResponse = [
|
||||||
'repositoryName' => $config['repositoryName'],
|
'repositoryName' => $config['repositoryName'],
|
||||||
'baseURL' => $baseURL,
|
'baseURL' => $baseURL,
|
||||||
'protocolVersion' => '2.0',
|
'protocolVersion' => '2.0',
|
||||||
|
@ -62,40 +62,38 @@ $identifyResponse = array(
|
||||||
'earliestDatestamp' => gmdate('Y-m-d\TH:i:s\Z', $earliest),
|
'earliestDatestamp' => gmdate('Y-m-d\TH:i:s\Z', $earliest),
|
||||||
'deletedRecord' => $config['deletedRecord'],
|
'deletedRecord' => $config['deletedRecord'],
|
||||||
'granularity' => 'YYYY-MM-DDThh:mm:ssZ'
|
'granularity' => 'YYYY-MM-DDThh:mm:ssZ'
|
||||||
);
|
];
|
||||||
|
|
||||||
$oai2 = new OAI2Server(
|
$oai2 = new OAI2Server(
|
||||||
$baseURL,
|
$baseURL,
|
||||||
$_GET,
|
$_GET,
|
||||||
$identifyResponse,
|
$identifyResponse,
|
||||||
array(
|
[
|
||||||
'GetRecord' =>
|
'GetRecord' => function ($identifier, $metadataPrefix) {
|
||||||
function($identifier, $metadataPrefix) {
|
|
||||||
global $records, $deleted;
|
global $records, $deleted;
|
||||||
if (empty($records[$metadataPrefix][$identifier])) {
|
if (empty($records[$metadataPrefix][$identifier])) {
|
||||||
return array();
|
return [];
|
||||||
} else {
|
} else {
|
||||||
return array(
|
return [
|
||||||
'identifier' => $identifier,
|
'identifier' => $identifier,
|
||||||
'timestamp' => filemtime($records[$metadataPrefix][$identifier]),
|
'timestamp' => filemtime($records[$metadataPrefix][$identifier]),
|
||||||
'deleted' => $deleted[$metadataPrefix][$identifier],
|
'deleted' => $deleted[$metadataPrefix][$identifier],
|
||||||
'metadata' => $records[$metadataPrefix][$identifier]
|
'metadata' => $records[$metadataPrefix][$identifier]
|
||||||
);
|
];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'ListRecords' =>
|
'ListRecords' => function ($metadataPrefix, $from = null, $until = null, $count = false, $deliveredRecords = 0, $maxItems = 100) {
|
||||||
function($metadataPrefix, $from = null, $until = null, $count = false, $deliveredRecords = 0, $maxItems = 100) {
|
|
||||||
global $records, $deleted, $timestamps;
|
global $records, $deleted, $timestamps;
|
||||||
$resultSet = array();
|
$resultSet = [];
|
||||||
foreach($timestamps[$metadataPrefix] as $timestamp => $identifiers) {
|
foreach ($timestamps[$metadataPrefix] as $timestamp => $identifiers) {
|
||||||
if ((is_null($from) || $timestamp >= $from) && (is_null($until) || $timestamp <= $until)) {
|
if ((is_null($from) || $timestamp >= $from) && (is_null($until) || $timestamp <= $until)) {
|
||||||
foreach($identifiers as $identifier) {
|
foreach ($identifiers as $identifier) {
|
||||||
$resultSet[] = array(
|
$resultSet[] = [
|
||||||
'identifier' => $identifier,
|
'identifier' => $identifier,
|
||||||
'timestamp' => filemtime($records[$metadataPrefix][$identifier]),
|
'timestamp' => filemtime($records[$metadataPrefix][$identifier]),
|
||||||
'deleted' => $deleted[$metadataPrefix][$identifier],
|
'deleted' => $deleted[$metadataPrefix][$identifier],
|
||||||
'metadata' => $records[$metadataPrefix][$identifier]
|
'metadata' => $records[$metadataPrefix][$identifier]
|
||||||
);
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -105,12 +103,11 @@ $oai2 = new OAI2Server(
|
||||||
return array_slice($resultSet, $deliveredRecords, $maxItems);
|
return array_slice($resultSet, $deliveredRecords, $maxItems);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'ListMetadataFormats' =>
|
'ListMetadataFormats' => function ($identifier = '') {
|
||||||
function($identifier = '') {
|
|
||||||
global $config, $records;
|
global $config, $records;
|
||||||
if (!empty($identifier)) {
|
if (!empty($identifier)) {
|
||||||
$formats = array();
|
$formats = [];
|
||||||
foreach($records as $format => $record) {
|
foreach ($records as $format => $record) {
|
||||||
if (!empty($record[$identifier])) {
|
if (!empty($record[$identifier])) {
|
||||||
$formats[$format] = $config['metadataPrefix'][$format];
|
$formats[$format] = $config['metadataPrefix'][$format];
|
||||||
}
|
}
|
||||||
|
@ -124,7 +121,7 @@ $oai2 = new OAI2Server(
|
||||||
return $config['metadataPrefix'];
|
return $config['metadataPrefix'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
),
|
],
|
||||||
$config
|
$config
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
137
oai2xml.php
137
oai2xml.php
|
@ -1,137 +0,0 @@
|
||||||
<?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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class OAI2XMLResponse {
|
|
||||||
|
|
||||||
public $doc; // DOMDocument. Handle of current XML Document object
|
|
||||||
|
|
||||||
public function __construct($uri, $verb, $request_args) {
|
|
||||||
if (substr($uri, -1, 1) == '/') {
|
|
||||||
$stylesheet = $uri.'oai2transform.xsl';
|
|
||||||
} else {
|
|
||||||
$stylesheet = (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') ? 'https://' : 'http://';
|
|
||||||
$stylesheet .= $_SERVER['HTTP_HOST'].pathinfo(parse_url($_SERVER['REQUEST_URI'], PHP_URL_PATH), PATHINFO_DIRNAME).'/oai2transform.xsl';
|
|
||||||
}
|
|
||||||
$this->verb = $verb;
|
|
||||||
$this->doc = new DOMDocument('1.0', 'UTF-8');
|
|
||||||
$this->doc->appendChild($this->doc->createProcessingInstruction('xml-stylesheet', 'type="text/xsl" href="'.$stylesheet.'"'));
|
|
||||||
$oai_node = $this->doc->createElement('OAI-PMH');
|
|
||||||
$oai_node->setAttribute('xmlns', 'http://www.openarchives.org/OAI/2.0/');
|
|
||||||
$oai_node->setAttribute('xmlns:xsi', 'http://www.w3.org/2001/XMLSchema-instance');
|
|
||||||
$oai_node->setAttribute('xsi:schemaLocation', 'http://www.openarchives.org/OAI/2.0/ http://www.openarchives.org/OAI/2.0/OAI-PMH.xsd');
|
|
||||||
$this->addChild($oai_node, 'responseDate', gmdate('Y-m-d\TH:i:s\Z'));
|
|
||||||
$this->doc->appendChild($oai_node);
|
|
||||||
$request = $this->addChild($this->doc->documentElement, 'request', $uri);
|
|
||||||
if (!empty($this->verb)) {
|
|
||||||
$request->setAttribute('verb', $this->verb);
|
|
||||||
}
|
|
||||||
foreach($request_args as $key => $value) {
|
|
||||||
$request->setAttribute($key, $value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add a child node to a parent node on a XML Doc: a worker function.
|
|
||||||
*
|
|
||||||
* @param $mom_node Type: DOMNode. The target node.
|
|
||||||
* @param $name Type: string. The name of child node is being added
|
|
||||||
* @param $value Type: string. Text for the adding node if it is a text node.
|
|
||||||
*
|
|
||||||
* @return DOMElement $added_node * The newly created node
|
|
||||||
*/
|
|
||||||
function addChild($mom_node, $name, $value = '') {
|
|
||||||
$added_node = $this->doc->createElement($name, $value);
|
|
||||||
$added_node = $mom_node->appendChild($added_node);
|
|
||||||
return $added_node;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add direct child nodes to verb node (OAI-PMH), e.g. response to ListMetadataFormats.
|
|
||||||
* Different verbs can have different required child nodes.
|
|
||||||
* @see createHeader, importFragment
|
|
||||||
*
|
|
||||||
* @param $nodeName Type: string. The name of appending node.
|
|
||||||
* @param $value Type: string. The content of appending node.
|
|
||||||
*/
|
|
||||||
function addToVerbNode($nodeName, $value = null) {
|
|
||||||
if (!isset($this->verbNode) && !empty($this->verb)) {
|
|
||||||
$this->verbNode = $this->addChild($this->doc->documentElement, $this->verb);
|
|
||||||
}
|
|
||||||
return $this->addChild($this->verbNode, $nodeName, $value);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Headers are enclosed inside of <record> to the query of ListRecords, ListIdentifiers and etc.
|
|
||||||
*
|
|
||||||
* @param $identifier Type: string. The identifier string for node <identifier>.
|
|
||||||
* @param $timestamp Type: timestamp. Timestamp in UTC format for node <datastamp>.
|
|
||||||
* @param $deleted Type: boolean. Deleted status for the record.
|
|
||||||
* @param $add_to_node Type: DOMElement. Default value is null.
|
|
||||||
* In normal cases, $add_to_node is the <record> node created previously.
|
|
||||||
* When it is null, the newly created header node is attatched to $this->verbNode.
|
|
||||||
* Otherwise it will be attached to the desired node defined in $add_to_node.
|
|
||||||
*/
|
|
||||||
function createHeader($identifier, $timestamp, $deleted = false, $add_to_node = null) {
|
|
||||||
if(is_null($add_to_node)) {
|
|
||||||
$header_node = $this->addToVerbNode('header');
|
|
||||||
} else {
|
|
||||||
$header_node = $this->addChild($add_to_node, 'header');
|
|
||||||
}
|
|
||||||
$this->addChild($header_node, 'identifier', $identifier);
|
|
||||||
$this->addChild($header_node, 'datestamp', $timestamp);
|
|
||||||
if($deleted) {
|
|
||||||
$header_node->setAttribute('status', 'deleted');
|
|
||||||
}
|
|
||||||
return $header_node;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If there are too many records request could not finished a resumpToken is generated to let harvester know
|
|
||||||
*
|
|
||||||
* @param $token Type: string. A random number created somewhere?
|
|
||||||
* @param $expirationdatetime Type: string. A string representing time.
|
|
||||||
* @param $num_rows Type: integer. Number of records retrieved.
|
|
||||||
* @param $cursor Type: string. Cursor can be used for database to retrieve next time.
|
|
||||||
*/
|
|
||||||
function createResumptionToken($token, $expirationdatetime, $num_rows, $cursor = null) {
|
|
||||||
$resump_node = $this->addChild($this->verbNode, 'resumptionToken', $token);
|
|
||||||
if(isset($expirationdatetime)) {
|
|
||||||
$resump_node->setAttribute('expirationDate', $expirationdatetime);
|
|
||||||
}
|
|
||||||
$resump_node->setAttribute('completeListSize', $num_rows);
|
|
||||||
$resump_node->setAttribute('cursor', $cursor);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Imports a XML fragment into a parent node on a XML Doc: a worker function.
|
|
||||||
*
|
|
||||||
* @param $mom_node Type: DOMNode. The target node.
|
|
||||||
* @param $fragment Type: DOMDocument. The XML fragment is being added
|
|
||||||
*
|
|
||||||
* @return DOMElement $added_node * The newly created node
|
|
||||||
*/
|
|
||||||
function importFragment($mom_node, $fragment) {
|
|
||||||
$added_node = $mom_node->appendChild($this->doc->importNode($fragment->documentElement, true));
|
|
||||||
return $added_node;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -20,7 +20,7 @@
|
||||||
// Make this script only executable via commandline interface!
|
// Make this script only executable via commandline interface!
|
||||||
if (php_sapi_name() !== 'cli') exit;
|
if (php_sapi_name() !== 'cli') exit;
|
||||||
|
|
||||||
require_once('config.php');
|
require_once './Configuration/Main.php';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Format output string
|
* Format output string
|
||||||
|
|
Loading…
Reference in New Issue