umlaut 3.0.5 → 3.1.0.pre1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (385) hide show
  1. checksums.yaml +15 -0
  2. data/README.md +48 -5
  3. data/app/assets/images/umlaut_icons.png +0 -0
  4. data/app/assets/images/umlaut_icons/famfamfam-book-go.png +0 -0
  5. data/app/assets/images/{famfamfam/book_open.png → umlaut_icons/famfamfam-book-open.png} +0 -0
  6. data/app/assets/images/umlaut_icons/famfamfam-book.png +0 -0
  7. data/app/assets/images/{famfamfam/cross.png → umlaut_icons/famfamfam-cross.png} +0 -0
  8. data/app/assets/images/umlaut_icons/famfamfam-error.png +0 -0
  9. data/app/assets/images/umlaut_icons/famfamfam-help.png +0 -0
  10. data/app/assets/images/umlaut_icons/famfamfam-information.png +0 -0
  11. data/app/assets/images/umlaut_icons/famfamfam-link.png +0 -0
  12. data/app/assets/images/umlaut_icons/famfamfam-page-sound.png +0 -0
  13. data/app/assets/images/umlaut_icons/famfamfam-page-text.png +0 -0
  14. data/app/assets/images/umlaut_icons/famfamfam-page-up.png +0 -0
  15. data/app/assets/images/{famfamfam/page_white.png → umlaut_icons/famfamfam-page-white.png} +0 -0
  16. data/app/assets/images/{famfamfam/readme.html → umlaut_icons/famfamfam-readme.html} +0 -0
  17. data/app/assets/images/{famfamfam/tiny_cross.png → umlaut_icons/famfamfam-tiny-cross.png} +0 -0
  18. data/app/assets/images/{list_closed.png → umlaut_icons/list-closed.png} +0 -0
  19. data/app/assets/images/{list_open.png → umlaut_icons/list-open.png} +0 -0
  20. data/app/assets/javascripts/umlaut.js +5 -4
  21. data/app/assets/javascripts/umlaut/ajax_windows.js +41 -23
  22. data/app/assets/javascripts/umlaut/expand_contract_toggle.js +21 -29
  23. data/app/assets/javascripts/umlaut/load_permalink.js +26 -0
  24. data/app/assets/javascripts/umlaut/search_autocomplete.js +103 -44
  25. data/app/assets/stylesheets/umlaut.css.scss +19 -0
  26. data/app/assets/stylesheets/umlaut/_admin.scss +14 -0
  27. data/app/assets/stylesheets/umlaut/_az.scss +29 -0
  28. data/app/assets/stylesheets/umlaut/_forms.scss +31 -0
  29. data/app/assets/stylesheets/umlaut/_icons.scss +64 -0
  30. data/app/assets/stylesheets/umlaut/_layout.scss +52 -0
  31. data/app/assets/stylesheets/umlaut/_misc.scss +59 -0
  32. data/app/assets/stylesheets/umlaut/_mixins.scss +65 -0
  33. data/app/assets/stylesheets/umlaut/_modal.scss +4 -0
  34. data/app/assets/stylesheets/umlaut/_resolve.scss +308 -0
  35. data/app/assets/stylesheets/umlaut/_results.scss +34 -0
  36. data/app/assets/stylesheets/umlaut/_search.scss +14 -0
  37. data/app/assets/stylesheets/umlaut/_spinner.scss +12 -0
  38. data/app/assets/stylesheets/umlaut/_variables.scss +72 -0
  39. data/app/controllers/export_email_controller.rb +22 -39
  40. data/app/controllers/journal_tocs_controller +90 -0
  41. data/app/controllers/js_helper_controller.rb +2 -3
  42. data/app/controllers/link_router_controller.rb +32 -43
  43. data/app/controllers/open_search_controller.rb +3 -4
  44. data/app/controllers/resolve_controller.rb +68 -107
  45. data/app/controllers/resource_controller.rb +14 -20
  46. data/app/controllers/search_controller.rb +75 -94
  47. data/app/controllers/search_methods/README.md +28 -0
  48. data/app/controllers/search_methods/sfx4.rb +50 -119
  49. data/app/controllers/search_methods/sfx4_solr/README.md +57 -0
  50. data/app/controllers/search_methods/sfx4_solr/local.rb +40 -0
  51. data/app/controllers/search_methods/sfx4_solr/searcher.rb +90 -0
  52. data/app/controllers/store_controller.rb +24 -30
  53. data/app/controllers/umlaut/controller_behavior.rb +45 -17
  54. data/app/controllers/umlaut/error_handling.rb +20 -25
  55. data/{lib → app/controllers}/umlaut_configurable.rb +35 -21
  56. data/app/controllers/umlaut_controller.rb +43 -46
  57. data/app/helpers/emailer_helper.rb +9 -16
  58. data/app/helpers/export_email_helper.rb +8 -8
  59. data/app/helpers/open_search_helper.rb +2 -3
  60. data/app/helpers/resolve_helper.rb +130 -100
  61. data/app/helpers/search_helper.rb +17 -5
  62. data/app/helpers/umlaut/footer_helper.rb +1 -1
  63. data/app/helpers/umlaut/helper.rb +34 -0
  64. data/app/helpers/umlaut/html_head_helper.rb +12 -21
  65. data/{lib → app/mixin_logic}/marc_helper.rb +1 -1
  66. data/{lib → app/mixin_logic}/metadata_helper.rb +6 -5
  67. data/{lib → app/mixin_logic}/umlaut_http.rb +0 -0
  68. data/{lib → app/mixin_logic}/xml_schema_helper.rb +0 -0
  69. data/app/models/collection.rb +97 -95
  70. data/{lib → app/models}/hip3/bib.rb +0 -0
  71. data/{lib → app/models}/hip3/bib_searcher.rb +0 -0
  72. data/{lib → app/models}/hip3/custom_field_lookup.rb +0 -0
  73. data/{lib → app/models}/hip3/holding.rb +0 -0
  74. data/{lib → app/models}/hip3/item.rb +0 -0
  75. data/{lib → app/models}/hip3/receipt.rb +0 -0
  76. data/{lib → app/models}/hip3/serial_copy.rb +0 -0
  77. data/app/models/permalink.rb +3 -8
  78. data/app/models/referent.rb +29 -21
  79. data/app/models/request.rb +19 -8
  80. data/app/models/service_response.rb +24 -3
  81. data/app/models/service_store.rb +108 -39
  82. data/app/models/service_store.rb-NEW +73 -0
  83. data/app/models/service_wave.rb +2 -0
  84. data/app/models/sfx4/abstract/README.md +68 -0
  85. data/app/models/sfx4/abstract/az_extra_info.rb +44 -0
  86. data/app/models/sfx4/abstract/az_letter_group.rb +24 -0
  87. data/app/models/sfx4/abstract/az_title.rb +96 -0
  88. data/app/models/sfx4/abstract/az_title_search.rb +24 -0
  89. data/app/models/sfx4/abstract/base.rb +77 -0
  90. data/app/models/sfx4/global/base.rb +24 -0
  91. data/app/models/sfx4/global/kb_objects.rb +34 -0
  92. data/app/models/sfx4/local/az_extra_info.rb +7 -0
  93. data/app/models/sfx4/local/az_letter_group.rb +7 -0
  94. data/app/models/sfx4/local/az_title.rb +7 -0
  95. data/app/models/sfx4/local/az_title_search.rb +7 -0
  96. data/app/models/sfx4/local/base.rb +26 -0
  97. data/{lib → app/presentation}/section_renderer.rb +0 -12
  98. data/app/referent_filters/dissertation_catch.rb +67 -0
  99. data/{lib → app/referent_filters}/referent_filter.rb +0 -0
  100. data/{lib → app}/service_adaptors/ajax_export.rb +0 -0
  101. data/{lib → app}/service_adaptors/amazon.rb +2 -0
  102. data/{lib → app}/service_adaptors/blacklight.rb +22 -10
  103. data/{lib → app}/service_adaptors/book_finder.rb +0 -0
  104. data/{lib → app}/service_adaptors/bx.rb +0 -0
  105. data/{lib → app}/service_adaptors/cover_thing.rb +0 -0
  106. data/{lib → app}/service_adaptors/dummy_service.rb +0 -0
  107. data/{lib → app}/service_adaptors/elsevier_cover.rb +0 -0
  108. data/{lib → app}/service_adaptors/email_export.rb +0 -0
  109. data/{lib → app}/service_adaptors/ezproxy.rb +0 -0
  110. data/{lib → app}/service_adaptors/google_book_search.rb +0 -0
  111. data/app/service_adaptors/google_scholar_link.rb +71 -0
  112. data/{lib → app}/service_adaptors/gpo.rb +0 -0
  113. data/{lib → app}/service_adaptors/hathi_trust.rb +0 -0
  114. data/{lib → app}/service_adaptors/hip3_service.rb +0 -0
  115. data/{lib → app}/service_adaptors/hip_holding_search.rb +0 -0
  116. data/{lib → app}/service_adaptors/internet_archive.rb +0 -0
  117. data/{lib → app}/service_adaptors/isbn_db.rb +0 -0
  118. data/{lib → app}/service_adaptors/isi.rb +22 -13
  119. data/{lib → app}/service_adaptors/jcr.rb +0 -0
  120. data/{lib → app}/service_adaptors/opac.rb +0 -0
  121. data/{lib → app}/service_adaptors/open_library.rb +0 -0
  122. data/{lib → app}/service_adaptors/open_library_cover.rb +0 -0
  123. data/{lib → app}/service_adaptors/pubmed.rb +0 -0
  124. data/{lib → app}/service_adaptors/request_to_fixture.rb +0 -0
  125. data/{lib → app}/service_adaptors/scopus.rb +0 -0
  126. data/{lib → app/service_adaptors}/service.rb +3 -3
  127. data/{lib → app}/service_adaptors/sfx.rb +154 -11
  128. data/{lib → app}/service_adaptors/sfx_backchannel_record.rb +0 -0
  129. data/app/service_adaptors/tr_link.rb +267 -0
  130. data/app/service_adaptors/tr_links.rb +278 -0
  131. data/{lib → app}/service_adaptors/txt_holding_export.rb +0 -0
  132. data/{lib → app}/service_adaptors/ulrichs_cover.rb +0 -0
  133. data/{lib → app}/service_adaptors/ulrichs_link.rb +0 -0
  134. data/{lib → app}/service_adaptors/worldcat.rb +0 -0
  135. data/{lib → app}/service_adaptors/worldcat_identities.rb +0 -0
  136. data/app/views/admin/service_errors/_dispatched_service.html.erb +13 -13
  137. data/app/views/admin/service_errors/index.html.erb +11 -21
  138. data/app/views/export_email/email.html.erb +17 -2
  139. data/app/views/export_email/send_email.html.erb +2 -1
  140. data/app/views/export_email/send_txt.html.erb +2 -1
  141. data/app/views/export_email/txt.html.erb +30 -2
  142. data/app/views/layouts/umlaut.html.erb +29 -49
  143. data/app/views/resolve/_background_progress.html.erb +43 -50
  144. data/app/views/resolve/_background_updater.html.erb +23 -36
  145. data/app/views/resolve/_citation.html.erb +63 -87
  146. data/app/views/resolve/_compact_citation.html.erb +15 -33
  147. data/app/views/resolve/_cover_image.html.erb +34 -43
  148. data/app/views/resolve/_fulltext.html.erb +29 -50
  149. data/app/views/resolve/_help.html.erb +17 -14
  150. data/app/views/resolve/_holding.html.erb +65 -82
  151. data/app/views/resolve/_manually_entered_warning.html.erb +7 -0
  152. data/app/views/resolve/_modal.html.erb +19 -0
  153. data/app/views/resolve/_related_items.html.erb +20 -35
  154. data/app/views/resolve/_search_inside.html.erb +34 -57
  155. data/app/views/resolve/_section_display.html.erb +34 -46
  156. data/app/views/resolve/_section_heading.html.erb +17 -0
  157. data/app/views/resolve/_service_errors.html.erb +27 -28
  158. data/app/views/resolve/_standard_response_item.html.erb +42 -72
  159. data/app/views/resolve/background_status.html.erb +19 -23
  160. data/app/views/resolve/get_permalink.html.erb +6 -0
  161. data/app/views/resolve/index.html.erb +47 -73
  162. data/app/views/search/_a_to_z.html.erb +9 -6
  163. data/app/views/search/_citation.html.erb +73 -94
  164. data/app/views/search/_pager.html.erb +6 -6
  165. data/app/views/search/books.html.erb +34 -102
  166. data/app/views/search/journal_search.html.erb +53 -90
  167. data/app/views/search/journals.html.erb +73 -167
  168. data/app/views/umlaut/_header.html.erb +9 -0
  169. data/app/views/umlaut/error.html.erb +44 -44
  170. data/config/locales/en.yml +5 -0
  171. data/lib/{CronTab.rb → cron_tab.rb} +0 -0
  172. data/lib/generators/templates/umlaut_services.yml +57 -43
  173. data/lib/generators/umlaut/install_generator.rb +12 -0
  174. data/lib/service_adaptors/README.md +0 -0
  175. data/lib/tasks/umlaut.rake +8 -24
  176. data/lib/umlaut.rb +1 -21
  177. data/lib/umlaut/version.rb +1 -1
  178. data/test/README.md +75 -0
  179. data/test/dummy/app/controllers/umlaut_controller.rb +3 -3
  180. data/test/dummy/config/application.rb +3 -0
  181. data/test/dummy/config/sunspot.yml +20 -0
  182. data/test/dummy/config/travis_database.yml +96 -0
  183. data/test/dummy/config/umlaut_services.yml +16 -23
  184. data/test/dummy/db/migrate/20120927163304_sfx4_global.rb +26 -0
  185. data/test/dummy/db/migrate/20120927164040_sfx4_local.rb +54 -0
  186. data/test/dummy/tmp/cache/assets/BBA/6A0/sprockets%2F2a335471966f33256b321f8324041981 +0 -0
  187. data/test/dummy/tmp/cache/assets/C2A/410/sprockets%2Fd654b74912b4773a2534616863fb6565 +0 -0
  188. data/test/dummy/tmp/cache/assets/C45/A30/sprockets%2F39494895e462697b478d3d0c79298a26 +0 -0
  189. data/test/dummy/tmp/cache/assets/C49/730/sprockets%2F212d35831188417b5131e3e693aa69a6 +0 -0
  190. data/test/dummy/tmp/cache/assets/C55/1D0/sprockets%2F5af2f2a3403040f736981863fd278529 +0 -0
  191. data/test/dummy/tmp/cache/assets/C5A/A80/sprockets%2F2328c7370b56b4151776981a5f6d394e +0 -0
  192. data/test/dummy/tmp/cache/assets/C5F/340/sprockets%2F99692920160b7a279b86a80415b79db7 +0 -0
  193. data/test/dummy/tmp/cache/assets/C70/4D0/sprockets%2F034ad2036e623081bd352800786dfe80 +0 -0
  194. data/test/dummy/tmp/cache/assets/C80/980/sprockets%2Fc94807409c1523d43e18d25f35d93c41 +0 -0
  195. data/test/dummy/tmp/cache/assets/C94/E20/sprockets%2F19579abc248373f419b1c3e287d74598 +0 -0
  196. data/test/dummy/tmp/cache/assets/C98/330/sprockets%2F0c81574ba94abc10657315d4190c201f +0 -0
  197. data/test/dummy/tmp/cache/assets/CA4/440/sprockets%2F0feb4425db6319d7d0582192a503671b +0 -0
  198. data/test/dummy/tmp/cache/assets/CA7/E80/sprockets%2F33310f455a870eaedd6d938e30620714 +0 -0
  199. data/test/dummy/tmp/cache/assets/CB8/3C0/sprockets%2F697cd846490fda9f0e20c558320610f8 +0 -0
  200. data/test/dummy/tmp/cache/assets/CBF/B60/sprockets%2F08ca89671549936265dcb673bf02e36f +0 -0
  201. data/test/dummy/tmp/cache/assets/CC7/F10/sprockets%2F939637f806eef3f0f21584659458ab2e +0 -0
  202. data/test/dummy/tmp/cache/assets/CC9/9F0/sprockets%2F306166316e2cafd13c15e62b51a2339d +0 -0
  203. data/test/dummy/tmp/cache/assets/CD8/370/sprockets%2F357970feca3ac29060c1e3861e2c0953 +0 -0
  204. data/test/dummy/tmp/cache/assets/CF7/2B0/sprockets%2F25a7c73655bd3598173b39d9f98bcd46 +0 -0
  205. data/test/dummy/tmp/cache/assets/CF8/DA0/sprockets%2Ff5b8c8499d3da954d392fa57739503c5 +0 -0
  206. data/test/dummy/tmp/cache/assets/CF9/590/sprockets%2F096bcc45cd31b3796fa2121cd203215f +0 -0
  207. data/test/dummy/tmp/cache/assets/CFE/080/sprockets%2F37fe9f4255baddbd549a659914929398 +0 -0
  208. data/test/dummy/tmp/cache/assets/D04/CC0/sprockets%2Fe9c1b92f2e437e1bbbe020d718739220 +0 -0
  209. data/test/dummy/tmp/cache/assets/D0B/090/sprockets%2F58009091c95e3ac328c91c68fca30e0e +0 -0
  210. data/test/dummy/tmp/cache/assets/D1A/B50/sprockets%2F2db5892438129fe94da8429b8be114ec +0 -0
  211. data/test/dummy/tmp/cache/assets/D2D/1A0/sprockets%2F376426b4896a3cff7969ce6c7b387e0e +0 -0
  212. data/test/dummy/tmp/cache/assets/D32/A10/sprockets%2F13fe41fee1fe35b49d145bcc06610705 +0 -0
  213. data/test/dummy/tmp/cache/assets/D37/2B0/sprockets%2F40834fb07d7318c1fddd5003bd9e04f6 +0 -0
  214. data/test/dummy/tmp/cache/assets/D43/0D0/sprockets%2F682843a8d0795a5fbcfeb2f0c81727d0 +0 -0
  215. data/test/dummy/tmp/cache/assets/D46/F80/sprockets%2Fb15c08e749067d7aad9baf5e9388221c +0 -0
  216. data/test/dummy/tmp/cache/assets/D4E/1B0/sprockets%2Ff7cbd26ba1d28d48de824f0e94586655 +0 -0
  217. data/test/dummy/tmp/cache/assets/D5A/EA0/sprockets%2Fd771ace226fc8215a3572e0aa35bb0d6 +0 -0
  218. data/test/dummy/tmp/cache/assets/D6C/7D0/sprockets%2F8a05d6981ec0d38c51739bef0b3a9c2b +0 -0
  219. data/test/dummy/tmp/cache/assets/D6E/CF0/sprockets%2F5661c6e4f0cea86fe4523f27261aade5 +0 -0
  220. data/test/dummy/tmp/cache/assets/D94/FF0/sprockets%2F3b56a1aa77de0d570c38a4a9d5f4b1d6 +0 -0
  221. data/test/dummy/tmp/cache/assets/D97/6B0/sprockets%2Fb070e8c799d1a4ad5e62e0a1ae3b83e6 +0 -0
  222. data/test/dummy/tmp/cache/assets/D9A/2F0/sprockets%2Fbba0f4b972dc53c15ce6c8c1993b82a4 +0 -0
  223. data/test/dummy/tmp/cache/assets/D9C/860/sprockets%2Fec2d9f20b270d70e698ff33e53c21fca +0 -0
  224. data/test/dummy/tmp/cache/assets/DA6/A80/sprockets%2F92e26d8e58d5bcc8b8f6c25d1b05b9c1 +0 -0
  225. data/test/dummy/tmp/cache/assets/DA8/BB0/sprockets%2F0cf1c7e9f966dce425517c0e2a844efe +0 -0
  226. data/test/dummy/tmp/cache/assets/DA9/BC0/sprockets%2Fcb9062b73291befe5e5bf2a70978dec8 +0 -0
  227. data/test/dummy/tmp/cache/assets/DB6/ED0/sprockets%2Ff4482d9b9f76fb65eef16430bde2f8e6 +0 -0
  228. data/test/dummy/tmp/cache/assets/DDC/400/sprockets%2Fcffd775d018f68ce5dba1ee0d951a994 +0 -0
  229. data/test/dummy/tmp/cache/assets/DE8/790/sprockets%2Fd1333bde2b9aafcc712d11dd09ab35d8 +0 -0
  230. data/test/dummy/tmp/cache/assets/DF7/960/sprockets%2F99ac6db10b44a64fbba4ee847b35ba8b +0 -0
  231. data/test/dummy/tmp/cache/assets/DFD/300/sprockets%2Fabac9489cf7f1db8ef00d72a1571ee1e +0 -0
  232. data/test/dummy/tmp/cache/assets/E02/C10/sprockets%2F1463cadfce3fc70e61d482f9fcb75ac7 +0 -0
  233. data/test/dummy/tmp/cache/assets/E04/890/sprockets%2F2f5173deea6c795b8fdde723bb4b63af +0 -0
  234. data/test/dummy/tmp/cache/assets/E38/FE0/sprockets%2Fe1fc875efa817cbb94a5d8de25ea4e6b +0 -0
  235. data/test/dummy/tmp/cache/assets/E50/F00/sprockets%2Fec61afdbb1cc4df075e3dad46f0af571 +0 -0
  236. data/test/dummy/tmp/cache/assets/E5E/5F0/sprockets%2Feffaeb4d7f9bf4c0cc840ff320fde046 +0 -0
  237. data/test/dummy/tmp/cache/assets/E5F/960/sprockets%2Fdc007b6cad5c7ef08e33ec28cfff0ef6 +0 -0
  238. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_accordion.scssc +0 -0
  239. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_alerts.scssc +0 -0
  240. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_breadcrumbs.scssc +0 -0
  241. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_button-groups.scssc +0 -0
  242. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_buttons.scssc +0 -0
  243. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_carousel.scssc +0 -0
  244. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_close.scssc +0 -0
  245. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_code.scssc +0 -0
  246. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_component-animations.scssc +0 -0
  247. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_dropdowns.scssc +0 -0
  248. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_forms.scssc +0 -0
  249. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_grid.scssc +0 -0
  250. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_hero-unit.scssc +0 -0
  251. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_labels-badges.scssc +0 -0
  252. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_layouts.scssc +0 -0
  253. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_media.scssc +0 -0
  254. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_mixins.scssc +0 -0
  255. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_modals.scssc +0 -0
  256. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_navbar.scssc +0 -0
  257. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_navs.scssc +0 -0
  258. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_pager.scssc +0 -0
  259. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_pagination.scssc +0 -0
  260. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_popovers.scssc +0 -0
  261. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_progress-bars.scssc +0 -0
  262. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_reset.scssc +0 -0
  263. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_responsive-1200px-min.scssc +0 -0
  264. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_responsive-767px-max.scssc +0 -0
  265. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_responsive-768px-979px.scssc +0 -0
  266. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_responsive-navbar.scssc +0 -0
  267. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_responsive-utilities.scssc +0 -0
  268. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_scaffolding.scssc +0 -0
  269. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_sprites.scssc +0 -0
  270. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_tables.scssc +0 -0
  271. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_thumbnails.scssc +0 -0
  272. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_tooltip.scssc +0 -0
  273. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_type.scssc +0 -0
  274. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_utilities.scssc +0 -0
  275. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_variables.scssc +0 -0
  276. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/_wells.scssc +0 -0
  277. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/bootstrap.scssc +0 -0
  278. data/test/dummy/tmp/cache/sass/310d901c169042855feb8cf3903e1e2821d27f13/responsive.scssc +0 -0
  279. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_admin.scssc +0 -0
  280. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_az.scssc +0 -0
  281. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_base.scssc +0 -0
  282. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_forms.scssc +0 -0
  283. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_icons.scssc +0 -0
  284. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_misc.scssc +0 -0
  285. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_mixins.scssc +0 -0
  286. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_modal.scssc +0 -0
  287. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_resolve.scssc +0 -0
  288. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_results.scssc +0 -0
  289. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_search.scssc +0 -0
  290. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_spinner.scssc +0 -0
  291. data/test/dummy/tmp/cache/sass/3af72421a0fbed39d3047687ca0ad6e6e5545041/_variables.scssc +0 -0
  292. data/test/dummy/tmp/cache/sass/93d65682d6bddb0b2e6788c1232f4849af20a35e/bootstrap-responsive.scssc +0 -0
  293. data/test/dummy/tmp/cache/sass/93d65682d6bddb0b2e6788c1232f4849af20a35e/bootstrap.scssc +0 -0
  294. data/test/dummy/tmp/cache/sass/fb58c04fd15fcc38a8c4d91b7070bcfeaf1c3799/umlaut.css.scssc +0 -0
  295. data/test/fixtures/dispatched_services.yml +0 -4
  296. data/test/fixtures/permalinks.yml +23 -4
  297. data/test/fixtures/referent_values.yml +226 -1493
  298. data/test/fixtures/referents.yml +14 -137
  299. data/test/fixtures/requests.yml +22 -260
  300. data/test/fixtures/service_responses.yml +169 -5
  301. data/test/fixtures/sfx4/global/KB_OBJECTS.yml +46 -0
  302. data/test/fixtures/sfx4/local/AZ_EXTRA_INFO.yml +4802 -0
  303. data/test/fixtures/sfx4/local/AZ_LETTER_GROUP.yml +105 -0
  304. data/test/fixtures/sfx4/local/AZ_TITLE.yml +165 -0
  305. data/test/fixtures/sfx4/local/AZ_TITLE_SEARCH.yml +341 -0
  306. data/test/fixtures/sfx_urls.yml +3 -0
  307. data/test/functional/export_email_controller_test.rb +43 -0
  308. data/test/functional/link_router_controller_test.rb +16 -0
  309. data/test/functional/resolve_controller_test.rb +102 -0
  310. data/test/functional/search_controller_test.rb +40 -0
  311. data/test/functional/store_controller_test.rb +29 -0
  312. data/test/helper/list_with_limit_test.rb +90 -0
  313. data/test/integration/permalinks_test.rb +14 -0
  314. data/test/support/search_methods/test_case.rb +0 -0
  315. data/test/support/service_adaptors/test_case.rb +0 -0
  316. data/test/support/test_with_cassette.rb +37 -0
  317. data/test/test_helper.rb +76 -30
  318. data/test/unit/aws_product_sign_test.rb +2 -0
  319. data/test/unit/determine_services_test.rb +163 -0
  320. data/test/unit/google_book_search_test.rb +8 -25
  321. data/test/unit/google_scholar_link_test.rb +81 -0
  322. data/test/unit/internet_archive_test.rb +33 -0
  323. data/test/unit/permalink_test.rb +15 -0
  324. data/test/unit/service_store_test.rb +47 -0
  325. data/test/unit/service_test.rb +28 -13
  326. data/test/unit/sfx/sfx_target_roll_up_test.rb +369 -0
  327. data/test/unit/sfx4_search_test.rb +102 -0
  328. data/test/unit/sfx4_solr_search_test.rb +77 -0
  329. data/test/unit/sfx_test.rb +38 -0
  330. data/test/unit/worldcat_test.rb +7 -0
  331. data/test/vcr_cassettes/google_book_search/frankenstein_by_OCLC_number.yml +107 -0
  332. data/test/vcr_cassettes/internet_archive/momo_by_title_author.yml +116 -0
  333. data/test/vcr_cassettes/resolve/fulltext_with_edition_warning.yml +44 -0
  334. data/test/vcr_cassettes/resolve/no_holdings.yml +649 -0
  335. data/test/vcr_cassettes/resolve/nytimes_by_issn.yml +224 -0
  336. data/test/vcr_cassettes/sfx/nytimes_by_issn.yml +1395 -0
  337. data/test/vcr_cassettes/sfx4_solr/find_by_group.yml +109 -0
  338. data/test/vcr_cassettes/sfx4_solr/find_by_title_begins_with.yml +33 -0
  339. data/test/vcr_cassettes/sfx4_solr/find_by_title_contains.yml +34 -0
  340. data/test/vcr_cassettes/sfx4_solr/find_by_title_exact.yml +32 -0
  341. data/test/view/holding_test.rb +30 -0
  342. metadata +435 -179
  343. data/app/assets/images/export_bg_bot.gif +0 -0
  344. data/app/assets/images/export_bg_mid.gif +0 -0
  345. data/app/assets/images/export_bg_top.gif +0 -0
  346. data/app/assets/images/famfamfam/page_sound.gif +0 -0
  347. data/app/assets/images/famfamfam/page_text.gif +0 -0
  348. data/app/assets/images/famfamfam/page_up.gif +0 -0
  349. data/app/assets/images/ico_go.gif +0 -0
  350. data/app/assets/images/more_info.gif +0 -0
  351. data/app/assets/images/rails.png +0 -0
  352. data/app/assets/images/request.gif +0 -0
  353. data/app/assets/javascripts/umlaut/simple_visible_toggle.js +0 -8
  354. data/app/assets/stylesheets/umlaut.css +0 -900
  355. data/app/assets/stylesheets/umlaut/_background_progress_spinner.scss +0 -21
  356. data/app/assets/stylesheets/umlaut/_expand_contract.scss +0 -28
  357. data/app/controllers/search_methods/sfx3.rb +0 -148
  358. data/app/models/sfx_db.rb +0 -10
  359. data/app/models/sfx_db/az_additional_title.rb +0 -11
  360. data/app/models/sfx_db/az_letter_group.rb +0 -11
  361. data/app/models/sfx_db/az_title.rb +0 -40
  362. data/app/models/sfx_db/az_title_v2.rb +0 -34
  363. data/app/models/sfx_db/isbn.rb +0 -12
  364. data/app/models/sfx_db/issn.rb +0 -12
  365. data/app/models/sfx_db/object.rb +0 -35
  366. data/app/models/sfx_db/object_portfolio.rb +0 -6
  367. data/app/models/sfx_db/publisher.rb +0 -10
  368. data/app/models/sfx_db/sfx_db_base.rb +0 -58
  369. data/app/models/sfx_db/target.rb +0 -10
  370. data/app/models/sfx_db/target_service.rb +0 -10
  371. data/app/models/sfx_db/title.rb +0 -10
  372. data/app/views/export_email/_email.html.erb +0 -25
  373. data/app/views/export_email/_send_email.html.erb +0 -3
  374. data/app/views/export_email/_send_txt.html.erb +0 -3
  375. data/app/views/export_email/_txt.html.erb +0 -62
  376. data/lib/holding.rb +0 -32
  377. data/lib/opensearch_feed.rb +0 -398
  378. data/lib/opensearch_query.rb +0 -98
  379. data/lib/referent_filters/dissertation_catch.rb +0 -45
  380. data/lib/service_adaptors/journal_tocs.progress +0 -68
  381. data/lib/service_adaptors/primo_service.rb +0 -452
  382. data/lib/service_adaptors/primo_source.rb +0 -82
  383. data/lib/umlaut/default_configuration.rb +0 -5
  384. data/test/dummy/config/primo.yml +0 -271
  385. data/test/unit/primo_service_test.rb +0 -947
