blacklight-spotlight 3.0.0.rc3 → 3.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (257) hide show
  1. checksums.yaml +4 -4
  2. data/app/assets/images/blacklight/close.svg +1 -0
  3. data/app/assets/images/blacklight/zoom_in.svg +1 -0
  4. data/app/assets/images/blacklight/zoom_out.svg +1 -0
  5. data/app/assets/javascripts/spotlight/admin/blocks/uploaded_items_block.js +8 -1
  6. data/app/assets/javascripts/spotlight/admin/catalog_edit.js +13 -47
  7. data/app/assets/javascripts/spotlight/admin/exhibit_tag_autocomplete.js +37 -0
  8. data/app/assets/javascripts/spotlight/admin/{reindex_monitor.js → progress_monitor.js} +26 -4
  9. data/app/assets/javascripts/spotlight/admin/sir-trevor/locales.js +1 -1
  10. data/app/assets/javascripts/spotlight/user/zpr_links.js.erb +29 -14
  11. data/app/assets/stylesheets/spotlight/_browse.scss +8 -0
  12. data/app/assets/stylesheets/spotlight/_catalog.scss +30 -4
  13. data/app/assets/stylesheets/spotlight/_exhibit_admin.scss +9 -0
  14. data/app/assets/stylesheets/spotlight/_modals.scss +3 -0
  15. data/app/assets/stylesheets/spotlight/_nestable.scss +8 -0
  16. data/app/assets/stylesheets/spotlight/_spotlight.scss +2 -0
  17. data/app/assets/stylesheets/spotlight/_view_larger.scss +22 -0
  18. data/app/assets/stylesheets/spotlight/browse_group_categories_block.scss +23 -0
  19. data/app/assets/stylesheets/spotlight/typeahead.css +23 -23
  20. data/app/builders/spotlight/bootstrap_breadcrumbs_builder.rb +4 -2
  21. data/app/controllers/concerns/spotlight/base.rb +1 -1
  22. data/app/controllers/concerns/spotlight/catalog.rb +7 -1
  23. data/app/controllers/spotlight/browse_controller.rb +9 -15
  24. data/app/controllers/spotlight/bulk_actions_controller.rb +62 -0
  25. data/app/controllers/spotlight/bulk_updates_controller.rb +67 -0
  26. data/app/controllers/spotlight/catalog_controller.rb +16 -13
  27. data/app/controllers/spotlight/dashboards_controller.rb +5 -4
  28. data/app/controllers/spotlight/exhibits_controller.rb +1 -1
  29. data/app/controllers/spotlight/featured_images_controller.rb +1 -1
  30. data/app/controllers/spotlight/job_trackers_controller.rb +17 -0
  31. data/app/controllers/spotlight/pages_controller.rb +6 -7
  32. data/app/controllers/spotlight/tags_controller.rb +39 -5
  33. data/app/helpers/spotlight/application_helper.rb +20 -1
  34. data/app/helpers/spotlight/job_trackers_helper.rb +31 -0
  35. data/app/helpers/spotlight/main_app_helpers.rb +3 -4
  36. data/app/helpers/spotlight/meta_helper.rb +2 -2
  37. data/app/helpers/spotlight/pages_helper.rb +1 -8
  38. data/app/helpers/spotlight/roles_helper.rb +1 -1
  39. data/app/helpers/spotlight/searches_helper.rb +1 -1
  40. data/app/jobs/concerns/spotlight/gather_documents.rb +35 -0
  41. data/app/jobs/concerns/spotlight/job_tracking.rb +64 -0
  42. data/app/jobs/concerns/spotlight/limit_concurrency.rb +33 -0
  43. data/app/jobs/spotlight/add_tags_job.rb +31 -0
  44. data/app/jobs/spotlight/add_uploads_from_csv.rb +4 -3
  45. data/app/jobs/spotlight/application_job.rb +8 -0
  46. data/app/jobs/spotlight/change_visibility_job.rb +33 -0
  47. data/app/jobs/spotlight/cleanup_job_trackers_job.rb +13 -0
  48. data/app/jobs/spotlight/default_thumbnail_job.rb +1 -3
  49. data/app/jobs/spotlight/process_bulk_updates_csv_job.rb +82 -0
  50. data/app/jobs/spotlight/reindex_exhibit_job.rb +39 -0
  51. data/app/jobs/spotlight/reindex_job.rb +64 -44
  52. data/app/jobs/spotlight/remove_tags_job.rb +31 -0
  53. data/app/jobs/spotlight/rename_sidecar_field_job.rb +3 -2
  54. data/app/jobs/spotlight/rename_tags_job.rb +33 -0
  55. data/app/jobs/spotlight/update_job_trackers_job.rb +20 -0
  56. data/app/models/concerns/spotlight/browse_category_search_builder.rb +59 -0
  57. data/app/models/concerns/spotlight/exhibit_analytics.rb +2 -6
  58. data/app/models/concerns/spotlight/exhibit_documents.rb +2 -2
  59. data/app/models/concerns/spotlight/resources/web.rb +1 -1
  60. data/app/models/concerns/spotlight/search_builder.rb +11 -0
  61. data/app/models/concerns/spotlight/solr_document.rb +1 -1
  62. data/app/models/concerns/spotlight/solr_document/atomic_updates.rb +2 -2
  63. data/app/models/concerns/spotlight/user.rb +7 -2
  64. data/app/models/sir_trevor_rails/blocks/solr_documents_block.rb +9 -0
  65. data/app/models/sir_trevor_rails/blocks/uploaded_items_block.rb +4 -0
  66. data/app/models/spotlight/ability.rb +6 -2
  67. data/app/models/spotlight/attachment.rb +1 -1
  68. data/app/models/spotlight/background_job_progress.rb +96 -0
  69. data/app/models/spotlight/blacklight_configuration.rb +26 -6
  70. data/app/models/spotlight/bulk_update.rb +8 -0
  71. data/app/models/spotlight/event.rb +13 -0
  72. data/app/models/spotlight/exhibit.rb +5 -14
  73. data/app/models/spotlight/feature_page.rb +0 -2
  74. data/app/models/spotlight/featured_image.rb +40 -1
  75. data/app/models/spotlight/field_metadata.rb +4 -8
  76. data/app/models/spotlight/job_tracker.rb +114 -0
  77. data/app/models/spotlight/page.rb +1 -3
  78. data/app/models/spotlight/page_configurations.rb +10 -9
  79. data/app/models/spotlight/resource.rb +28 -62
  80. data/app/models/spotlight/resources/iiif_harvester.rb +12 -3
  81. data/app/models/spotlight/resources/iiif_manifest.rb +3 -1
  82. data/app/models/spotlight/resources/iiif_service.rb +9 -2
  83. data/app/models/spotlight/resources/json_upload.rb +12 -0
  84. data/app/models/spotlight/resources/upload.rb +25 -2
  85. data/app/models/spotlight/role.rb +1 -2
  86. data/app/models/spotlight/solr_document_sidecar.rb +2 -1
  87. data/app/presenters/spotlight/iiif_manifest_presenter.rb +2 -2
  88. data/app/services/spotlight/bulk_updates_csv_template_service.rb +93 -0
  89. data/app/services/spotlight/etl.rb +7 -0
  90. data/app/services/spotlight/etl/context.rb +52 -0
  91. data/app/services/spotlight/etl/executor.rb +192 -0
  92. data/app/services/spotlight/etl/loaders.rb +12 -0
  93. data/app/services/spotlight/etl/pipeline.rb +81 -0
  94. data/app/services/spotlight/etl/solr_loader.rb +96 -0
  95. data/app/services/spotlight/etl/sources.rb +25 -0
  96. data/app/services/spotlight/etl/step.rb +82 -0
  97. data/app/services/spotlight/etl/transforms.rb +64 -0
  98. data/app/services/spotlight/iiif_resource_resolver.rb +1 -1
  99. data/app/services/spotlight/validity_checker.rb +5 -5
  100. data/app/uploaders/spotlight/bulk_updates_uploader.rb +7 -0
  101. data/app/uploaders/spotlight/featured_image_uploader.rb +1 -1
  102. data/app/views/catalog/_add_tags.html.erb +28 -0
  103. data/app/views/catalog/_bulk_actions.html.erb +12 -0
  104. data/app/views/catalog/_change_visibility.html.erb +35 -0
  105. data/app/views/catalog/_curator_actions.html.erb +3 -0
  106. data/app/views/catalog/_remove_tags.html.erb +37 -0
  107. data/app/views/catalog/_save_search.html.erb +1 -1
  108. data/app/views/spotlight/about_pages/_sidebar.html.erb +2 -2
  109. data/app/views/spotlight/browse/_search.html.erb +3 -3
  110. data/app/views/spotlight/browse/_search_box.html.erb +9 -9
  111. data/app/views/spotlight/browse/_search_title.html.erb +1 -1
  112. data/app/views/spotlight/bulk_updates/_download.html.erb +23 -0
  113. data/app/views/spotlight/bulk_updates/_overview.html.erb +1 -0
  114. data/app/views/spotlight/bulk_updates/_progress_panel.html.erb +19 -0
  115. data/app/views/spotlight/bulk_updates/_upload.html.erb +12 -0
  116. data/app/views/spotlight/bulk_updates/edit.html.erb +37 -0
  117. data/app/views/spotlight/catalog/_admin_header.html.erb +1 -1
  118. data/app/views/spotlight/catalog/_admin_index_header_default.html.erb +1 -1
  119. data/app/views/spotlight/catalog/_admin_thumbnail_default.html.erb +3 -2
  120. data/app/views/spotlight/catalog/_edit_default.html.erb +2 -2
  121. data/app/views/spotlight/catalog/_reindex_progress_panel.html.erb +1 -1
  122. data/app/views/spotlight/catalog/index.iiif_json.jbuilder +1 -1
  123. data/app/views/spotlight/custom_fields/_form.html.erb +1 -1
  124. data/app/views/spotlight/custom_search_fields/_form.html.erb +1 -1
  125. data/app/views/spotlight/dashboards/_reindexing_activity.html.erb +6 -6
  126. data/app/views/spotlight/exhibits/_exhibit_card.html.erb +1 -1
  127. data/app/views/spotlight/feature_pages/_sidebar.html.erb +1 -1
  128. data/app/views/spotlight/featured_images/_form.html.erb +1 -1
  129. data/app/views/spotlight/job_trackers/show.html.erb +79 -0
  130. data/app/views/spotlight/metadata_configurations/_metadata_field.html.erb +2 -2
  131. data/app/views/spotlight/pages/_form.html.erb +2 -2
  132. data/app/views/spotlight/pages/_order_pages.html.erb +2 -2
  133. data/app/views/spotlight/pages/_view_type_group.html.erb +3 -3
  134. data/app/views/spotlight/pages/show.html.erb +1 -1
  135. data/app/views/spotlight/resources/_form.html.erb +1 -1
  136. data/app/views/spotlight/resources/csv_upload/_form.html.erb +1 -1
  137. data/app/views/spotlight/resources/iiif/_form.html.erb +1 -1
  138. data/app/views/spotlight/resources/json_upload/_form.html.erb +2 -2
  139. data/app/views/spotlight/resources/upload/_form.html.erb +1 -1
  140. data/app/views/spotlight/search_configurations/_document_index_view_types.html.erb +2 -2
  141. data/app/views/spotlight/searches/_form.html.erb +2 -2
  142. data/app/views/spotlight/shared/_curation_sidebar.html.erb +3 -0
  143. data/app/views/spotlight/shared/_locale_picker.html.erb +1 -1
  144. data/app/views/spotlight/sir_trevor/blocks/_browse_group_categories_block.html.erb +7 -6
  145. data/app/views/spotlight/sir_trevor/blocks/_search_results_block.html.erb +1 -1
  146. data/app/views/spotlight/sir_trevor/blocks/_solr_documents_block.html.erb +8 -8
  147. data/app/views/spotlight/sir_trevor/blocks/_solr_documents_carousel_block.html.erb +7 -7
  148. data/app/views/spotlight/sir_trevor/blocks/_solr_documents_features_block.html.erb +9 -7
  149. data/app/views/spotlight/sir_trevor/blocks/_solr_documents_grid_block.html.erb +5 -5
  150. data/app/views/spotlight/sir_trevor/blocks/_uploaded_items_block.html.erb +4 -0
  151. data/app/views/spotlight/tags/_tag.html.erb +24 -0
  152. data/app/views/spotlight/tags/index.html.erb +12 -16
  153. data/app/views/spotlight/translations/show.yaml.yamlbuilder +6 -0
  154. data/config/i18n-tasks.yml +1 -0
  155. data/config/locales/spotlight.ar.yml +11 -1
  156. data/config/locales/spotlight.en.yml +170 -10
  157. data/config/routes.rb +29 -1
  158. data/db/migrate/20210122082032_create_job_trackers.rb +22 -0
  159. data/db/migrate/20210126123041_create_events.rb +15 -0
  160. data/db/migrate/20210305070001_remove_class_from_sirtrevor_image_blocks.rb +20 -0
  161. data/db/migrate/20210305171150_create_bulk_updates.rb +9 -0
  162. data/db/migrate/20210308090000_migrate_caption_values_for_title_key.rb +29 -0
  163. data/db/migrate/20210506070809_add_indexes_for_featured_images.rb +9 -0
  164. data/lib/generators/spotlight/install_generator.rb +2 -2
  165. data/lib/generators/spotlight/scaffold_resource_generator.rb +5 -13
  166. data/lib/generators/spotlight/templates/config/initializers/riiif.rb +7 -5
  167. data/lib/generators/spotlight/templates/config/initializers/spotlight_initializer.rb +1 -1
  168. data/lib/generators/spotlight/templates/solr/config/schema.xml +1 -1
  169. data/lib/migration/iiif.rb +3 -3
  170. data/lib/spotlight/engine.rb +22 -1
  171. data/lib/spotlight/version.rb +1 -1
  172. data/spec/controllers/spotlight/browse_controller_spec.rb +0 -6
  173. data/spec/controllers/spotlight/bulk_actions_controller_spec.rb +124 -0
  174. data/spec/controllers/spotlight/bulk_updates_controller_spec.rb +77 -0
  175. data/spec/controllers/spotlight/catalog_controller_spec.rb +15 -13
  176. data/spec/controllers/spotlight/featured_images_controller_spec.rb +3 -3
  177. data/spec/controllers/spotlight/job_trackers_controller_spec.rb +37 -0
  178. data/spec/controllers/spotlight/tags_controller_spec.rb +5 -1
  179. data/spec/examples.txt +1502 -1437
  180. data/spec/factories/bulk_updates.rb +15 -0
  181. data/spec/factories/exhibits.rb +4 -0
  182. data/spec/factories/job_trackers.rb +11 -0
  183. data/spec/factories/users.rb +27 -8
  184. data/spec/features/add_items_spec.rb +10 -5
  185. data/spec/features/bulk_actions_spec.rb +72 -0
  186. data/spec/features/catalog_spec.rb +1 -0
  187. data/spec/features/import_exhibit_spec.rb +5 -1
  188. data/spec/features/javascript/blocks/solr_documents_block_spec.rb +4 -4
  189. data/spec/features/javascript/blocks/uploaded_items_block_spec.rb +27 -1
  190. data/spec/features/javascript/reindex_monitor_spec.rb +1 -1
  191. data/spec/features/site_users_management_spec.rb +5 -5
  192. data/spec/fixtures/bulk-update-template.csv +57 -0
  193. data/spec/fixtures/updated-bulk-update-template-w-tags.csv +4 -0
  194. data/spec/fixtures/updated-bulk-update-template.csv +4 -0
  195. data/spec/helpers/spotlight/application_helper_spec.rb +5 -6
  196. data/spec/helpers/spotlight/pages_helper_spec.rb +8 -15
  197. data/spec/helpers/spotlight/roles_helper_spec.rb +1 -1
  198. data/spec/i18n_spec.rb +1 -0
  199. data/spec/jobs/spotlight/add_tags_job_spec.rb +34 -0
  200. data/spec/jobs/spotlight/add_uploads_from_csv_spec.rb +4 -1
  201. data/spec/jobs/spotlight/change_visibility_job_spec.rb +30 -0
  202. data/spec/jobs/spotlight/process_bulk_updates_csv_job_spec.rb +78 -0
  203. data/spec/jobs/spotlight/reindex_exhibit_job_spec.rb +43 -0
  204. data/spec/jobs/spotlight/reindex_job_spec.rb +34 -60
  205. data/spec/jobs/spotlight/remove_tags_job_spec.rb +39 -0
  206. data/spec/lib/migration/iiif_spec.rb +1 -1
  207. data/spec/models/spotlight/ability_spec.rb +27 -0
  208. data/spec/models/spotlight/background_job_progress_spec.rb +137 -0
  209. data/spec/models/spotlight/blacklight_configuration_spec.rb +22 -17
  210. data/spec/models/spotlight/browse_category_search_builder_spec.rb +49 -0
  211. data/spec/models/spotlight/exhibit_spec.rb +4 -58
  212. data/spec/models/spotlight/featured_image_spec.rb +13 -1
  213. data/spec/models/spotlight/resource_spec.rb +89 -87
  214. data/spec/models/spotlight/resources/iiif_harvester_spec.rb +9 -10
  215. data/spec/models/spotlight/solr_document/atomic_updates_spec.rb +10 -0
  216. data/spec/models/spotlight/solr_document_sidecar_spec.rb +1 -0
  217. data/spec/presenters/spotlight/iiif_manifest_presenter_spec.rb +1 -1
  218. data/spec/services/spotlight/bulk_updates_csv_template_service_spec.rb +26 -0
  219. data/spec/services/spotlight/etl/context_spec.rb +66 -0
  220. data/spec/services/spotlight/etl/executor_spec.rb +149 -0
  221. data/spec/services/spotlight/etl/pipeline_spec.rb +22 -0
  222. data/spec/services/spotlight/etl/solr_loader_spec.rb +76 -0
  223. data/spec/services/spotlight/etl/step_spec.rb +70 -0
  224. data/spec/spec_helper.rb +2 -5
  225. data/spec/test_app_templates/Gemfile.extra +1 -0
  226. data/spec/test_app_templates/lib/generators/test_app_generator.rb +1 -1
  227. data/spec/uploaders/spotlight/featured_image_uploader_spec.rb +2 -2
  228. data/spec/views/spotlight/catalog/_edit_default.html.erb_spec.rb +1 -1
  229. data/spec/views/spotlight/dashboards/_reindexing_activity.html.erb_spec.rb +24 -19
  230. data/spec/views/spotlight/job_trackers/show.html.erb_spec.rb +65 -0
  231. data/spec/views/spotlight/metadata_configurations/_metadata_field.html.erb_spec.rb +3 -4
  232. data/spec/views/spotlight/sir_trevor/blocks/_browse_block.html.erb_spec.rb +1 -1
  233. data/spec/views/spotlight/sir_trevor/blocks/_iframe_block.html.erb_spec.rb +1 -1
  234. data/spec/views/spotlight/sir_trevor/blocks/_link_to_search_block.html.erb_spec.rb +1 -1
  235. data/spec/views/spotlight/sir_trevor/blocks/_rule_block.html.erb_spec.rb +1 -1
  236. data/spec/views/spotlight/sir_trevor/blocks/_solr_documents_block.html.erb_spec.rb +8 -4
  237. data/spec/views/spotlight/sir_trevor/blocks/_solr_documents_carousel_block.html.erb_spec.rb +7 -2
  238. data/spec/views/spotlight/sir_trevor/blocks/_solr_documents_embed_block.html.erb_spec.rb +8 -5
  239. data/spec/views/spotlight/sir_trevor/blocks/_solr_documents_features_block.html.erb_spec.rb +3 -2
  240. data/spec/views/spotlight/sir_trevor/blocks/_solr_documents_grid_block.html.erb_spec.rb +9 -4
  241. data/spec/views/spotlight/tags/index.html.erb_spec.rb +5 -2
  242. metadata +143 -34
  243. data/app/assets/images/blacklight/add_circle.svg +0 -1
  244. data/app/assets/images/blacklight/custom_fullscreen.svg +0 -1
  245. data/app/assets/images/blacklight/remove_circle.svg +0 -1
  246. data/app/assets/images/blacklight/resize_small.svg +0 -1
  247. data/app/models/concerns/spotlight/resources/open_graph.rb +0 -36
  248. data/app/models/spotlight/reindex_progress.rb +0 -78
  249. data/app/models/spotlight/reindexing_log_entry.rb +0 -42
  250. data/app/services/spotlight/resources/iiif_builder.rb +0 -19
  251. data/app/services/spotlight/solr_document_builder.rb +0 -77
  252. data/app/services/spotlight/upload_solr_document_builder.rb +0 -57
  253. data/spec/factories/reindexing_log_entries.rb +0 -54
  254. data/spec/models/spotlight/reindex_progress_spec.rb +0 -122
  255. data/spec/models/spotlight/reindexing_log_entry_spec.rb +0 -129
  256. data/spec/models/spotlight/resources/open_graph_spec.rb +0 -65
  257. data/spec/services/spotlight/solr_document_builder_spec.rb +0 -66
