various & sundry z3950 and MARC-related improvements
authorgfawcett <gfawcett@6d9bc8c9-1ec2-4278-b937-99fde70a366f>
Sun, 5 Apr 2009 18:33:55 +0000 (18:33 +0000)
committergfawcett <gfawcett@6d9bc8c9-1ec2-4278-b937-99fde70a366f>
Sun, 5 Apr 2009 18:33:55 +0000 (18:33 +0000)
Search queries no longer need explicit @and prefixes (they are now implicit);

a cleaner phys-item search-results page. Still only max 10 results showing, though.

various visual improvements.

git-svn-id: svn://svn.open-ils.org/ILS-Contrib/servres/trunk@275 6d9bc8c9-1ec2-4278-b937-99fde70a366f

13 files changed:
conifer/TODO
conifer/libsystems/z3950/marcxml.py
conifer/libsystems/z3950/yaz_search.py
conifer/static/main.css
conifer/syrup/views.py
conifer/templates/components/course.xhtml
conifer/templates/graham_z3950_test.xhtml [deleted file]
conifer/templates/item_add_cat_search.xhtml
conifer/templates/item_add_elec.xhtml
conifer/templates/item_add_heading.xhtml
conifer/templates/item_add_phys.xhtml
conifer/templates/item_add_url.xhtml
conifer/templates/item_metadata.xhtml

index 188c0b0..1b73ba8 100644 (file)
@@ -13,6 +13,8 @@ IMPORTANT:
 * People should be able to register themselves into open courses.
   That is, actually become a member of them.
 
+* accented characters in z3950 queries seem to fail. More tests needed.
+
 MAYBE:
 
 * Generating barcodes in emails, printable screens? (3 of 9 enough?)
index 1a208fb..741556a 100644 (file)
@@ -38,8 +38,16 @@ def marcxml_dictionary_to_dc(dct):
         value = dct.get(marc)
         if value:
             out[dc] = value
-    if '245b' in meta and 'dc:title' in out:
-        out['dc:title'] += (' %s' % meta['245b'])
+    title = out.get('dc:title')
+    if title:
+        if '245b' in dct:
+            title += (' %s' % dct['245b'])
+        # if title ends with a single character, strip it. usually a
+        # spurious punctuation character.
+        init, last = title.rsplit(' ',1)
+        if len(last) == 1:
+            title = init
+        out['dc:title'] = title
     return out
 
     
index 242dd62..8f1c996 100644 (file)
@@ -12,14 +12,26 @@ from marcxml import marcxml_to_dictionary
 
 LOG = sys.stderr #None              #  for pexpect debugging, try LOG = sys.stderr
 YAZ_CLIENT = 'yaz-client'
-GENERAL_TIMEOUT = 20
-PRESENT_TIMEOUT = 30
+GENERAL_TIMEOUT = 40
+PRESENT_TIMEOUT = 60
 
 def search(host, database, query, start=1, limit=None):
 
+    # first, let's look at our query. I'm assuming @prefix queries for
+    # now, so we need to put queries in that form if they aren't
+    # yet. I'm also assuming query is a conjunction (terms are
+    # AND'ed) if a '@' query isn't provided.
+    if not query.startswith('@'):
+        words = [w for w in query.split(' ') if w.strip()]
+        tmp   = (['@and'] * (len(words) - 1)) + words
+        query = ' '.join(tmp)
+    query = query.encode('utf-8') # is this okay? Is it enough??
+
     server = pexpect.spawn('yaz-client', timeout=GENERAL_TIMEOUT, logfile=LOG)
     #server.expect('Z>')
-    for line in ('open %s' % host, 'base %s' % database, 'format xml'):
+    for line in ('open %s' % host, 
+                 'base %s' % database, 
+                 'format xml'):
         server.sendline(line)
         server.expect('Z>')
 