@@ -0,0 +1,28 @@
1
+ SearchMethods
2
+ ===
3
+ SearchMethods' allow Umlaut to search an arbitrary data source for A-Z Journals.
4
+ Implementations must adhere to the following method signatures.
5
+ The module is included in the SearchController and has access to instance methods and
6
+ instance variables from the SearchController.
7
+
8
+ Class Methods:
9
+ ---
10
+ These class methods are used only in the `:load_sfx_urls` rake task.
11
+ They are expected to be deprecated in future Umlaut releases.
12
+
13
+ - `::fetch_urls?() `==> `Boolean` <br>
14
+ Returns a Boolean indicating whether the module will return URLs
15
+
16
+ - `::fetch_urls()` ==> `Array<String>` <br>
17
+ Returns an array of strings representing URLs that are "owned" by the search
18
+ system
19
+
20
+ Instance Methods:
21
+ ---
22
+ - `#find_by_title()` ==> `[Array<OpenURL::ContextObject>, integer]` <br>
23
+ Returns a two element array consisting of an Array of OpenURL::ContextObject
24
+ and the total count of records returned by the search
25
+
26
+ - `#find_by_group()` ==> `[Array<OpenURL::ContextObject>, integer]` <br>
27
+ Returns a two element array consisting of an Array of OpenURL::ContextObject
28
+ and the total count of records returned by the search
@@ -1,81 +1,47 @@
1
1
  require 'nokogiri'
