• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

pulibrary / pdc_describe / 4e4e59fc-9df4-4838-9fd4-6c7ea33cdb7c

07 Apr 2025 06:36PM UTC coverage: 1.283% (-94.6%) from 95.862%
4e4e59fc-9df4-4838-9fd4-6c7ea33cdb7c

Pull #1994

circleci

hectorcorrea
Switched to use the autocomplete that we aleady use for ROR. Integrated it with the existing logic for creators
Pull Request #1994: Started adding auto complete to contributors

0 of 46 new or added lines in 2 files covered. (0.0%)

4806 existing lines in 74 files now uncovered.

65 of 5065 relevant lines covered (1.28%)

0.01 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

0.0
/app/models/work.rb
1
# frozen_string_literal: true
2

3
# rubocop:disable Metrics/ClassLength
UNCOV
4
class Work < ApplicationRecord
×
5
  # Errors for cases where there is no valid Group
UNCOV
6
  class InvalidGroupError < ::ArgumentError; end
×
7

UNCOV
8
  has_many :work_activity, -> { order(updated_at: :desc) }, dependent: :destroy
×
UNCOV
9
  has_many :user_work, -> { order(updated_at: :desc) }, dependent: :destroy
×
UNCOV
10
  has_many :upload_snapshots, -> { order(updated_at: :desc) }, dependent: :destroy
×
11

UNCOV
12
  belongs_to :group, class_name: "Group"
×
UNCOV
13
  belongs_to :curator, class_name: "User", foreign_key: "curator_user_id", optional: true
×
14

UNCOV
15
  attribute :work_type, :string, default: "DATASET"
×
UNCOV
16
  attribute :profile, :string, default: "DATACITE"
×
17

UNCOV
18
  attr_accessor :user_entered_doi
×
19

UNCOV
20
  alias state_history user_work
×
21

UNCOV
22
  delegate :valid_to_submit, :valid_to_draft, :valid_to_approve, :valid_to_complete, to: :work_validator
×
23

UNCOV
24
  include AASM
×
25

UNCOV
26
  aasm column: :state do
×
UNCOV
27
    state :none, initial: true
×
UNCOV
28
    state :draft, :awaiting_approval, :approved, :withdrawn, :deletion_marker
×
29

UNCOV
30
    event :draft, after: :draft_doi do
×
UNCOV
31
      transitions from: :none, to: :draft, guard: :valid_to_draft
×
UNCOV
32
    end
×
33

UNCOV
34
    event :complete_submission do
×
UNCOV
35
      transitions from: :draft, to: :awaiting_approval, guard: :valid_to_complete
×
UNCOV
36
    end
×
37

UNCOV
38
    event :request_changes do
×
UNCOV
39
      transitions from: :awaiting_approval, to: :awaiting_approval, guard: :valid_to_submit
×
UNCOV
40
    end
×
41

UNCOV
42
    event :revert_to_draft do
×
UNCOV
43
      transitions from: :awaiting_approval, to: :draft, guard: :valid_to_draft
×
UNCOV
44
    end
×
45

UNCOV
46
    event :approve do
×
UNCOV
47
      transitions from: :awaiting_approval, to: :approved, guard: :valid_to_approve, after: :publish
×
UNCOV
48
    end
×
49

UNCOV
50
    event :withdraw do
×
UNCOV
51
      transitions from: [:draft, :awaiting_approval, :approved], to: :withdrawn
×
UNCOV
52
    end
×
53

UNCOV
54
    event :resubmit do
×
UNCOV
55
      transitions from: :withdrawn, to: :draft
×
UNCOV
56
    end
×
57

UNCOV
58
    event :remove do
×
UNCOV
59
      transitions from: :withdrawn, to: :deletion_marker
×
UNCOV
60
    end
×
61

UNCOV
62
    after_all_events :track_state_change
×
UNCOV
63
  end
×
64

UNCOV
65
  def state=(new_state)
×
UNCOV
66
    new_state_sym = new_state.to_sym
×
UNCOV
67
    valid_states = self.class.aasm.states.map(&:name)