@@ -45,18 +45,14 @@ module Spotlight
45
45
 
46
46
  # This gets the number of *documents* with a field
47
47
  def document_counts
48
- @document_count ||= begin
49
- solr_response.facet_queries.each_with_object({}) do |(k, v), h|
50
- h[k.split(/:/).first] = v
51
- end
48
+ @document_count ||= solr_response.facet_queries.each_with_object({}) do |(k, v), h|
49
+ h[k.split(/:/).first] = v
52
50
  end
53
51
  end
54
52
 
55
53
  def terms
56
- @terms ||= begin
57
- solr_response.aggregations.each_with_object({}) do |(facet_name, facet), h|
58
- h[facet_name] = facet.items.map(&:label)
59
- end
54
+ @terms ||= solr_response.aggregations.each_with_object({}) do |(facet_name, facet), h|
55
+ h[facet_name] = facet.items.map(&:label)
60
56
  end
61
57
  end
62
58
 
@@ -0,0 +1,114 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Spotlight
4
+ # Associate background jobs with records
5
+ class JobTracker < ActiveRecord::Base
6
+ scope :recent, -> { order('updated_at DESC').limit(5) }
7
+ scope :in_progress, -> { where.not(status: %w[completed failed]) }
8
+ scope :completed, -> { where(status: %w[completed failed]) }
9
+
10
+ belongs_to :on, polymorphic: true
11
+ belongs_to :resource, polymorphic: true
12
+ belongs_to :user, optional: true, class_name: Spotlight::Engine.config.user_class # rubocop:disable Rails/ReflectionClassName
13
+ has_many :events, as: :resource, dependent: :delete_all
14
+ has_many :job_trackers, as: :on, dependent: Rails.version > '6.1' ? :destroy_async : :destroy
15
+ has_many :subevents, through: :job_trackers, source: :events
16
+
17
+ serialize :data
18
+
19
+ after_initialize do
20
+ self.data ||= {}
21
+ end
22
+
23
+ after_commit do
24
+ next unless on.is_a? Spotlight::JobTracker
25
+
26
+ UpdateJobTrackersJob.perform_later(self)
27
+ end
28
+
29
+ def label
30
+ "[#{job_class.titleize}] #{resource_label}"
31
+ end
32
+
33
+ def resource_label
34
+ return resource.filename if resource.is_a? ActiveStorage::Blob
35
+ return resource.name if resource.is_a? Upload
36
+
37
+ resource_id
38
+ end
39
+
40
+ def job_status
41
+ return {} unless job_id
42
+
43
+ @job_status ||= ActiveJob::Status.get(job_id)
44
+ end
45
+
46
+ def progress_label
47
+ return number_with_delimiter(progress) unless total?
48
+
49
+ "#{number_with_delimiter(progress)} / #{number_with_delimiter(total)}"
50
+ end
51
+
52
+ def progress
53
+ data[:progress] || job_status[:progress] || 0
54
+ end
55
+
56
+ def total(default: progress)
57
+ [progress, data[:total] || job_status[:total] || default].max
58
+ end
59
+
60
+ def total?
61
+ total(default: 0).positive?
62
+ end
63
+
64
+ def percent
65
+ return nil unless total?
66
+
67
+ (100.0 * progress) / total
68
+ end
69
+
70
+ def status
71
+ @status ||= super
72
+ @status ||= 'missing'
73
+ @status = 'in_progress' if @status == 'completed' && job_trackers.any? { |t| t.in_progress? || t.enqueued? }
74
+
75
+ @status
76
+ end
77
+
78
+ def enqueued?
79
+ status == 'enqueued'
80
+ end
81
+
82
+ def in_progress?
83
+ status == 'in_progress'
84
+ end
85
+
86
+ def completed?
87
+ status == 'completed'
88
+ end
89
+
90
+ def failed?
91
+ status == 'failed'
92
+ end
93
+
94
+ def append_log_entry(type:, exhibit: nil, **args)
95
+ events.create(type: type, exhibit: exhibit, data: args)
96
+ rescue StandardError => e
97
+ Rails.logger.error("Unable to create log entry for job tracker #{id}: #{e}")
98
+ end
99
+
100
+ def top_level_job_tracker
101
+ if on.is_a?(Spotlight::JobTracker)
102
+ on.top_level_job_tracker
103
+ else
104
+ self
105
+ end
106
+ end
107
+
108
+ private
109
+
110
+ def number_with_delimiter(*args)
111
+ ActiveSupport::NumberHelper.number_to_delimited(*args)
112
+ end
113
+ end
114
+ end
@@ -51,9 +51,7 @@ module Spotlight
51
51
  end