2
-
3
2
  module SearchMethods
4
3
  module Sfx4
5
4
  include MetadataHelper # for normalize_lccn
6
-
5
+
7
6
  protected
8
-
9
- # used by umlaut:load_sfx_urls task. Kind of hacky way of trying to extract
10
- # target URLs from SFX4.
11
- def self.fetch_sfx_urls(sfx_global_db = "sfxglb41")
12
- connection = SfxDb::Object.connection
7
+ # Class method for the module that gets called by the umlaut:load_sfx_urls task.
8
+ # Determines whether we should attempt to fetch SFX urls.
9
+ # Will probably be deprecated in the near future.
10
+ def self.fetch_urls?
11
+ sfx4_base.connection_configured?
12
+ end
13
13
 
14
- # Crazy crazy URLs to try to find PARSE_PARAMS in Sfx4 db that have a period in
15
- # them, so they look like they might be URLs. Parse params could be at target service
16
- # level, or at portfolio level; and could be in local overrides or in global kb.
17
- # This is crazy crazy SQL to get this, sorry. Talking directly to SFX db isn't
18
- # a great idea, but best way we've found to get this for now. Might make more
19
- # sense to try to use the (very very slow) SFX export in the future instead.
20
- sql = %{
21
- SELECT
22
- COALESCE(LCL_SERVICE_LINKING_INFO.PARSE_PARAM,KB_TARGET_SERVICES.PARSE_PARAM) PARSE_PARAM
23
- FROM
24
- LCL_TARGET_INVENTORY
25
- JOIN #{sfx_global_db}.KB_TARGET_SERVICES
26
- ON KB_TARGET_SERVICES.TARGET_ID = LCL_TARGET_INVENTORY.TARGET_ID
27
- JOIN LCL_SERVICE_INVENTORY
28
- ON LCL_TARGET_INVENTORY.TARGET_ID = LCL_SERVICE_INVENTORY.TARGET_ID
29
- LEFT OUTER JOIN LCL_SERVICE_LINKING_INFO
30
- ON LCL_SERVICE_INVENTORY.TARGET_SERVICE_ID = LCL_SERVICE_LINKING_INFO.TARGET_SERVICE_ID
31
- WHERE
32
- ( LCL_SERVICE_LINKING_INFO.PARSE_PARAM like '%.%' OR
33
- KB_TARGET_SERVICES.PARSE_PARAM like '%.%' )
34
- AND
35
- LCL_SERVICE_INVENTORY.ACTIVATION_STATUS='ACTIVE'
36
- AND
37
- LCL_TARGET_INVENTORY.ACTIVATION_STATUS = 'ACTIVE'
38
-
39
- UNION
40
- -- object portfolio parse param version
41
- SELECT
42
- COALESCE(LCL_OBJECT_PORTFOLIO_LINKING_INFO.PARSE_PARAM, KB_OBJECT_PORTFOLIOS.PARSE_PARAM) PARSE_PARAM
43
- FROM
44
- #{sfx_global_db}.KB_OBJECT_PORTFOLIOS
45
- JOIN LCL_SERVICE_INVENTORY
46
- ON KB_OBJECT_PORTFOLIOS.TARGET_SERVICE_ID = LCL_SERVICE_INVENTORY.TARGET_SERVICE_ID
47
- JOIN LCL_OBJECT_PORTFOLIO_INVENTORY
48
- ON KB_OBJECT_PORTFOLIOS.OP_ID = LCL_OBJECT_PORTFOLIO_INVENTORY.OP_ID
49
- left outer join LCL_OBJECT_PORTFOLIO_LINKING_INFO
50
- ON KB_OBJECT_PORTFOLIOS.OP_ID = LCL_OBJECT_PORTFOLIO_LINKING_INFO.OP_ID
51
- WHERE
52
- ( KB_OBJECT_PORTFOLIOS.PARSE_PARAM like '%.%' OR
53
- LCL_OBJECT_PORTFOLIO_LINKING_INFO.PARSE_PARAM like '%.%' )
54
- AND LCL_OBJECT_PORTFOLIO_INVENTORY.ACTIVATION_STATUS = 'ACTIVE'
55
- AND LCL_SERVICE_INVENTORY.ACTIVATION_STATUS='ACTIVE'
56
- }
57
-
58
- results = connection.select_all(sql)
59
-
60
- urls = []
61
- results.each do |line|
62
- param_string = line["PARSE_PARAM"]
63
-
64
- # Try to get things that look sort of like URLs out. Brutal force,
65
- # sorry.
66
- url_re = Regexp.new('(https?://\S+\.\S+)(\s|$)')
67
- urls.concat( param_string.scan( url_re ).collect {|matches| matches[0]} )
68
- end
69
- urls.uniq!
70
- return urls
14
+ # Class method for the module that gets called by the umlaut:load_sfx_urls task.
15
+ # Kind of hacky way of trying to extract target URLs from SFX4.
16
+ # Will probably be deprecated in the near future.
17
+ def self.fetch_urls
18
+ sfx4_base.fetch_urls
71
19
  end