×
UNCOV
68
    raise(StandardError, "Invalid state '#{new_state}'") unless valid_states.include?(new_state_sym)
×
UNCOV
69
    aasm_write_state_without_persistence(new_state_sym)
×
UNCOV
70
  end
×
71

72
  ##
73
  # Is this work editable by a given user?
74
  # A work is editable when:
75
  # * it is being edited by the person who made it
76
  # * it is being edited by a group admin of the group where is resides
77
  # * it is being edited by a super admin
78
  # @param [User]
79
  # @return [Boolean]
UNCOV
80
  def editable_by?(user)
×
UNCOV
81
    submitted_by?(user) || administered_by?(user)
×
UNCOV
82
  end
×
83

UNCOV
84
  def editable_in_current_state?(user)
×
85
    # anyone with edit privleges can edit a work while it is in draft
UNCOV
86
    return editable_by?(user) if draft?
×
87

88
    # Only admisitrators can edit a work in other states
UNCOV
89
    administered_by?(user)
×
UNCOV
90
  end
×
91

UNCOV
92
  def submitted_by?(user)
×
UNCOV
93
    created_by_user_id == user.id
×
UNCOV
94
  end
×
95

UNCOV
96
  def administered_by?(user)
×
UNCOV
97
    user.has_role?(:group_admin, group)
×
UNCOV
98
  end
×
99

UNCOV
100
  class << self
×
UNCOV
101
    def find_by_doi(doi)
×
UNCOV
102
      prefix = "10.34770/"
×
UNCOV
103
      doi = "#{prefix}#{doi}" unless doi.blank? || doi.start_with?(prefix)
×
UNCOV
104
      Work.find_by!("metadata @> ?", JSON.dump(doi:))
×
UNCOV
105
    end
×
106

UNCOV
107
    def find_by_ark(ark)
×
UNCOV
108
      prefix = "ark:/"
×
UNCOV
109
      ark = "#{prefix}#{ark}" unless ark.blank? || ark.start_with?(prefix)
×
UNCOV
110
      Work.find_by!("metadata @> ?", JSON.dump(ark:))
×
UNCOV
111
    end
×
112

UNCOV
113
    delegate :resource_type_general_values, to: PDCMetadata::Resource
×
UNCOV
114
  end
×
115

UNCOV
116
  include Rails.application.routes.url_helpers
×
117

UNCOV
118
  before_save do |work|
×
119
    # Ensure that the metadata JSONB postgres field is persisted properly
UNCOV
120
    work.metadata = JSON.parse(work.resource.to_json)
×
UNCOV
121
  end
×
122

UNCOV
123
  after_save do |work|
×
UNCOV
124
    if work.approved?
×
UNCOV
125
      work.reload
×
UNCOV
126
    end
×
UNCOV
127
  end
×
128

UNCOV
129
  validate do |_work|
×
UNCOV
130
    work_validator.valid?
×
UNCOV
131
  end
×
132

133
  # Overload ActiveRecord.reload method
134
  # https://apidock.com/rails/ActiveRecord/Base/reload
135
  #
136
  # NOTE: Usually `after_save` is a better place to put this kind of code:
137
  #
138
  #   after_save do |work|
139
  #     work.resource = nil
140
  #   end
141
  #
142
  # but that does not work in this case because the block points to a different
143
  # memory object for `work` than the we want we want to reload.
UNCOV
144
  def reload(options = nil)
×
UNCOV
145
    super
×
146
    # Force `resource` to be reloaded
UNCOV
147
    @resource = nil
×
UNCOV
148
    self
×
UNCOV
149
  end
×
150

UNCOV
151
  def title
×
UNCOV
152
    resource.main_title
×
UNCOV
153
  end
×
154

UNCOV
155
  def uploads_attributes
×
UNCOV
156
    return [] if approved? # once approved we no longer allow the updating of uploads via the application
×
UNCOV
157
    uploads.map do |upload|