52
52
 
53
53
  def content
54
- @content ||= begin
55
- Spotlight::PageContent.for(self, :content)
56
- end
54
+ @content ||= Spotlight::PageContent.for(self, :content)
57
55
  end
58
56
 
59
57
  def content_type
@@ -18,14 +18,12 @@ module Spotlight
18
18
  delegate :available_view_fields,
19
19
  :blacklight_config,
20
20
  :current_exhibit,
21
- :document_show_link_field,
22
- :index_fields,
23
- :index_field_label,
24
21
  :spotlight,
25
22
  :t,
26
- :view_label,
27
23
  to: :context
28
24
 
25
+ DOCUMENT_TITLE_KEY = '__spotlight_document_title___'
26
+
29
27
  attr_reader :context, :page
30
28
 
31
29
  def initialize(context:, page:)
@@ -35,7 +33,7 @@ module Spotlight
35
33
 
36
34
  def as_json(*)
37
35
  {
38
- 'blacklight-configuration-index-fields': available_index_fields,
36
+ 'blacklight-configuration-index-fields': available_caption_fields,
39
37
  'blacklight-configuration-search-views': available_view_configs,
40
38
  'attachment-endpoint': attachment_endpoint,
41
39
  'autocomplete-exhibit-catalog-path': exhibit_autocomplete_endpoint,
@@ -48,15 +46,18 @@ module Spotlight
48
46
 
49
47
  private
50
48
 
51
- def available_index_fields
52
- fields = blacklight_config.index_fields.map { |k, _v| { key: k, label: index_field_label(blacklight_config.document_model.new, k) } }
53
- fields.unshift(key: document_show_link_field, label: t(:'spotlight.pages.form.title_placeholder')) unless index_fields.include? document_show_link_field
49
+ def available_caption_fields
50
+ default_caption_fields + blacklight_config.index_fields.reject { |_k, v| v.caption == false }.map { |k, v| { key: k, label: v.display_label } }
51
+ end
54
52
 
53
+ def default_caption_fields
54
+ fields = []
55
+ fields << { key: DOCUMENT_TITLE_KEY, label: t(:'spotlight.pages.form.title_placeholder') } unless blacklight_config.index_fields.key? DOCUMENT_TITLE_KEY
55
56
  fields
56
57
  end
57
58
 
58
59
  def available_view_configs
59
- available_view_fields.map { |k, _| { key: k, label: view_label(k) } }
60
+ available_view_fields.map { |k, v| { key: k, label: v.display_label(k) } }
60
61
  end
61
62
 
62
63
  def attachment_endpoint
@@ -4,10 +4,17 @@ module Spotlight
4
4
  ##
5
5
  # Exhibit resources
6
6
  class Resource < ActiveRecord::Base
7
- include ActiveSupport::Benchmarkable
8
-
9
- class_attribute :document_builder_class
10
- self.document_builder_class = SolrDocumentBuilder
7
+ class_attribute :indexing_pipeline, default: (Spotlight::Etl::Pipeline.new do |pipeline|
8
+ pipeline.sources = [Spotlight::Etl::Sources::IdentitySource]
9
+ pipeline.transforms = [
10
+ reject_blank: Spotlight::Etl::Transforms::RejectBlank,
11
+ reject_missing: Spotlight::Etl::Transforms::RejectMissingUniqueId,
12
+ apply_exhibit_metadata: Spotlight::Etl::Transforms::ApplyExhibitMetadata,
13
+ apply_application_metadata: Spotlight::Etl::Transforms::ApplyApplicationMetadata,
14
+ apply_pipeline_metadata: Spotlight::Etl::Transforms::ApplyPipelineMetadata
15
+ ]
16
+ pipeline.loaders = [Spotlight::Etl::SolrLoader]
17
+ end)
11
18
 
12
19
  extend ActiveModel::Callbacks
13
20
  define_model_callbacks :index
@@ -16,24 +23,22 @@ module Spotlight
16
23
 
17
24
  belongs_to :exhibit
18
25
  has_many :solr_document_sidecars
26
+ has_many :events, as: :resource
19
27
 
20
28
  serialize :data, Hash
21
29
 
22
- after_index :commit
23
- after_index :touch_exhibit!
24
-
25
30
  ##
26
31
  # Persist the record to the database, and trigger a reindex to solr
27
32
  #
28
33
  # @param [Hash] All arguments will be passed through to ActiveRecord's #save method
29
- def save_and_index(*args)
30
- save(*args) && reindex_later
34
+ def save_and_index(reindex_options: {}, **args)
35
+ save(*args) && reindex_later(**reindex_options)
31
36
  end
32
37
 
33
38
  ##
34
39
  # Enqueue an asynchronous reindexing job for this resource
35
- def reindex_later
36
- Spotlight::ReindexJob.perform_later(self)
40
+ def reindex_later(**args)
41
+ Spotlight::ReindexJob.perform_later(self, **args)
37
42
  end
38
43
 
39
44
  def document_model
@@ -45,67 +50,28 @@ module Spotlight
45
50
  # Index the result of {#to_solr} into the index in batches of {#batch_size}
46
51
  #
47
52
  # @return [Integer] number of records indexed
48
- def reindex(reindexing_log_entry = nil)
49
- benchmark "Reindexing #{self} (batch size: #{batch_size})" do
50
- count = 0
51
-
52
- run_callbacks :index do
53
- document_builder.documents_to_index.each_slice(batch_size) do |batch|
54
- write_to_index(batch)
55
- count += batch.length
56
- reindexing_log_entry&.update(items_reindexed_count: count)
57
- end
58
-
59
- count
53
+ def reindex(touch: true, **args, &block)
54
+ i = 0
55
+ run_callbacks :index do
56
+ indexing_pipeline.call(Spotlight::Etl::Context.new(self, commit: true, **args)) do |data|
57
+ i += 1
58
+ block&.call(data)
60
59
  end
61
60
  end
62
- end
63
-
64
- def document_builder
65
- @document_builder ||= document_builder_class.new(self)
66
- end
67
-
68
- private
69
-
70
- def blacklight_solr
71
- @solr ||= RSolr.connect(connection_config.merge(adapter: connection_config[:http_adapter]))
72
- end
73
61
 
74
- def connection_config
75
- Blacklight.connection_config
76
- end
62
+ touch_exhibit! if touch
77
63
 
78
- def batch_size
79
- Spotlight::Engine.config.solr_batch_size
64
+ i
80
65
  end
81
66
 
82
- def write_to_index(batch)
83
- documents = documents_that_have_ids(batch)
84
- return unless write? && documents.present?
85
-
86
- blacklight_solr.update params: { commitWithin: 500 },
87
- data: documents.to_json,
88
- headers: { 'Content-Type' => 'application/json' }
67
+ def estimated_size(**args)
68
+ indexing_pipeline.estimated_size(Spotlight::Etl::Context.new(self, **args))
89
69
  end
90
70
 
91
- def commit
92
- return unless write?
93
-
94
- blacklight_solr.commit
95
- rescue StandardError => e
96
- Rails.logger.warn "Unable to commit to solr: #{e}"
97
- end
71
+ private
98
72
 
99
73
  def touch_exhibit!
100
- exhibit.touch
101
- end
102
-
103
- def write?
104
- Spotlight::Engine.config.writable_index
105
- end
106
-
107
- def documents_that_have_ids(document_list)
108
- document_list.reject { |d| d[document_builder.document_model.unique_key.to_sym].blank? }
74
+ exhibit&.touch
109
75
  end
110
76
  end
111
77
  end
@@ -7,7 +7,6 @@ module Spotlight
7
7
  # harvest Images from IIIF Manifest and turn them into a Spotlight::Resource
8
8
  # Note: IIIF API : http://iiif.io/api/presentation/2.0
9
9
  class IiifHarvester < Spotlight::Resource
10
- self.document_builder_class = Spotlight::Resources::IiifBuilder
11
10
  self.weight = -5000
12
11
 
13
12
  validate :valid_url?
@@ -16,6 +15,16 @@ module Spotlight
16
15
  @iiif_manifests ||= IiifService.parse(url)
17
16
  end
18
17
 
18
+ def self.indexing_pipeline
19
+ @indexing_pipeline ||= super.dup.tap do |pipeline|
20
+ pipeline.sources = [Spotlight::Etl::Sources::SourceMethodSource(:iiif_manifests)]
21
+
22
+ pipeline.transforms = [
23
+ ->(data, p) { data.merge(p.source.to_solr(exhibit: p.context.resource.exhibit)) }
24
+ ] + pipeline.transforms
25
+ end
26
+ end
27
+
19
28
  private
20
29
 
21
30
  def valid_url?
@@ -24,8 +33,8 @@ module Spotlight
24
33
 
25
34
  def url_is_iiif?(url)
26
35
  valid_content_types = ['application/json', 'application/ld+json']
27
- req = Faraday.head(url)
28
- req = Faraday.get(url) if req.status == 405
36
+ req = Spotlight::Resources::IiifService.http_client.head(url)
37
+ req = Spotlight::Resources::IiifService.http_client.get(url) if req.status == 405
29
38
  return unless req.success?
30
39
 
31
40
  valid_content_types.any? do |valid_type|
@@ -14,7 +14,9 @@ module Spotlight
14
14
  @solr_hash = {}
15
15
  end
16
16
 
17
- def to_solr
17
+ def to_solr(exhibit: nil)
18
+ @exhibit = exhibit if exhibit
19
+
18
20
  add_document_id
19
21
  add_label
20
22
  add_thumbnail_url
@@ -29,6 +29,13 @@ module Spotlight
29
29
  recursive_manifests(new(url))
30
30
  end
31
31
 
32
+ def self.http_client
33
+ Faraday.new do |b|
34
+ b.use FaradayMiddleware::FollowRedirects
35
+ b.adapter Faraday.default_adapter
36
+ end
37
+ end
38
+
32
39
  protected
33
40
 
34
41
  def object
@@ -41,8 +48,8 @@ module Spotlight
41
48
 
42
49
  class << self
43
50
  def iiif_response(url)
44
- Faraday.get(url).body
45
- rescue Faraday::Error::ConnectionFailed, Faraday::TimeoutError => e
51
+ Spotlight::Resources::IiifService.http_client.get(url).body
52
+ rescue Faraday::ConnectionFailed, Faraday::TimeoutError => e
46
53
  Rails.logger.warn("HTTP GET for #{url} failed with #{e}")
47
54
  {}.to_json
48
55
  end
@@ -5,6 +5,18 @@ module Spotlight
5
5
  # Raw solr document uploads
6
6
  class JsonUpload < Spotlight::Resource
7
7
  store :data, accessors: :json
8
+
9
+ # The indexing pipeline for JSON uploads copies the data from the stored
10
+ # `#data` field directly into the indexed document.
11
+ def self.indexing_pipeline
12
+ @indexing_pipeline ||= super.dup.tap do |pipeline|
13
+ pipeline.sources = [Spotlight::Etl::Sources::StoredData]
14
+
15
+ pipeline.transforms = [
16
+ Spotlight::Etl::Transforms::IdentityTransform
17
+ ] + pipeline.transforms
18
+ end
19
+ end
8
20
  end
9
21
  end
10
22
  end
@@ -10,8 +10,6 @@ module Spotlight
10
10
  # we want to do this before reindexing
11
11
  after_create :update_document_sidecar
12
12
 
13
- self.document_builder_class = UploadSolrDocumentBuilder
14
-
15
13
  def self.fields(exhibit)
16
14
  @fields ||= {}
17
15
  @fields[exhibit] ||= begin
@@ -25,6 +23,15 @@ module Spotlight
25
23
  end
26
24
  end
27
25
 
26
+ def self.indexing_pipeline
27
+ @indexing_pipeline ||= super.dup.tap do |pipeline|
28
+ pipeline.transforms = [
29
+ ->(data, p) { data.merge({ p.context.document_model.unique_key.to_sym => p.source.compound_id }) },
30
+ Spotlight::Etl::Transforms::SourceMethodTransform(:to_solr)
31
+ ] + pipeline.transforms
32
+ end
33
+ end
34
+
28
35
  def compound_id
29
36
  "#{exhibit_id}-#{id}"
30
37
  end
@@ -33,6 +40,22 @@ module Spotlight
33
40
  @sidecar ||= document_model.new(id: compound_id).sidecar(exhibit)
34
41
  end
35
42
 
43
+ def to_solr
44
+ return {} unless upload&.file_present?
45
+
46
+ spotlight_routes = Spotlight::Engine.routes.url_helpers
47
+ riiif = Riiif::Engine.routes.url_helpers
48
+
49
+ dimensions = Riiif::Image.new(upload_id).info
50
+
51
+ {
52
+ spotlight_full_image_width_ssm: dimensions.width,
53
+ spotlight_full_image_height_ssm: dimensions.height,
54
+ Spotlight::Engine.config.thumbnail_field => riiif.image_path(upload, size: '!400,400'),
55
+ Spotlight::Engine.config.iiif_manifest_field => spotlight_routes.manifest_exhibit_solr_document_path(exhibit, compound_id)
56
+ }
57
+ end
58
+
36
59
  private
37
60
 
38
61
  def configured_fields