72
20
 
73
-
21
+ # Class method for the module.
22
+ # Returns the SFX4 base class in order to establish a connection.
23
+ def self.sfx4_base
24
+ # Need to do this convoluted Module.const_get so that we find the
25
+ # correct class. Otherwise the module looks locally and can't find it.
26
+ Module.const_get(:Sfx4).const_get(:Local).const_get(:Base)
27
+ end
28
+
29
+ # Instance method that returns the SFX4 AzTitle class for this search method.
30
+ # Can be overridden by search methods that want to include this one.
31
+ def az_title_klass
32
+ # Need to do this convoluted Module.const_get so that we find the
33
+ # correct class. Otherwise the module looks locally and can't find it.
34
+ Module.const_get(:Sfx4).const_get(:Local).const_get(:AzTitle)
35
+ end
36
+
37
+ # Instance method that returns the SFX4 DB connection for this search method.
38
+ def sfx4_db_connection
39
+ az_title_klass.connection
40
+ end
41
+
74
42
  # Needs to return ContextObjects
75
43
  def find_by_title
76
44
  connection = sfx4_db_connection
77
-
78
-
79
45
  query_match_clause = case search_type_param
80
46
  when "contains"
81
47
  terms = title_query_param.split(" ")
@@ -88,7 +54,6 @@ module SearchMethods
88
54
  query = terms.collect do |term|