×
UNCOV
158
      {
×
UNCOV
159
        id: upload.id,
×
UNCOV
160
        key: upload.key,
×
UNCOV
161
        filename: upload.filename.to_s,
×
UNCOV
162
        created_at: upload.created_at,
×
UNCOV
163
        url: upload.url
×
UNCOV
164
      }
×
UNCOV
165
    end
×
UNCOV
166
  end
×
167

UNCOV
168
  def form_attributes
×
UNCOV
169
    {
×
UNCOV
170
      uploads: uploads_attributes
×
UNCOV
171
    }
×
UNCOV
172
  end
×
173

UNCOV
174
  def draft_doi
×
UNCOV
175
    return if resource.doi.present?
×
UNCOV
176
    resource.doi = datacite_service.draft_doi
×
UNCOV
177
    save!
×
UNCOV
178
  end
×
179

180
  # Return the DOI formatted as a URL, so it can be used as a link on display pages
181
  # @return [String] A url formatted version of the DOI
UNCOV
182
  def doi_url
×
UNCOV
183
    return "https://doi.org/#{doi}" unless doi.starts_with?("https://doi.org")
×
184
    doi
×
UNCOV
185
  end
×
186

UNCOV
187
  def created_by_user
×
UNCOV
188
    User.find(created_by_user_id)
×
UNCOV
189
  rescue ActiveRecord::RecordNotFound
×
UNCOV
190
    nil
×
UNCOV
191
  end
×
192

UNCOV
193
  def resource=(resource)
×
UNCOV
194
    @resource = resource
×
195
    # Ensure that the metadata JSONB postgres field is persisted properly
UNCOV
196
    self.metadata = JSON.parse(resource.to_json)
×
UNCOV
197
  end
×
198

UNCOV
199
  def resource
×
UNCOV
200
    @resource ||= PDCMetadata::Resource.new_from_jsonb(metadata)
×
UNCOV
201
  end
×
202

UNCOV
203
  def url
×
204
    return unless persisted?
×
205

206
    @url ||= url_for(self)
×
UNCOV
207
  end
×
208

UNCOV
209
  def files_location_upload?
×
UNCOV
210
    files_location.blank? || files_location == "file_upload"
×
UNCOV
211
  end
×
212

UNCOV
213
  def files_location_cluster?
×
UNCOV
214
    files_location == "file_cluster"
×
UNCOV
215
  end
×
216

UNCOV
217
  def files_location_other?
×
UNCOV
218
    files_location == "file_other"
×
UNCOV
219
  end
×
220

UNCOV
221
  def change_curator(curator_user_id, current_user)
×
UNCOV
222
    if curator_user_id == "no-one"
×
UNCOV
223
      clear_curator(current_user)
×
UNCOV
224
    else
×
UNCOV
225
      update_curator(curator_user_id, current_user)
×
UNCOV
226
    end
×
UNCOV
227
  end
×
228

UNCOV
229
  def clear_curator(current_user)
×
230
    # Update the curator on the Work
UNCOV
231
    self.curator_user_id = nil
×
UNCOV
232
    save!
×
233

234
    # ...and log the activity
UNCOV
235
    WorkActivity.add_work_activity(id, "Unassigned existing curator", current_user.id, activity_type: WorkActivity::SYSTEM)
×
UNCOV
236
  end
×
237

UNCOV
238
  def update_curator(curator_user_id, current_user)
×
239
    # Update the curator on the Work
UNCOV
240
    self.curator_user_id = curator_user_id
×
UNCOV
241
    save!
×
242

243
    # ...and log the activity
UNCOV
244
    new_curator = User.find(curator_user_id)
×
245

UNCOV
246
    work_url = "[#{title}](#{Rails.application.routes.url_helpers.work_url(self)})"
×
247

248
    # Troubleshooting https://github.com/pulibrary/pdc_describe/issues/1783
UNCOV
249
    if work_url.include?("/describe/describe/")
×
250
      Rails.logger.error("URL #{work_url} included /describe/describe/ and was fixed. See https://github.com/pulibrary/pdc_describe/issues/1783")
×
251
      work_url = work_url.gsub("/describe/describe/", "/describe/")