index 8be1ac7..32f787c 100644 (file)
@@ -245,6 +245,7 @@ p.todo, div.todo { background-color: #fdd; padding: 6; margin: 12; border-left:
 .metadata_table tbody td {
     padding: 8; border: #ddd 1px solid;
 }
+.metadata_table input { width: 600; }
 
 .metadata_table tbody th {
     background-color: #eee;
@@ -271,4 +272,4 @@ li.sort_item:hover { background-color: #eee; }
 
 ul.heading_tree li  { list-style: none; }
 ul.heading_tree { margin: 0; padding-left: 0; }
-ul.heading_tree ul { margin: 0; padding-left: 25; }
\ No newline at end of file
+ul.heading_tree ul { margin: 0; padding-left: 25; }
index 671ee19..0b77000 100644 (file)
@@ -275,23 +275,6 @@ def z3950_test(request):
     res_str = "" . join(collector)
     return g.render('z3950_test.xhtml', res_str=res_str)
 
-def graham_z3950_test(request):
-    raise NotImplementedError   # delete this function, its template, etc.
-    query = request.GET.get('query', '@and "Denis" "Gravel"')
-    from conifer.libsystems.z3950 import yaz_search
-    from conifer.libsystems.evergreen.item_status import lookup_availability
-    host, db, query = ('dwarf.cs.uoguelph.ca:2210', 'conifer', query)
-    #host, db, query = ('z3950.loc.gov:7090', 'VOYAGER', '@attr 1=4 @attr 4=1 "dylan"')
-    results = yaz_search.search(host, db, query)
-    for result in results:
-        bibid = result.get('901c')
-        if bibid:
-            # it would be better to do these asynchronously.
-            avail = lookup_availability(bibid) 
-            if avail:
-                result['avail'] = avail
-    return g.render('graham_z3950_test.xhtml', results=results, query=query)
-
 def browse(request, browse_option=''):
     #the defaults should be moved into a config file or something...
     page_num = int(request.GET.get('page', 1))
@@ -728,7 +711,7 @@ def item_add(request, course_id, item_id):
 @instructors_only
 def item_add_cat_search(request, course_id, item_id):
     if request.method != 'POST':
-        return g.render('item_add_cat_search.xhtml', results=[], query='@and dylan thomas')
+        return g.render('item_add_cat_search.xhtml', results=[], query='')
 
     # POST handler
     query     = request.POST.get('query','').strip()
@@ -743,7 +726,8 @@ def item_add_cat_search(request, course_id, item_id):
         # User has selected an item; add it to course site.
         #fixme, this block copied from item_add. refactor.
         parent_item_id = item_id
-        if parent_item_id=='0':
+        if parent_item_id == '0': 
+            # no heading (toplevel)
             parent_item = None
             course = get_object_or_404(models.Course, pk=course_id)
         else:
@@ -757,14 +741,13 @@ def item_add_cat_search(request, course_id, item_id):
         dublin = marcxml_dictionary_to_dc(pickitem)
 
         item = course.item_set.create(parent_heading=parent_item,
-                                      title=pickitem.get('245a', 'Untitled'),
+                                      title=dublin.get('dc:title','Untitled'),
                                       item_type='PHYS')
         item.save()
 
         for dc, value in dublin.items():
             md = item.metadata_set.create(item=item, name=dc, value=value)
-        # store the whole darn MARC-dict as well.
-        json = simplejson.dumps(pickitem)
+        # store the whole darn MARC-dict as well (JSON)
         item.metadata_set.create(item=item, name='syrup:marc', value=raw_pickitem)
         item.save()
         return HttpResponseRedirect('../../../%d/' % item.id)
index 779b331..712aa8a 100644 (file)
@@ -93,7 +93,7 @@ searchtext = _('search this course...')
       <p py:if="show_more"><a href="javascript:show_metadata();" id="show_metadata">Show more attributes</a></p>
       <div  id="metadatatable">
       ${Markup(metadata_formset.management_form)}
-      <table class="pagetable">
+      <table class="metadata_table">
        <thead>
          <tr><th>Attribute</th><th>Value</th><th/><th>Delete?</th></tr>
        </thead>
diff --git a/conifer/templates/graham_z3950_test.xhtml b/conifer/templates/graham_z3950_test.xhtml
deleted file mode 100644 (file)
index 0ba131f..0000000
+++ /dev/null
@@ -1,52 +0,0 @@
-<?python
-title = _('Z39.50 Test, with Evergreen giving item availability')
-# I just made up these keys...
-keys = [('245a', 'Title'), ('100a', 'Author'), ('260c', 'PubDate'), ('901c', 'BibID')]
-?>
-<html xmlns="http://www.w3.org/1999/xhtml"
-      xmlns:xi="http://www.w3.org/2001/XInclude"
-      xmlns:py="http://genshi.edgewall.org/">
-<xi:include href="master.xhtml"/>
-<xi:include href="paginate.xhtml"/>
-<head>
-  <title>${title}</title>
-  <style>
-    tbody.available { background-color: #dfd; }
-    .lesser { font-size: smaller; color: gray; display: none; }
-  </style>
-  <script type="text/javascript">
-    <!-- !This ought to be in paginate.xhtml, not here. how to do? -->
-    $(function() { $('.pagetable').tablesorter(); });
-  </script>
-</head>
-<body>
-    <h1>${title}</h1>
-    <form method="GET" action=".">
-      <input type="text" name="query" value="${query}" style="font-size: larger;"/>
-      <input type="submit" value="Search"/>
-    </form>
-    <p><a href="javascript:$('.lesser').show(); void(0);">show more detail</a></p>
-    <table py:for="res in results" style="margin: 1em 0; border: black 1px solid;">
-      <?python 
-       is_available = res.get('avail') and (res['avail']['available'] or res['avail']['holdable'])
-      ?>
-      <tbody class="${is_available and 'available' or ''}">
-      <tr py:for="k, title in keys" py:if="k in res">
-       <th>${title}</th><td>${res[k]}</td>
-      </tr>
-      <tr py:if="'avail' in res">
-       <th>Availability</th>
-       <td>
-         <div py:for="k,v in res['avail'].items()">
-         ${'%s = %s' % (k,v)}
-         </div>
-       </td>
-      </tr>
-      <?python allkeys = res.keys(); allkeys.sort(); ?>
-      <tr py:for="k in allkeys" class="lesser">
-       <th>${k}</th><td>${res[k]}</td>
-      </tr>
-      </tbody>
-    </table>
-</body>
-</html>
index 2681c1a..ed6f011 100644 (file)
@@ -1,8 +1,8 @@
 <?python
 from django.utils.simplejson import dumps
+from conifer.libsystems.z3950.marcxml import marcxml_dictionary_to_dc as to_dublin
 title = _('Add physical item: Catalogue search')
-# I just made up these keys...
-keys = [('245a', 'Title'), ('100a', 'Author'), ('260b', 'Publisher'), ('260c', 'PubDate'), ('700a', 'Editor')]
+dc_keys = ['dc:title', 'dc:creator', 'dc:publisher', 'dc:date']
 ?>
 <html xmlns="http://www.w3.org/1999/xhtml"
       xmlns:xi="http://www.w3.org/2001/XInclude"
@@ -11,9 +11,6 @@ keys = [('245a', 'Title'), ('100a', 'Author'), ('260b', 'Publisher'), ('260c', '
 <xi:include href="paginate.xhtml"/>
 <head>
   <title>${title}</title>
-  <style>
-    .lesser { font-size: smaller; color: gray; display: none; }
-  </style>
   <script type="text/javascript">
     <!-- !This ought to be in paginate.xhtml, not here. how to do? -->
     $(function() { $('.pagetable').tablesorter(); });
@@ -25,28 +22,41 @@ keys = [('245a', 'Title'), ('100a', 'Author'), ('260b', 'Publisher'), ('260c', '
 <body>
     <h1>${title}</h1>
     <form method="POST" action=".">
-      <input type="text" id="query" name="query" value="${query}" style="font-size: larger;"/>
+      <input type="text" id="query" name="query" value="${query}" 
+            style="font-size: larger; width: 600;"/>
       <input type="submit" value="Search"/>
     </form>
-    <p><a href="javascript:$('.lesser').show(); void(0);">show more detail</a></p>
-    <table py:for="res in results" style="margin: 1em 0; border: black 1px solid;">
-      <tbody>
-      <tr py:for="k, title in keys" py:if="k in res">
-       <th>${title}</th><td>${res[k]}</td>
-      </tr>
-      <tr>
-       <th/><td>
-       <form action="." method="POST">
-         <input type="hidden" name="pickitem" value="${dumps(res)}"/>
-         <input type="submit" value="Pick this item"/>
-       </form>
-      </td>
-      </tr>
-      <?python allkeys = res.keys(); allkeys.sort(); ?>
-      <tr py:for="k in allkeys" class="lesser">
-       <th>${k}</th><td>${res[k]}</td>
-      </tr>
+
+    <table class="pagetable">
+      <thead>
+       <tr><th>Title</th><th>Author</th><th>Publisher</th><th>PubDate</th></tr>
+      </thead>
+      <tbody py:for="resultnum, res in enumerate(results)"
+            py:with="dc=to_dublin(res)">
+       <tr>
+         <td>
+           ${dc.get('dc:title', '???')}
+           <a href="javascript:$('#full_${resultnum}').toggle(); void(0);">details</a>
+         </td>
+         <td py:for="k in dc_keys[1:]">${dc.get(k) or '&mdash;'}</td>
+         <td>
+           <form action="." method="POST">
+             <input type="hidden" name="pickitem" value="${dumps(res)}"/>
+             <input type="submit" value="Pick this item"/>
+           </form>
+         </td>
+       </tr>
+       <tr id="full_${resultnum}" style="display: none;">
+         <td colspan="4" style="padding-left: 36;">
+           <table class="metadata_table">
+             <?python allkeys = res.keys(); allkeys.sort(); ?>
+             <tr py:for="k in allkeys">
+               <th>${k}</th><td>${res[k]}</td>
+             </tr>
+           </table>
+         </td>
+       </tr>
       </tbody>
     </table>
-</body>
+ </body>
 </html>
index 18bd0f9..f17f03c 100644 (file)
@@ -24,7 +24,7 @@ course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
     <div py:if="not is_edit">
       <form action=".?item_type=${item_type}" method="POST"
            enctype="multipart/form-data">
-       <table>
+       <table class="metadata_table">
          <tr><th>Title of document</th><td><input type="text" 
          name="title"
          value="${item.title}"/></td></tr>
index 94ab2b2..426c550 100644 (file)
@@ -21,7 +21,7 @@ course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
     ${offer_to_delete(item)}
     <h3>${title}</h3>
     <form action=".?item_type=${item_type}" method="POST">
-      <table>
+      <table class="metadata_table">
        <tr><th>Heading</th><td>
        <input type="text" name="title"
               value="${item.title}"/>
index 01692a3..febc734 100644 (file)
@@ -21,8 +21,13 @@ course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
     <h3>${title}</h3>
 
     <form action=".?item_type=${item_type}" method="POST">
-      <p>Display name: <input type="text" name="title" value="${item.title}"/></p>
-
+      <table class="metadata_table">
+       <tr>
+         <th>Display name</th>
+         <td><input type="text" name="title" value="${item.title}"/></td>
+       </tr>
+      </table>
+      <h3>Metadata</h3>
       ${item_metadata_formset(show_more=False)}
       <p>
        <input py:if="not is_edit" type="submit" value="Add heading"/>
index da68d9e..513bd8d 100644 (file)
@@ -21,7 +21,7 @@ course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
     ${offer_to_delete(item)}
     <h3>${title}</h3>
     <form action=".?item_type=${item_type}" method="POST">
-      <table>
+      <table class="metadata_table">
        <tr><th>Title</th><td><input type="text" name="title" value="${item.title}"/></td></tr>
        <tr><th>URL</th><td><input type="text" name="url" value="${item.url}"/></td></tr>
       </table>
index 1014bcb..9ce7f75 100644 (file)
@@ -1,4 +1,5 @@
 <?python
+from django.utils.simplejson import loads
 course_title = '%s: %s (%s)' % (course.code, course.title, course.term)
 hier = item.hierarchy()[:-1]
 title = item.title
@@ -41,7 +42,22 @@ is_editor = course.can_edit(request.user)
       <h2 class="metadata_subhead">Additional metadata</h2>
       <table class="metadata_table">
        <tr py:for="attr in metadata">
-         <th>${attr.get_name_display()}</th><td>${attr.value}</td>
+         <th>${attr.get_name_display()}</th>
+         <td py:if="attr.name != 'syrup:marc'">
+           ${attr.value}
+         </td>
+         <td py:if="attr.name == 'syrup:marc'">
+           <div id="marcdatashow"><a href="javascript:$('#marcdata').show();$('#marcdatashow').hide();void(0);">show</a></div>
+           <div id="marcdata" style="display: none;">
+             <?python dct = loads(attr.value); keys=dct.keys(); keys.sort() ?>
+             <table>
+               <tr py:for="k in keys">
+                 <th>${k}</th>
+                 <td>${dct[k]}</td>
+               </tr>
+             </table>
+           </div>
+         </td>
        </tr>
       </table>
       </div>