89
55
  "+" + connection.quote_string(term) + "*"
90
56
  end.join(" ")
91
-
92
57
  "MATCH (TS.TITLE_SEARCH) AGAINST ('#{query}' IN BOOLEAN MODE)"
93
58
  when "begins"
94
59
  # For 'begins', searching against TITLE itself rather than TITLE_SEARCH gives us
@@ -102,7 +67,6 @@ module SearchMethods
102
67
  T.TITLE_SORT = '#{connection.quote_string(title_query_param)}'
103
68
  )"
104
69
  end.upcase
105
-
106
70
  from_where_clause = %{
107
71
  FROM
108
72
  AZ_TITLE T, AZ_TITLE_SEARCH TS
@@ -111,7 +75,6 @@ module SearchMethods
111
75
  #{query_match_clause} AND
112
76
  T.AZ_PROFILE = '#{connection.quote_string(sfx_az_profile)}'
113
77
  }
114
-
115
78
  statement = %{
116
79
  SELECT
117
80
  DISTINCT T.OBJECT_ID
@@ -120,16 +83,12 @@ module SearchMethods
120
83
  T.SCRIPT DESC, T.TITLE_SORT
121
84
  LIMIT #{batch_size.to_i}
122
85
  OFFSET #{(batch_size * (page - 1)).to_i}
123
- }
124
-
86
+ }
125
87
  # do the count