×
UNCOV
252
    end
×
253

UNCOV
254
    message = if curator_user_id.to_i == current_user.id
×
UNCOV
255
                "Self-assigned @#{current_user.uid} as curator for work #{work_url}"
×
UNCOV
256
              else
×
UNCOV
257
                "Set curator to @#{new_curator.uid} for work #{work_url}"
×
UNCOV
258
              end
×
UNCOV
259
    WorkActivity.add_work_activity(id, message, current_user.id, activity_type: WorkActivity::SYSTEM)
×
UNCOV
260
  end
×
261

UNCOV
262
  def add_message(message, current_user_id)
×
UNCOV
263
    WorkActivity.add_work_activity(id, message, current_user_id, activity_type: WorkActivity::MESSAGE)
×
UNCOV
264
  end
×
265

UNCOV
266
  def add_provenance_note(date, note, current_user_id, change_label = "")
×
UNCOV
267
    WorkActivity.add_work_activity(id, { note:, change_label: }.to_json, current_user_id, activity_type: WorkActivity::PROVENANCE_NOTES, created_at: date)
×
UNCOV
268
  end
×
269

UNCOV
270
  def log_changes(resource_compare, current_user_id)
×
UNCOV
271
    return if resource_compare.identical?
×
UNCOV
272
    WorkActivity.add_work_activity(id, resource_compare.differences.to_json, current_user_id, activity_type: WorkActivity::CHANGES)
×
UNCOV
273
  end
×
274

UNCOV
275
  def log_file_changes(current_user_id)
×
UNCOV
276
    return if changes.count == 0
×
UNCOV
277
    WorkActivity.add_work_activity(id, changes.to_json, current_user_id, activity_type: WorkActivity::FILE_CHANGES)
×
UNCOV
278
  end
×
279

UNCOV
280
  def activities
×
UNCOV
281
    WorkActivity.activities_for_work(id, WorkActivity::MESSAGE_ACTIVITY_TYPES + WorkActivity::CHANGE_LOG_ACTIVITY_TYPES)
×
UNCOV
282
  end
×
283

UNCOV
284
  def new_notification_count_for_user(user_id)
×
UNCOV
285
    WorkActivityNotification.joins(:work_activity)
×
UNCOV
286
                            .where(user_id:, read_at: nil)
×
UNCOV
287
                            .where(work_activity: { work_id: id })
×
UNCOV
288
                            .count
×
UNCOV
289
  end
×
290

291
  # Marks as read the notifications for the given user_id in this work.
292
  # In practice, the user_id is the id of the current user and therefore this method marks the current's user
293
  # notifications as read.
UNCOV
294
  def mark_new_notifications_as_read(user_id)
×
295
    # Notice that we fetch and update the information in batches
296
    # so that we don't issue individual SQL SELECT + SQL UPDATE
297
    # for each notification.
298
    #
299
    # Rails batching information:
300
    #   https://guides.rubyonrails.org/active_record_querying.html
301
    #   https://api.rubyonrails.org/classes/ActiveRecord/Batches.html
302

303
    # Disable this validation since we want to force a SQL UPDATE.
304
    # rubocop:disable Rails/SkipsModelValidations
UNCOV
305
    now_utc = Time.now.utc
×
UNCOV
306
    WorkActivityNotification.joins(:work_activity).where("user_id=? and work_id=?", user_id, id).in_batches(of: 1000).update_all(read_at: now_utc)
×
307
    # rubocop:enable Rails/SkipsModelValidations
UNCOV
308
  end
×
309

UNCOV
310
  def current_transition
×
UNCOV
311
    aasm.current_event.to_s.humanize.delete("!")
×
UNCOV
312
  end
×
313

314
  # Retrieve the S3 file uploads associated with the Work
315
  # @return [Array<S3File>]
UNCOV
316
  def uploads
×
UNCOV
317
    return post_curation_uploads if approved?
×
318

UNCOV
319
    pre_curation_uploads
×
UNCOV
320
  end
×
321

