Browse Source
This patch introduces a new Z39.50 interface for searching Z39.50 compliant databases for MARC authority records. These databases aren't as common as their bibliographic equivalents, but they're out there and very useful. I have included info at the bottom of this messsage for sample authority databases you can try. To test this patch: 1) Set up Z39.50 client targets for authority databases. (I've included information at the bottom of this message for LibrariesAustralia's test server for authorities as well as instructions on how to use your Koha's z39.50 authority server as well. The Library of Congress also has authority databases available (unsure if these are test or prod), and you might have access to others through OCLC or RLIN. OCLC provides login credentials for their test databases. 2) Go to the Authorities module 3) Click on the new "Z39.50 search button" 4) Select your authority search targets from the list. 5) Do a search for an authority you would like using either the "Raw" input box or the more specific input boxes for names, subjects, subject sub divisions, or titles. (I like searching Name (personal): Eric on the LibrariesAustralia test DB.) 6) You should see a table listing the server, heading, authority type, and two other columns (MARC and a nameless column). "Authority type" is the type of authority it will become when imported in to Koha. In the Eric example, "PERSO_NAME". 7) Click on "MARC" next to the results of interest to review the MARC authority record. 8) When you're satisfied with a record, click on "Import". 9) The pop-up window will close and your original Koha window will change to the "Adding authority Personal Name" screen (in the Eric example). 10) All the relevant fields should be filled out for the record. Review them and make any changes as necessary. (N.B. The 001 will be cleared when saved, so if you have a use for the imported control number, move it to the 010, 016, or 035 as appropriate. If you have a default value for the 003, this will also likely be overwritten. Move it if necessary. The 005 will also be updated when saved, so do not worry about that.) 11) When you're satisfied, click save. 12) Presto! You've imported your first authority record via Z39.50! -- Here is the info for the LibrariesAustralia test Z39.50 authority database: Z39.50 server: LibrariesAustralia Authorities Hostname: z3950-test.librariesaustralia.nla.gov.au Port: 210 Database: AuthTraining Userid: ANLEZ Password: z39.50 Syntax: MARC21/USMARC Encoding: utf8 - The U.S.A. Library of Congress also provides Z39.50 access to its Name and Subject Authorities (http://www.loc.gov/z3950/lcserver.html). Name Authority: Z39.50 server: Library of Congress Name Authority File Hostname: lx2.loc.gov Port: 210 Database: NAF Syntax: MARC21/USMARC Encoding: utf8 Subject Authority: Z39.50 server: Library of Congress Subject Authority File Hostname: lx2.loc.gov Port: 210 Database: SAF Syntax: MARC21/USMARC Encoding: utf8 (N.B. Both of these databases also include title authorities.) - For testing purposes, you can also set up a Z39.50 client target, which points at your own Koha instance's Z39.50 authority server. To find the hostname, go to /etc/koha-conf.xml and find the value for the <listen id="authorityserver"> element. Depending on your configuration, this could be something like the following: unix:/zebra/koha/var/run/zebradb/authoritysocket (N.B. You might be using a different scheme than unix sockets...) To find the database, scroll down to the bottom of koha-conf.xml until you reach the <config> element. Within this, look for the value of the element <authorityserver>. It should probably be "authorities". To set up this Z39.50 client target in Koha... Z39.50 server: my koha authorities Hostname: unix:/zebra/koha/var/run/zebradb/authoritysocket Port: Database: authorities Userid: Password: Syntax: MARC21/USMARC (or whichever flavour you need) Encoding: utf8 Signed-off-by: Mason James <mtj@kohaaloha.com> Bug 10096 [FOLLOW-UP] - Add a z39.50 interface for authority searching This patch adds the "recordtype" column to the "z3950servers" table. The value in this column (biblio or authority) then controls whether the z3950 server shows up in a bibliographic search (through the Acq and Cataloguing modules) or in an authority search (through the Authorities module). I also edited the z3950 management console to show this value and allow users to edit it. The default value is "biblio", since the vast majority of z3950 targets will be bibliographic. However, there is an option to add/edit a z3950 target as a source of authority records. Test Plan: 1) Apply both patches 2) Run updatedatabase.pl (after setting your KOHA_CONF and PERL5 environmental variables) 3) Use the test plan from the 1st patch N.B. Make sure that your Z39.50 client target has a Record Type of Authority, otherwise it won't display when you're doing a Z3950 search for authorities. Signed-off-by: Mason James <mtj@kohaaloha.com> Bug 10096 [FOLLOW-UP] - fix tabs/whitespace errors to pass QA Signed-off-by: Kyle M Hall <kyle@bywatersolutions.com> Signed-off-by: Galen Charlton <gmc@esilibrary.com>new/bootstrap-opac
11 changed files with 766 additions and 63 deletions
@ -0,0 +1,108 @@ |
|||
#!/usr/bin/perl |
|||
|
|||
# This is a completely new Z3950 clients search using async ZOOM -TG 02/11/06 |
|||
# Copyright 2000-2002 Katipo Communications |
|||
# |
|||
# This is a new Z3950 authority search using the current Z3950 bibliographic search as a model 07/05/2013 |
|||
# Parts Copyright 2013 Prosentient Systems |
|||
# |
|||
# This file is part of Koha. |
|||
# |
|||
# Koha 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 2 of the License, or (at your option) any later |
|||
# version. |
|||
# |
|||
# Koha 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 Koha; if not, write to the Free Software Foundation, Inc., |
|||
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. |
|||
|
|||
use strict; |
|||
use warnings; |
|||
use CGI qw / -utf8 /; |
|||
|
|||
use C4::Auth; |
|||
use C4::Output; |
|||
use C4::Context; |
|||
use C4::Breeding; |
|||
use C4::Koha; |
|||
|
|||
my $input = new CGI; |
|||
my $dbh = C4::Context->dbh; |
|||
my $error = $input->param('error'); |
|||
my $nameany = $input->param('nameany'); |
|||
my $authorany = $input->param('authorany'); |
|||
my $authorcorp = $input->param('authorcorp'); |
|||
my $authorpersonal = $input->param('authorpersonal'); |
|||
my $authormeetingcon = $input->param('authormeetingcon'); |
|||
my $title = $input->param('title'); |
|||
my $uniformtitle = $input->param('uniformtitle'); |
|||
my $subject = $input->param('subject'); |
|||
my $subjectsubdiv = $input->param('subjectsubdiv'); |
|||
my $srchany = $input->param('srchany'); |
|||
my $op = $input->param('op')||''; |
|||
my $page = $input->param('current_page') || 1; |
|||
$page = $input->param('goto_page') if $input->param('changepage_goto'); |
|||
|
|||
my ( $template, $loggedinuser, $cookie ) = get_template_and_user({ |
|||
template_name => "cataloguing/z3950_auth_search.tmpl", |
|||
query => $input, |
|||
type => "intranet", |
|||
authnotrequired => 1, |
|||
flagsrequired => { catalogue => 1 }, |
|||
}); |
|||
|
|||
$template->param( |
|||
nameany => $nameany, |
|||
authorany => $authorany, |
|||
authorcorp => $authorcorp, |
|||
authorpersonal => $authorpersonal, |
|||
authormeetingcon => $authormeetingcon, |
|||
title => $title, |
|||
uniformtitle => $uniformtitle, |
|||
subject => $subject, |
|||
subjectsubdiv => $subjectsubdiv, |
|||
srchany => $srchany, |
|||
); |
|||
|
|||
if ( $op ne "do_search" ) { |
|||
my $sth = $dbh->prepare("SELECT id,host,name,checked FROM z3950servers WHERE recordtype = 'authority' ORDER BY rank, name"); |
|||
$sth->execute(); |
|||
my $serverloop = $sth->fetchall_arrayref( {} ); |
|||
$template->param( |
|||
serverloop => $serverloop, |
|||
opsearch => "search", |
|||
); |
|||
output_html_with_http_headers $input, $cookie, $template->output; |
|||
exit; |
|||
} |
|||
|
|||
my @id = $input->param('id'); |
|||
if ( @id==0 ) { |
|||
# empty server list -> report and exit |
|||
$template->param( emptyserverlist => 1 ); |
|||
output_html_with_http_headers $input, $cookie, $template->output; |
|||
exit; |
|||
} |
|||
|
|||
my $pars= { |
|||
random => $input->param('random') || rand(1000000000), |
|||
page => $page, |
|||
id => \@id, |
|||
nameany => $nameany, |
|||
authorany => $authorany, |
|||
authorcorp => $authorcorp, |
|||
authorpersonal => $authorpersonal, |
|||
authormeetingcon => $authormeetingcon, |
|||
title => $title, |
|||
uniformtitle => $uniformtitle, |
|||
subject => $subject, |
|||
subjectsubdiv => $subjectsubdiv, |
|||
srchany => $srchany, |
|||
}; |
|||
Z3950SearchAuth($pars, $template); |
|||
output_html_with_http_headers $input, $cookie, $template->output; |
@ -0,0 +1,227 @@ |
|||
[% INCLUDE 'doc-head-open.inc' %] |
|||
<title>Koha › Z39.50 search results</title> |
|||
[% INCLUDE 'greybox.inc' %] |
|||
[% INCLUDE 'doc-head-close.inc' %] |
|||
<script type="text/javascript" src="[% themelang %]/lib/jquery/plugins/jquery.checkboxes.min.js"></script> |
|||
<link rel="stylesheet" type="text/css" href="[% themelang %]/css/datatables.css" /> |
|||
<script type="text/javascript" src="[% themelang %]/lib/jquery/plugins/jquery.dataTables.min.js"></script> |
|||
[% INCLUDE 'datatables-strings.inc' %] |
|||
<script type="text/javascript" src="[% themelang %]/js/datatables.js"></script> |
|||
<script type="text/javascript"> |
|||
//<![CDATA[ |
|||
function Import(Breeding,AuthType) { |
|||
opener.document.location="../authorities/authorities.pl?breedingid="+Breeding+"&authtypecode="+AuthType; |
|||
window.close(); |
|||
return false; |
|||
} |
|||
|
|||
function closemenu(){ |
|||
$(".linktools").hide(); |
|||
$("tr").removeClass("selected"); |
|||
} |
|||
|
|||
$(document).ready(function(){ |
|||
$("#CheckAll").click(function(){ |
|||
$(".checkboxed").checkCheckboxes(); |
|||
return false; |
|||
}); |
|||
$("#CheckNone").click(function(){ |
|||
$(".checkboxed").unCheckCheckboxes(); |
|||
return false; |
|||
}); |
|||
$("#resultst").dataTable($.extend(true, {}, dataTablesDefaults, { |
|||
"sDom": 't', |
|||
"aoColumnDefs": [ |
|||
{ "aTargets": [ -1,-2,-3 ], "bSortable": false, "bSearchable": false }, |
|||
], |
|||
"aaSorting": [[ 1, "asc" ]], |
|||
"bPaginate": false |
|||
})); |
|||
/* Inline edit/delete links */ |
|||
$("td").click(function(event){ |
|||
var $tgt = $(event.target); |
|||
var row = $(this).parent(); |
|||
$(".linktools").hide(); |
|||
$("tr").removeClass("selected"); |
|||
row.addClass("selected"); |
|||
if($tgt.is("a")||$tgt.is(":nth-child(7)")||$tgt.is(":nth-child(8)")||$tgt.is(":nth-child(9)")||$tgt.is(":nth-child(10)")){ |
|||
return true; |
|||
} else { |
|||
var position = $(this).offset(); |
|||
var top = position.top+5; |
|||
var left = position.left+5; |
|||
$(".linktools",row).show().css("position","absolute").css("top",top).css("left",left); |
|||
} |
|||
}); |
|||
$("form[name='f']").submit(function(){ |
|||
if ($('input[type=checkbox]').filter(':checked').length == 0) { |
|||
alert(_("Please choose at least one Z39.50 target")); |
|||
return false; |
|||
} else |
|||
return true; |
|||
}); |
|||
}); |
|||
|
|||
[% IF ( total_pages ) %] |
|||
function validate_goto_page(){ |
|||
var page = $('#goto_page').val(); |
|||
if(isNaN(page)) { |
|||
alert(_("The page entered is not a number.")); |
|||
return false; |
|||
} |
|||
else if(page < 1 || page > [% total_pages %] ) { |
|||
alert(_("The page should be a number between 1 and ") + [% total_pages %] + "."); |
|||
return false; |
|||
} |
|||
else { |
|||
return true; |
|||
} |
|||
} |
|||
[% END %] |
|||
|
|||
//]]> |
|||
</script> |
|||
<style type="text/css"> |
|||
.linktools { background-color:#FFF;border-top:1px solid #DDD; border-left: 1px solid #DDD; border-right: 1px solid #666; border-bottom:1px solid #666;display: none; white-space: nowrap;} |
|||
.linktools a { font-size : 85%; text-decoration:none; padding:.3em;;background-color:#FFF; display:block;float:left;border-right:1px solid #DDD;} |
|||
.linktools a:hover { background-color:#EEE;color:#CC3300;border-right:1px solid #CCC;} |
|||
tr.selected { background-color : #FFFFCC; } tr.selected td { background-color : transparent; } |
|||
</style> |
|||
|
|||
[% IF ( opsearch ) %] |
|||
<style type="text/css"> |
|||
#custom-doc { width:53em;*width:51.72em;min-width:689px; margin:auto; text-align:left; } |
|||
</style> |
|||
</head> |
|||
<body id="cat_z3950_auth_search" class="cat"> |
|||
<div id="custom-doc" class="yui-t7"> |
|||
[% ELSE %] |
|||
</head> |
|||
<body style="padding:.5em;"> |
|||
<div> |
|||
[% END %] |
|||
<div id="bd"> |
|||
[% IF ( opsearch ) %] |
|||
<h2>Z39.50 Authority search points</h2> |
|||
<form method="post" action="z3950_auth_search.pl" name="f" class="checkboxed"> |
|||
<input type="hidden" name="op" id="op" value="do_search" /> |
|||
<div class="yui-g"> |
|||
<div class="yui-u first"> |
|||
<fieldset class="rows"> |
|||
<ol> |
|||
<li><label for="srchany">Keyword (any): </label> <input type="text" id="srchany" name="srchany" value="" /></li> |
|||
<li><label for="nameany">Name (any): </label> <input type="text" id="nameany" name="nameany" value="" /></li> |
|||
<li><label for="authorany">Author (any): </label> <input type="text" id="authorany" name="authorany" value="" /></li> |
|||
<li><label for="authorpersonal">Author (personal): </label> <input type="text" id="authorpersonal" name="authorpersonal" value="" /></li> |
|||
<li><label for="authorcorp">Author (corporate): </label> <input type="text" id="authorcorp" name="authorcorp" value="" /></li> |
|||
<li><label for="authormeetingcon">Author (meeting/conference): </label> <input type="text" id="authormeetingcon" name="authormeetingcon" value="" /></li> |
|||
</ol> |
|||
</fieldset> |
|||
</div> |
|||
<div class="yui-u"> |
|||
<fieldset class="rows"> |
|||
<ol> |
|||
<li><label for="subject">Subject heading: </label> <input type="text" id="subject" name="subject" value="" /></li> |
|||
<li><label for="subjectsubdiv">Subject sub-division: </label> <input type="text" id="subjectsubdiv" name="subjectsubdiv" value="" /></li> |
|||
<li><label for="title">Title (any): </label> <input type="text" id="title" name="title" value="[% title |html %]" /></li> |
|||
<li><label for="uniformtitle">Title (uniform): </label> <input type="text" id="uniformtitle" name="uniformtitle" value="[% uniformtitle |html %]" /></li> |
|||
</ol> |
|||
</fieldset> |
|||
</div> |
|||
</div> |
|||
<div class="yui-g"> |
|||
<h2>Search targets <span style="display: inline; font-size: 70%; padding-left: 1em;"><span class="checkall"><a id="CheckAll" href="#">Select all</a></span><span class="clearall"><a id="CheckNone" href="#">Clear all</a></span></span></h2> |
|||
[% FOREACH serverloo IN serverloop %] |
|||
<p> |
|||
[% IF ( serverloo.checked ) %] |
|||
<input type="checkbox" name="id" id="z3950_[% serverloo.id %]" value="[% serverloo.id %]" checked="checked" /> |
|||
[% ELSE %] |
|||
<input type="checkbox" name="id" id="z3950_[% serverloo.id %]" value="[% serverloo.id %]" /> |
|||
[% END %] |
|||
<label for="z3950_[% serverloo.id %]">[% serverloo.name %]</label> |
|||
</p> |
|||
[% END %] |
|||
</div> |
|||
<fieldset class="action"><input type="submit" class="submit" value="Search" onclick="cursor :'wait'"/> <a class="cancel close" href="#">Cancel</a></fieldset> |
|||
</form> |
|||
|
|||
|
|||
[% ELSE %] |
|||
<h2>Results for Authority Records</h2> |
|||
[% IF ( breeding_loop ) %] |
|||
<table id="resultst"> |
|||
<thead> <tr> |
|||
<th>Server</th> |
|||
<th>Heading</th> |
|||
<th>Authority Type</th> |
|||
<th>MARC</th> |
|||
<!-- <th>Card</th> --> |
|||
<th> </th> |
|||
</tr></thead> |
|||
<tbody>[% FOREACH breeding_loo IN breeding_loop %] |
|||
[% IF ( breeding_loo.breedingid ) %] |
|||
<tr id="row[% breeding_loo.breedingid %]"> |
|||
<td>[% breeding_loo.server %] <div class="linktools"><a href="/cgi-bin/koha/catalogue/showmarc.pl?importid=[% breeding_loo.breedingid %]" rel="gb_page_center[600,500]">Preview MARC</a> <a href="#" onclick="Import([% breeding_loo.breedingid %],'[% breeding_loo.heading_code %]'); return false">Import</a><a href="#" onclick="closemenu();return false;" title="Close this menu"> X </a></div> </td> |
|||
<td>[% breeding_loo.heading %]</td> |
|||
<td>[% breeding_loo.heading_code %]</td> |
|||
<td><a href="/cgi-bin/koha/catalogue/showmarc.pl?importid=[% breeding_loo.breedingid %]" title="MARC" rel="gb_page_center[600,500]">MARC</a></td> |
|||
<!-- <td><a href="/cgi-bin/koha/catalogue/showmarc.pl?viewas=card&importid=[% breeding_loo.breedingid %]" title="MARC" rel="gb_page_center[600,500]">Card</a></td> --> |
|||
<td><a href="#" onclick="Import([% breeding_loo.breedingid %],'[% breeding_loo.heading_code %]'); return false">Import</a></td> |
|||
</tr> |
|||
[% END %] |
|||
[% END %]</tbody> |
|||
</table> |
|||
|
|||
<form method="post" action="z3950_auth_search.pl" id="page_form" name="page_form" class="checkboxed"> |
|||
<input type="hidden" name="op" id="op" value="do_search" /> |
|||
<input type="hidden" name="current_page" id="current_page" value="[% current_page %]" /> |
|||
<input type="hidden" id="nameany" name="nameany" value="[% nameany %]" /> |
|||
<input type="hidden" id="authorany" name="authorany" value="[% authorany %]" /> |
|||
<input type="hidden" id="authorcorp" name="authorcorp" value="[% authorcorp %]" /> |
|||
<input type="hidden" id="authorpersonal" name="authorpersonal" value="[% authorpersonal %]" /> |
|||
<input type="hidden" id="authormeetingcon" name="authormeetingcon" value="[% authormeetingcon %]" /> |
|||
<input type="hidden" id="title" name="title" value="[% title %]" /> |
|||
<input type="hidden" id="uniformtitle" name="uniformtitle" value="[% uniformtitle %]" /> |
|||
<input type="hidden" id="subject" name="subject" value="[% subject %]" /> |
|||
<input type="hidden" id="subjectsubdiv" name="subjectsubdiv" value="[% subjectsubdiv %]" /> |
|||
<input type="hidden" id="heading" name="heading" value="[% heading %]" /> |
|||
<input type="hidden" id="srchany" name="srchany" value="[% srchany %]" /> |
|||
|
|||
[% FOREACH server IN servers %] |
|||
<input type="hidden" name="id" id="z3950_[% server.id %]" value="[% server.id %]" /> |
|||
[% END %] |
|||
|
|||
[% IF ( show_prevbutton ) %] |
|||
<input type="button" name="changepage_prev" value="Previous Page" onclick="$('#current_page').val([% current_page %]-1);$('#page_form').submit();" /> |
|||
[% END %] |
|||
Page [% current_page %] / [% total_pages %] |
|||
[% IF ( show_nextbutton ) %] |
|||
<input type="button" name="changepage_next" value="Next Page" onclick="$('#current_page').val([% current_page %]+1);$('#page_form').submit();" /> |
|||
[% END %] |
|||
<br />Go to page : <input id="goto_page" name="goto_page" value="[% current_page %]" size="4" /><input type="submit" name="changepage_goto" onclick="return validate_goto_page();" value="Go" /> |
|||
</form> |
|||
|
|||
<p><form method="get" action="/cgi-bin/koha/cataloguing/z3950_auth_search.pl"><input type="submit" value="Try Another Search"/></form></p> |
|||
[% ELSE %] |
|||
[% IF ( errconn ) %] |
|||
<div class="dialog alert"> |
|||
<ul> |
|||
[% FOREACH errcon IN errconn %] |
|||
[% IF ( errcon.error == '10000' ) %]<li>Connection failed to [% errcon.server %]</li> |
|||
[% ELSIF ( errcon.error == '10007' ) %]<li>Connection timeout to [% errcon.server %]</li>[% END %] |
|||
[% END %] |
|||
</ul> |
|||
</div> |
|||
[% END %] |
|||
<div class="dialog message">Nothing found.</div> |
|||
<p><form method="get" action="/cgi-bin/koha/cataloguing/z3950_auth_search.pl"><input type="submit" value="Try Another Search"/></form></p> |
|||
[% END %] |
|||
|
|||
[% END %] |
|||
</div> |
|||
</div> |
|||
|
|||
[% IF ( numberpending ) %]<h3 align="center">Still [% numberpending %] servers to search</h3>[% END %] |
|||
|
|||
</body> |
|||
</html> |
Loading…
Reference in new issue