126
- total_hits = SfxDb::Object.count_by_sql(
127
- "SELECT COUNT(DISTINCT(T.OBJECT_ID)) #{from_where_clause}"
128
- )
129
-
88
+ total_hits = az_title_klass.count_by_sql(
89
+ "SELECT COUNT(DISTINCT(T.OBJECT_ID)) #{from_where_clause}")
130
90
  object_ids = connection.select_all(statement).collect {|i| i.values.first}
131
-
132
- sql = SfxDb::Object.send(:sanitize_sql_array,
91
+ sql = az_title_klass.send(:sanitize_sql_array,
133
92
  [%{
134
93
  SELECT
135
94
  EI.OBJECT_ID, T.TITLE_DISPLAY, EI.EXTRA_INFO_XML
@@ -143,41 +102,32 @@ module SearchMethods
143
102
  ORDER BY
144
103
  T.SCRIPT DESC, T.TITLE_SORT
145
104
  },
146
- sfx_az_profile,
147
- object_ids])
148
-
105
+ sfx_az_profile, object_ids])
149
106
  title_objects = connection.select_all(sql)
150
-
151
107
  # Make em into context objects
152
108
  context_objects = title_objects.collect do |sfx_obj|
153
109
  ctx = OpenURL::ContextObject.new
154
110
  # Start out wtih everything in search, to preserve date/vol/etc