322
  # Retrieve the S3 file uploads named "README"
323
  # @return [Array<S3File>]
UNCOV
324
  def readme_uploads
×
325
    uploads.select { |s3_file| s3_file.filename.include?("README") }
×
UNCOV
326
  end
×
327

328
  # Retrieve the S3 file uploads which are research artifacts proper (not README or other files providing metadata/documentation)
329
  # @return [Array<S3File>]
UNCOV
330
  def artifact_uploads
×
331
    uploads.reject { |s3_file| s3_file.filename.include?("README") }
×
UNCOV
332
  end
×
333

334
  # Returns the list of files for the work with some basic information about each of them.
335
  # This method is much faster than `uploads` because it does not return the actual S3File
336
  # objects to the client, instead it returns just a few selected data elements.
337
  # rubocop:disable Metrics/MethodLength
UNCOV
338
  def file_list
×
UNCOV
339
    start = Time.zone.now
×
UNCOV
340
    s3_files = approved? ? post_curation_uploads : pre_curation_uploads
×
UNCOV
341
    files_info = s3_files.map do |s3_file|
×
UNCOV
342
      {
×
UNCOV
343
        "safe_id": s3_file.safe_id,
×
UNCOV
344
        "filename": s3_file.filename,
×
UNCOV
345
        "filename_display": s3_file.filename_display,
×
UNCOV
346
        "last_modified": s3_file.last_modified,
×
UNCOV
347
        "last_modified_display": s3_file.last_modified_display,
×
UNCOV
348
        "size": s3_file.size,
×
UNCOV
349
        "display_size": s3_file.display_size,
×
UNCOV
350
        "url": s3_file.url,
×
UNCOV
351
        "is_folder": s3_file.is_folder
×
UNCOV
352
      }
×
UNCOV
353
    end
×
UNCOV
354
    log_performance(start, "file_list called for #{id}")
×
UNCOV
355
    files_info
×
UNCOV
356
  end
×
357
  # rubocop:enable Metrics/MethodLength
358

UNCOV
359
  def total_file_size
×
UNCOV
360
    total_size = 0
×
UNCOV
361
    file_list.each do |file|
×
362
      total_size += file[:size]
×
UNCOV
363
    end
×
UNCOV
364
    total_size
×
UNCOV
365
  end
×
366

367
  # Calculates the total file size from a given list of files
368
  # This is so that we don't fetch the list twice from AWS since it can be expensive when
369
  # there are thousands of files on the work.
UNCOV
370
  def total_file_size_from_list(files)
×
UNCOV
371
    files.sum { |file| file[:size] }
×
UNCOV
372
  end
×
373

374
  # Fetches the data from S3 directly bypassing ActiveStorage
UNCOV
375
  def pre_curation_uploads
×
UNCOV
376
    s3_query_service.client_s3_files.sort_by(&:filename)
×
UNCOV
377
  end
×
378

379
  # Accesses post-curation S3 Bucket Objects
UNCOV
380
  def post_curation_s3_resources
×
UNCOV
381
    if approved?
×
UNCOV
382
      s3_resources
×
UNCOV
383
    else
×
UNCOV
384
      []
×
UNCOV
385
    end
×
UNCOV
386
  end
×
387

388
  # Returns the files in post-curation for the work
UNCOV
389
  def post_curation_uploads(force_post_curation: false)
×
UNCOV
390
    if force_post_curation
×
391
      # Always use the post-curation data regardless of the work's status
UNCOV
392
      post_curation_s3_query_service = S3QueryService.new(self, "postcuration")
×
UNCOV
393
      post_curation_s3_query_service.data_profile.fetch(:objects, [])
×
UNCOV
394
    else
×
395
      # Return the list based of files honoring the work status
UNCOV
396
      post_curation_s3_resources
×
UNCOV
397
    end
×
UNCOV
398
  end
×
399

UNCOV
400
  def s3_files
×
401
    pre_curation_uploads
×
UNCOV
402
  end
×
403

UNCOV
404
  def s3_client
×
UNCOV
405
    s3_query_service.client
×
UNCOV
406
  end
×
407

UNCOV
408
  delegate :bucket_name, :prefix, to: :s3_query_service
×
UNCOV
409
  delegate :doi_attribute_url, :curator_or_current_uid, to: :datacite_service
×
410

411
  # Generates the S3 Object key
412
  # @return [String]
UNCOV
413
  def s3_object_key
×
UNCOV
414
    "#{doi}/#{id}"
×
UNCOV
415
  end
×
416

417
  # Transmit a HEAD request for the S3 Bucket directory for this Work
418
  # @param bucket_name location to be checked to be found
419
  # @return [Aws::S3::Types::HeadObjectOutput]
UNCOV
420
  def find_post_curation_s3_dir(bucket_name:)
×
421
    # TODO: Directories really do not exists in S3
422
    #      if we really need this check then we need to do something else to check the bucket
UNCOV
423
    s3_client.head_object({
×
UNCOV
424
                            bucket: bucket_name,
×
UNCOV
425
                            key: s3_object_key
×
UNCOV
426
                          })
×
UNCOV
427
    true
×
UNCOV
428
  rescue Aws::S3::Errors::NotFound
×
UNCOV
429
    nil
×
UNCOV
430
  end
×
431

432
  # Generates the JSON serialized expression of the Work
433
  # @param args [Array<Hash>]
434
  # @option args [Boolean] :force_post_curation Force the request of AWS S3
435
  #   Resources, clearing the in-memory cache
436
  # @return [String]
UNCOV
437
  def as_json(*args)
×
UNCOV
438
    files = files_as_json(*args)
×
439

440
    # to_json returns a string of serialized JSON.
441
    # as_json returns the corresponding hash.
UNCOV
442
    {
×
UNCOV
443
      "resource" => resource.as_json,
×
UNCOV
444
      "files" => files,
×
UNCOV
445
      "group" => group.as_json.except("id"),
×
UNCOV
446
      "embargo_date" => embargo_date_as_json,
×
UNCOV
447
      "created_at" => format_date_for_solr(created_at),
×
UNCOV
448
      "updated_at" => format_date_for_solr(updated_at)
×
UNCOV
449
    }
×
UNCOV
450
  end
×
451

452
  # Format the date for Apache Solr
453
  # @param date [ActiveSupport::TimeWithZone]
454
  # @return [String]
UNCOV
455
  def format_date_for_solr(date)
×
UNCOV
456
    date.strftime("%Y-%m-%dT%H:%M:%SZ")
×
UNCOV
457
  end
×
458

UNCOV
459
  def pre_curation_uploads_count
×
UNCOV
460
    s3_query_service.file_count
×
UNCOV
461
  end
×
462

UNCOV
463
  delegate :ark, :doi, :resource_type, :resource_type=, :resource_type_general, :resource_type_general=,
×
UNCOV
464
           :to_xml, to: :resource
×
465

466
  # S3QueryService object associated with this Work
467
  # @return [S3QueryService]
UNCOV
468
  def s3_query_service
×
UNCOV
469
    mode = approved? ? "postcuration" : "precuration"
×
UNCOV
470
    @s3_query_service ||= S3QueryService.new(self, mode)
×
UNCOV
471
  end
×
472

UNCOV
473
  def past_snapshots
×
474
    UploadSnapshot.where(work: self)
×
UNCOV
475
  end
×
476

477
  # Build or find persisted UploadSnapshot models for this Work
478
  # @param [integer] user_id optional user to assign the snapshot to
479
  # @return [UploadSnapshot]
UNCOV
480
  def reload_snapshots(user_id: nil)
×
UNCOV
481
    work_changes = []
×
UNCOV
482
    s3_files = pre_curation_uploads
×
UNCOV
483
    s3_filenames = s3_files.map(&:filename)
×
484

UNCOV
485
    upload_snapshot = latest_snapshot
×
486

UNCOV
487
    upload_snapshot.snapshot_deletions(work_changes, s3_filenames)
×
488

UNCOV
489
    upload_snapshot.snapshot_modifications(work_changes, s3_files)