155
111
  ctx.import_context_object( context_object_from_params )
156
-
157
112
  extra_info_xml = Nokogiri::XML( sfx_obj["EXTRA_INFO_XML"] )
158
-
159
113
  # Put SFX object id in rft.object_id, that's what SFX does.
160
114
  ctx.referent.set_metadata('object_id', sfx_obj["OBJECT_ID"].to_s )
161
115
  ctx.referent.set_metadata("jtitle", sfx_obj["TITLE_DISPLAY"] || "Unknown Title")
162
-
163
116
  issn = extra_info_xml.search("item[key=issn]").text
164
117
  isbn = extra_info_xml.search("item[key=isbn]").text
165
-
166
118
  # LCCN is stored corrupted in xml in SFX db, without prefix like "sn" that
167
119
  # is a significant part of lccn. Our reverse engineering of SFX failed,
168
120
  # apparently there's a workaround in SFX app code. Forget it, bail
169
121
  # don't try to use lccn.
170
122
  #lccn = extra_info_xml.search("item[key=lccn]").text
171
-
172
123
  ctx.referent.set_metadata("issn", issn ) unless issn.blank?
173
124
  ctx.referent.set_metadata("isbn", isbn) unless isbn.blank?
174
125
  #ctx.referent.add_identifier("info:lccn/#{normalize_lccn(lccn)}") unless lccn.blank?
175
-
176
126
  ctx
177
127
  end
178
128
  return [context_objects, total_hits]
179
129
  end
180
-
130
+
181
131
  # Used for clicks on A, B, C, 0-9, etc.
182
132
  def find_by_group
183
133
  connection = sfx4_db_connection
@@ -194,69 +144,50 @@ module SearchMethods
194
144
  }
195
145
  count_sql = %{
196
146
  SELECT count(*)
197
-
198
147
  #{from_where_clause}
199
148
  }
200
-
201
149
  fetch_sql = %{
202
150
  SELECT
203
151
  EI.OBJECT_ID, T.TITLE_DISPLAY, EI.EXTRA_INFO_XML
204
-
205
152
  #{from_where_clause}
206
-
207
153
  ORDER BY
208
154
  T.SCRIPT DESC, T.TITLE_SORT
209
155
  LIMIT #{batch_size.to_i}
210
156
  OFFSET #{(batch_size * (page - 1)).to_i}
211
157
  }
212
-
213
- total_count = SfxDb::Object.count_by_sql( count_sql )
158
+ total_count = az_title_klass.count_by_sql( count_sql )
214
159
  context_objects = sfx4_db_to_ctxobj( connection.select_all(fetch_sql) )
215
-
216
160
  return [context_objects, total_count]
217
161
  end
218
-
219
- def sfx4_db_connection
220
- SfxDb::Object.connection
221
- end
222
-
162
+
223
163
  def sfx4_quoted_letter_group_condition
224
- " AZ_LETTER_GROUP.AZ_LETTER_GROUP_NAME " +
225
- case params[:id]
226
- when "0-9"
227
- " IN ('0','1','2','3','4','5','6','7','8','9')"
228
- when /^Other/i
229
- "= 'Others'"
230
- else
231
- "= '#{sfx4_db_connection.quote_string(params[:id].upcase)}'"
232
- end
164
+ " AZ_LETTER_GROUP.AZ_LETTER_GROUP_NAME " + case params[:id]
165
+ when "0-9"
166
+ " IN ('0','1','2','3','4','5','6','7','8','9')"
167
+ when /^Other/i
168
+ "= 'Others'"
169
+ else
170
+ "= '#{sfx4_db_connection.quote_string(params[:id].upcase)}'"
171
+ end
233
172
  end
234
-
173
+
235
174
  def sfx4_db_to_ctxobj(title_rows)
236
175
  title_rows.collect do |sfx_obj|
237
176
  ctx = OpenURL::ContextObject.new
238
177
  # Start out wtih everything in search, to preserve date/vol/etc
239
178
  ctx.import_context_object( context_object_from_params )
240
-
241
179
  extra_info_xml = Nokogiri::XML( sfx_obj["EXTRA_INFO_XML"] )
242
-
243
180
  # Put SFX object id in rft.object_id, that's what SFX does.
244
181
  ctx.referent.set_metadata('object_id', sfx_obj["OBJECT_ID"])
245
182
  ctx.referent.set_metadata("jtitle", sfx_obj["TITLE_DISPLAY"] || "Unknown Title")
246
-
247
183
  issn = extra_info_xml.search("item[key=issn]").text
248
184
  isbn = extra_info_xml.search("item[key=isbn]").text
249
185
  lccn = extra_info_xml.search("item[key=lccn]").text
250
-
251
186
  ctx.referent.set_metadata("issn", issn ) unless issn.blank?
252
187
  ctx.referent.set_metadata("isbn", isbn) unless isbn.blank?
253
188
  ctx.referent.add_identifier("info:lccn/#{normalize_lccn(lccn)}") unless lccn.blank?
254
-
255
189
  ctx
256
190
  end
257
-
258
191
  end
259
-
260
192
  end
261
193
  end