×
490

491
    # Create WorkActivity models with the set of changes
UNCOV
492
    unless work_changes.empty?
×
UNCOV
493
      new_snapshot = UploadSnapshot.new(work: self, url: s3_query_service.prefix)
×
UNCOV
494
      new_snapshot.store_files(s3_files)
×
UNCOV
495
      new_snapshot.save!
×
UNCOV
496
      WorkActivity.add_work_activity(id, work_changes.to_json, user_id, activity_type: WorkActivity::FILE_CHANGES)
×
UNCOV
497
    end
×
UNCOV
498
  end
×
499

UNCOV
500
  def self.presenter_class
×
UNCOV
501
    WorkPresenter
×
UNCOV
502
  end
×
503

UNCOV
504
  def presenter
×
UNCOV
505
    self.class.presenter_class.new(work: self)
×
UNCOV
506
  end
×
507

UNCOV
508
  def changes
×
UNCOV
509
    @changes ||= []
×
UNCOV
510
  end
×
511

UNCOV
512
  def track_change(action, filename)
×
UNCOV
513
    changes << { action:, filename: }
×
UNCOV
514
  end
×
515

516
  # rubocop:disable Naming/PredicateName
UNCOV
517
  def has_rights?(rights_id)
×
UNCOV
518
    resource.rights_many.index { |rights| rights.identifier == rights_id } != nil
×
UNCOV
519
  end
×
520
  # rubocop:enable Naming/PredicateName
521

522
  # This is the solr id / work show page in PDC Discovery
UNCOV
523
  def pdc_discovery_url
×
UNCOV
524
    "https://datacommons.princeton.edu/discovery/catalog/doi-#{doi.tr('/', '-').tr('.', '-')}"
×
UNCOV
525
  end
×
526

527
  # Determine whether or not the Work is under active embargo
528
  # @return [Boolean]
UNCOV
529
  def embargoed?
×
UNCOV
530
    return false if embargo_date.blank?
×
531

UNCOV
532
    current_date = Time.zone.now
×
UNCOV
533
    embargo_date >= current_date
×
UNCOV
534
  end
×
535

UNCOV
536
  def upload_count
×
537
    @upload_count ||= s3_query_service.count_objects
×
UNCOV
538
  end
×
539

UNCOV
540
  protected
×
541

UNCOV
542
    def work_validator
×
UNCOV
543
      @work_validator ||= WorkValidator.new(self)
×
UNCOV
544
    end
×
545

546
    # This must be protected, NOT private for ActiveRecord to work properly with this attribute.
547
    #   Protected will still keep others from setting the metatdata, but allows ActiveRecord the access it needs
UNCOV
548
    def metadata=(metadata)
×
UNCOV
549
      super
×
UNCOV
550
      @resource = PDCMetadata::Resource.new_from_jsonb(metadata)
×
UNCOV
551
    end
×
552

UNCOV
553
  private
×
554

UNCOV
555
    def publish(user)
×
UNCOV
556
      datacite_service.publish_doi(user)
×
UNCOV
557
      update_ark_information
×
UNCOV
558
      publish_precurated_files(user)
×
UNCOV
559
      save!
×
UNCOV
560
    end
×
561

562
    # Update EZID (our provider of ARKs) with the new information for this work.
UNCOV
563
    def update_ark_information
×
564
      # We only want to update the ark url under certain conditions.
565
      # Set this value in config/update_ark_url.yml
UNCOV
566
      if Rails.configuration.update_ark_url
×
UNCOV
567
        if ark.present?
×
UNCOV
568
          Ark.update(ark, datacite_service.doi_attribute_url)
×
UNCOV
569
        end
×
UNCOV
570
      end
×
UNCOV
571
    end
×
572

UNCOV
573
    def track_state_change(user, state = aasm.to_state)
×
UNCOV
574
      uw = UserWork.new(user_id: user.id, work_id: id, state:)
×
UNCOV
575
      uw.save!