262
-
@@ -0,0 +1,57 @@
1
+ Sfx4Solr
2
+ ---
3
+ This searcher module leverages [Sunspot](http://sunspot.github.com/) to index SFX AZ Titles in Solr.
4
+
5
+ In order to use this module, you first need to set up Sunspot in your applications by adding the sunspot_rails
6
+ gem to you Gemfile
7
+
8
+ gem 'sunspot_rails'
9
+
10
+ and requiring sunspot_rails in your config/application.rb under umlaut
11
+
12
+ require "umlaut"
13
+ require "sunspot_rails"
14
+
15
+ Then run
16
+
17
+ $ bundle install
18
+ $ rails generate sunspot_rails:install
19
+
20
+ This will create a file config/sunspot.yml. You can configure the information about your Solr instance here.
21
+ If you don't feel like running your own solr instance, [Websolr](http://websolr.com/) has built in Sunspot support.
22
+
23
+ You'll also need to specify your SFX4 DBs in your databases.yml. The configuration names are different from the Sfx4
24
+ SearchMethod in order to avoid (promote?) ambiguity and confusion.
25
+
26
+ sfx4_global:
27
+ adapter: mysql2
28
+ host: sfx.library.edu
29
+ port: 3310
30
+ database: sfxglb41
31
+ username:
32
+ password:
33
+ pool: 30
34
+ encoding: utf8
35
+
36
+ sfx_db:
37
+ adapter: mysql2
38
+ host: sfx.library.edu
39
+ port: 3310
40
+ database: sfxlcl41
41
+ username:
42
+ password:
43
+ pool: 30
44
+ encoding: utf8
45
+
46
+ Once your Solr instance is configured and started, run
47
+
48
+ $ rake sunspot:reindex[,Sfx4::Local::AzTitle]
49
+
50
+ to index the AZ titles in Solr. This reindex should probably be run periodically and coincide with SFX KB updates.
51
+
52
+ After the reindex is complete, add this to your UmlautController search config
53
+
54
+ az_search_method SearchMethods::Sfx4Solr::Local
55
+
56
+ and you should be up and running with SFX indexed in Solr.
57
+
@@ -0,0 +1,40 @@
1
+ module SearchMethods
2
+ module Sfx4Solr
3
+ module Local
4
+ protected
5
+ # Class method for the module that gets called by the umlaut:load_sfx_urls task.
6
+ # Determines whether we should attempt to fetch SFX urls.
7
+ # Will probably be deprecated in the near future.
8
+ def self.fetch_urls?
9
+ sfx4_base.connection_configured?
10
+ end
11
+
12
+ # Class method for the module that gets called by the umlaut:load_sfx_urls task.
13
+ # Kind of hacky way of trying to extract target URLs from SFX4.
14
+ # Will probably be deprecated in the near future.
15
+ def self.fetch_urls
16
+ sfx4_base.fetch_urls
17
+ end
18
+
19
+ # Class method for the module.
20
+ # Returns the SFX4 base class in order to establish a connection.
21
+ def self.sfx4_base
22
+ # Need to do this convoluted Module.const_get so that we find the
23
+ # correct class. Otherwise the module looks locally and can't find it.
24
+ Module.const_get(:Sfx4).const_get(:Local).const_get(:Base)
25
+ end
26
+
27
+ # Instance method that returns the SFX4 AzTitle class for this search method.
28
+ # Can be overridden by search methods that want to include this one.
29
+ def az_title_klass
30
+ # Need to do this convoluted Module.const_get so that we find the
31
+ # correct class. Otherwise the module looks locally and can't find it.
32
+ Module.const_get(:Sfx4).const_get(:Local).const_get(:AzTitle)
33
+ end
34
+
35
+ # Include the core sfx4solr search methods.
36
+ # Search class is determined by az_title_klass
37
+ include SearchMethods::Sfx4Solr::Searcher
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,90 @@
1
+ module SearchMethods
2
+ module Sfx4Solr
3
+ module Searcher
4
+ protected
5
+ def find_by_title
6
+ raise_sunspot_not_implemented_exception unless sunspot?
7
+ _find_by_title(title_query_param, search_type_param, context_object_from_params, page)
8
+ end
9
+
10
+ def find_by_group
11
+ raise_sunspot_not_implemented_exception unless sunspot?
12
+ _find_by_group(_letter_group_param, context_object_from_params, page)
13
+ end
14
+
15
+ private
16
+ # Is Umlaut configured to use Sunspot?
17
+ def sunspot?
18
+ begin
19
+ Sunspot and Sunspot.const_defined?(:Rails)
20
+ rescue
21
+ false
22
+ end
23
+ end
24
+
25
+ # Raise a NotImplementedError is Sunspot
26
+ def raise_sunspot_not_implemented_exception
27
+ raise NotImplementedError.new("Sunspot::Rails has not been implemented in this Umlaut instance.")
28
+ end
29
+
30
+ def _search_by_title(query, search_type, page=1)
31
+ search = case search_type
32
+ when "contains"
33
+ az_title_klass.search {
34
+ keywords query, :fields => [:title]
35
+ order_by(:title_sort, :asc)
36
+ paginate(:page => page, :per_page => 20)
37
+ }
38
+ when "begins"
39
+ az_title_klass.search {
40
+ with(:title_exact).starting_with(query)
41
+ order_by(:title_sort, :asc)
42
+ paginate(:page => page, :per_page => 20)
43
+ }
44
+ else # exact
45
+ az_title_klass.search {
46
+ with(:title_exact, query)
47
+ order_by(:title_sort, :asc)
48
+ paginate(:page => page, :per_page => 20)
49
+ }
50
+ end
51
+ end
52
+
53
+ def _find_by_title(query, search_type, context_object, page=1)
54
+ search = _search_by_title(query, search_type, page)
55
+ return [search.hits.map{|hit| _to_context_object(hit, context_object)}, search.total]
56
+ end
57
+
58
+ def _find_by_group(letter_group, context_object, page=1)
59
+ search = az_title_klass.search {
60
+ with(:letter_group, letter_group)
61
+ order_by(:title_sort, :asc)
62
+ paginate(:page => page, :per_page => 20)
63
+ }
64
+ return [search.hits.map{|hit| _to_context_object(hit, context_object)}, search.total]
65
+ end
66
+
67
+ def _to_context_object(hit, context_object)
68
+ ctx = OpenURL::ContextObject.new
69
+ # Start out wtih everything in search, to preserve date/vol/etc
70
+ ctx.import_context_object(context_object)
71
+ # Put SFX object id in rft.object_id, that's what SFX does.
72
+ ctx.referent.set_metadata('object_id', hit.stored(:object_id) )
73
+ ctx.referent.set_metadata("jtitle", hit.stored(:title_display) || "Unknown Title")
74
+ ctx.referent.set_metadata("issn", hit.stored(:issn)) unless hit.stored(:issn).nil? or hit.stored(:issn).blank?
75
+ ctx.referent.set_metadata("isbn", hit.stored(:isbn)) unless hit.stored(:isbn).nil? or hit.stored(:isbn).blank?
76
+ ctx.referent.add_identifier("info:lccn/#{hit.stored(:lccn)}") unless hit.stored(:lccn).nil? or hit.stored(:lccn).blank?
77
+ return ctx
78
+ end
79
+
80
+ def _letter_group_param
81
+ case params[:id]
82
+ when /^Other/i
83
+ "Others"
84
+ else
85
+ "#{params[:id].upcase}"
86
+ end
87
+ end
88
+ end
89
+ end
90
+ end