×
UNCOV
576
      WorkActivity.add_work_activity(id, "marked as #{state.to_s.titleize}", user.id, activity_type: WorkActivity::SYSTEM)
×
UNCOV
577
      WorkStateTransitionNotification.new(self, user.id).send
×
UNCOV
578
    end
×
579

580
    # Request S3 Bucket Objects associated with this Work
581
    # @return [Array<S3File>]
UNCOV
582
    def s3_resources
×
UNCOV
583
      data_profile = s3_query_service.data_profile
×
UNCOV
584
      data_profile.fetch(:objects, [])
×
UNCOV
585
    end
×
UNCOV
586
    alias pre_curation_s3_resources s3_resources
×
587

UNCOV
588
    def s3_object_persisted?(s3_file)
×
589
      uploads_keys = uploads.map(&:key)
×
590
      uploads_keys.include?(s3_file.key)
×
UNCOV
591
    end
×
592

UNCOV
593
    def publish_precurated_files(user)
×
594
      # We need to explicitly check the to post-curation bucket here.
UNCOV
595
      s3_post_curation_query_service = S3QueryService.new(self, "postcuration")
×
596

UNCOV
597
      s3_dir = find_post_curation_s3_dir(bucket_name: s3_post_curation_query_service.bucket_name)
×
UNCOV
598
      raise(StandardError, "Attempting to publish a Work with an existing S3 Bucket directory for: #{s3_object_key}") unless s3_dir.nil?
×
599

600
      # Copy the pre-curation S3 Objects to the post-curation S3 Bucket...
UNCOV
601
      s3_query_service.publish_files(user)
×
UNCOV
602
    end
×
603

UNCOV
604
    def latest_snapshot
×
UNCOV
605
      return upload_snapshots.first unless upload_snapshots.empty?
×
606

UNCOV
607
      UploadSnapshot.new(work: self, files: [])
×
UNCOV
608
    end
×
609

UNCOV
610
    def datacite_service
×
UNCOV
611
      @datacite_service ||= PULDatacite.new(self)
×
UNCOV
612
    end
×
613

UNCOV
614
    def files_as_json(*args)
×
UNCOV
615
      return [] if embargoed?
×
616

UNCOV
617
      force_post_curation = args.any? { |arg| arg[:force_post_curation] == true }
×
618

619
      # Pre-curation files are not accessible externally,
620
      # so we are not interested in listing them in JSON.
UNCOV
621
      post_curation_uploads(force_post_curation:).map do |upload|
×
UNCOV
622
        {
×
UNCOV
623
          "filename": upload.filename,
×
UNCOV
624
          "size": upload.size,
×
UNCOV
625
          "display_size": upload.display_size,
×
UNCOV
626
          "url": upload.globus_url
×
UNCOV
627
        }
×
UNCOV
628
      end
×
UNCOV
629
    end
×
630

UNCOV
631
    def embargo_date_as_json
×
UNCOV
632
      if embargo_date.present?
×
UNCOV
633
        embargo_datetime = embargo_date.to_datetime
×
UNCOV
634
        embargo_date_iso8601 = embargo_datetime.iso8601
×
635
        # Apache Solr timestamps require the following format:
636
        # 1972-05-20T17:33:18Z
637
        # https://solr.apache.org/guide/solr/latest/indexing-guide/date-formatting-math.html
UNCOV
638
        embargo_date_iso8601.gsub(/\+.+$/, "Z")
×
UNCOV
639
      end
×
UNCOV
640
    end
×
641

UNCOV
642
    def log_performance(start, message)
×
UNCOV
643
      elapsed = Time.zone.now - start
×
UNCOV
644
      if elapsed > 20
×
645
        Rails.logger.warn("PERFORMANCE: #{message}. Elapsed: #{elapsed} seconds")
×
UNCOV
646
      else
×
UNCOV
647
        Rails.logger.info("PERFORMANCE: #{message}. Elapsed: #{elapsed} seconds")
×
UNCOV
648
      end
×
UNCOV
649
    end
×
UNCOV
650
end
×
651
# rubocop:enable Metrics/ClassLength
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc