From 56ebc3dc076336dc113b2947f4446a96776c33f7 Mon Sep 17 00:00:00 2001 From: miha Date: Sat, 10 Apr 2021 12:33:32 +0200 Subject: [PATCH 1/7] [SCI-5635] add a migration to fix broken steps, fix the api so it sets last_modified_by_id, minor step model refactor migration fix remove the conditoin for set_last_modified_by callback set the last_modified_by_id in before_save callback (in case it is not set yet) remove trailing spaces Set last modified in protocl import service --- app/controllers/api/v1/steps_controller.rb | 7 ++- app/models/step.rb | 46 +++++++++++-------- .../import_protocol_service.rb | 3 +- ...0410100006_fix_last_changed_by_on_steps.rb | 33 +++++++++++++ spec/requests/api/v1/steps_controller_spec.rb | 11 +++++ 5 files changed, 79 insertions(+), 21 deletions(-) create mode 100644 db/migrate/20210410100006_fix_last_changed_by_on_steps.rb diff --git a/app/controllers/api/v1/steps_controller.rb b/app/controllers/api/v1/steps_controller.rb index 10f3c5897..1bbebd5b2 100644 --- a/app/controllers/api/v1/steps_controller.rb +++ b/app/controllers/api/v1/steps_controller.rb @@ -32,13 +32,16 @@ module Api step = @protocol.steps.create!(step_params.merge!(completed: false, user: current_user, - position: @protocol.number_of_steps)) + position: @protocol.number_of_steps, + last_modified_by_id: current_user.id)) render jsonapi: step, serializer: StepSerializer, status: :created end def update - @step.assign_attributes(step_params) + @step.assign_attributes( + step_params.merge!(last_modified_by_id: current_user.id) + ) if @step.changed? && @step.save! if @step.saved_change_to_attribute?(:completed) diff --git a/app/models/step.rb b/app/models/step.rb index ed9f6f2e2..ee2589583 100644 --- a/app/models/step.rb +++ b/app/models/step.rb @@ -107,11 +107,6 @@ class Step < ApplicationRecord StepComment.from(comments, :comments).order(created_at: :asc) end - def save(current_user=nil) - @current_user = current_user - super() - end - def space_taken st = 0 assets.each do |asset| @@ -202,21 +197,36 @@ class Step < ApplicationRecord end def set_last_modified_by - if @current_user&.is_a?(User) - self.tables.each do |t| - t.created_by ||= @current_user - t.last_modified_by = @current_user if t.changed? + self.last_modified_by_id ||= user_id + + tables.each do |t| + t.created_by ||= last_modified_by_id + if t.changed? + t.last_modified_by = last_modified_by_id + t.save! end - self.assets.each do |a| - a.created_by ||= @current_user - a.last_modified_by = @current_user if a.changed? + end + + assets.each do |a| + a.created_by ||= last_modified_by_id + if a.changed? + a.last_modified_by = last_modified_by_id + a.save! end - self.checklists.each do |checklist| - checklist.created_by ||= @current_user - checklist.last_modified_by = @current_user if checklist.changed? - checklist.checklist_items.each do |checklist_item| - checklist_item.created_by ||= @current_user - checklist_item.last_modified_by = @current_user if checklist_item.changed? + end + + checklists.each do |checklist| + checklist.created_by ||= last_modified_by_id + if checklist.changed? + checklist.last_modified_by = last_modified_by_id + checklist.save! + end + + checklist.checklist_items.each do |checklist_item| + checklist_item.created_by ||= last_modified_by_id + if checklist_item.changed? + checklist_item.last_modified_by = last_modified_by_id + checklist_item.save! end end end diff --git a/app/services/protocol_importers/import_protocol_service.rb b/app/services/protocol_importers/import_protocol_service.rb index 2074b57a1..ead924f01 100644 --- a/app/services/protocol_importers/import_protocol_service.rb +++ b/app/services/protocol_importers/import_protocol_service.rb @@ -24,7 +24,8 @@ module ProtocolImporters s = Step.new(step_params .symbolize_keys .slice(:name, :position, :description, :tables_attributes) - .merge(user: @user, completed: false)) + .merge(user: @user, completed: false) + .merge(last_modified_by_id: @user.id)) # 'Manually' create assets here. "Accept nasted attributes" won't work for assets s.assets << AttachmentsBuilder.generate(step_params.deep_symbolize_keys, user: @user, team: @team) diff --git a/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb new file mode 100644 index 000000000..1e5e2302d --- /dev/null +++ b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb @@ -0,0 +1,33 @@ +# frozen_string_literal: true + +class Step < ApplicationRecord + # Making sure we don't get undesired callbacks in the future + # But still enjoy the handyness of using the model +end + +class FixLastChangedByOnSteps < ActiveRecord::Migration[6.1] + # There was an issue with API steps endpoint not settings last_modified_by_id + # this migration fix the affected records + # At the moment it seems only the complete and uncomplete actions were problematic + # + # The enterprise build of SciNote also uses the Audit Trails + # In this instance we don't neet to worry about these changes + # because we're not recording the changes to last_modified_by_id + + def up + Step.where(last_modified_by_id: nil).each do |step| + # Try to find corresponding activity, if step was changed activity should exist + # we cannot completely rely on that, so for the fallback we use the step user + activity = Activity + .where(type_of: %i(complete_step uncomplete_step)) + .where("values -> 'message_items' -> 'step' ->> 'id' = ?", step.id.to_s) + .order(created_at: :desc) + .first + if activity && user_id = activity.values.dig('message_items', 'user', 'id') + step.update!(last_modified_by_id: user_id) + else + step.update!(last_modified_by_id: step.user_id) + end + end + end +end diff --git a/spec/requests/api/v1/steps_controller_spec.rb b/spec/requests/api/v1/steps_controller_spec.rb index 069da1851..71b85b29a 100644 --- a/spec/requests/api/v1/steps_controller_spec.rb +++ b/spec/requests/api/v1/steps_controller_spec.rb @@ -133,6 +133,12 @@ RSpec.describe 'Api::V1::StepsController', type: :request do ) ) end + + it 'sets the last_changed_by' do + action + + expect(Step.find(json['data']['id']).last_modified_by_id).to be @user.id + end end context 'when has missing param' do @@ -205,6 +211,11 @@ RSpec.describe 'Api::V1::StepsController', type: :request do ) ) end + + it 'sets the last_changed_by' do + action + expect(Step.find(json['data']['id']).last_modified_by_id).to be @user.id + end end context 'when has missing param' do From b917a0c3e0bdf07cb0741f3600cfdb5923bf5f7c Mon Sep 17 00:00:00 2001 From: miha Date: Mon, 12 Apr 2021 10:30:37 +0200 Subject: [PATCH 2/7] fix a stupid --- app/models/step.rb | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/app/models/step.rb b/app/models/step.rb index ee2589583..142a61fb3 100644 --- a/app/models/step.rb +++ b/app/models/step.rb @@ -200,32 +200,32 @@ class Step < ApplicationRecord self.last_modified_by_id ||= user_id tables.each do |t| - t.created_by ||= last_modified_by_id + t.created_by_id ||= last_modified_by_id if t.changed? - t.last_modified_by = last_modified_by_id + t.last_modified_by_id = last_modified_by_id t.save! end end assets.each do |a| - a.created_by ||= last_modified_by_id + a.created_by.id ||= last_modified_by_id if a.changed? - a.last_modified_by = last_modified_by_id + a.last_modified_by_id = last_modified_by_id a.save! end end checklists.each do |checklist| - checklist.created_by ||= last_modified_by_id + checklist.created_by_id ||= last_modified_by_id if checklist.changed? - checklist.last_modified_by = last_modified_by_id + checklist.last_modified_by_id = last_modified_by_id checklist.save! end checklist.checklist_items.each do |checklist_item| - checklist_item.created_by ||= last_modified_by_id + checklist_item.created_by_id ||= last_modified_by_id if checklist_item.changed? - checklist_item.last_modified_by = last_modified_by_id + checklist_item.last_modified_by_id = last_modified_by_id checklist_item.save! end end From 2e4509385656aa84ef832c7460eae64bfebb56ff Mon Sep 17 00:00:00 2001 From: miha Date: Mon, 12 Apr 2021 10:31:35 +0200 Subject: [PATCH 3/7] fix a stupid2 --- app/models/step.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/models/step.rb b/app/models/step.rb index 142a61fb3..3cb000b27 100644 --- a/app/models/step.rb +++ b/app/models/step.rb @@ -208,7 +208,7 @@ class Step < ApplicationRecord end assets.each do |a| - a.created_by.id ||= last_modified_by_id + a.created_by_id ||= last_modified_by_id if a.changed? a.last_modified_by_id = last_modified_by_id a.save! From 4f7f257b78fa10ee0593cc3628186a26df63193c Mon Sep 17 00:00:00 2001 From: miha Date: Mon, 12 Apr 2021 12:20:05 +0200 Subject: [PATCH 4/7] Remove cascading save from step --- app/models/step.rb | 32 -------------------------------- 1 file changed, 32 deletions(-) diff --git a/app/models/step.rb b/app/models/step.rb index 3cb000b27..940f52ecc 100644 --- a/app/models/step.rb +++ b/app/models/step.rb @@ -198,37 +198,5 @@ class Step < ApplicationRecord def set_last_modified_by self.last_modified_by_id ||= user_id - - tables.each do |t| - t.created_by_id ||= last_modified_by_id - if t.changed? - t.last_modified_by_id = last_modified_by_id - t.save! - end - end - - assets.each do |a| - a.created_by_id ||= last_modified_by_id - if a.changed? - a.last_modified_by_id = last_modified_by_id - a.save! - end - end - - checklists.each do |checklist| - checklist.created_by_id ||= last_modified_by_id - if checklist.changed? - checklist.last_modified_by_id = last_modified_by_id - checklist.save! - end - - checklist.checklist_items.each do |checklist_item| - checklist_item.created_by_id ||= last_modified_by_id - if checklist_item.changed? - checklist_item.last_modified_by_id = last_modified_by_id - checklist_item.save! - end - end - end end end From b6a4a5326f532a64e462dccfe55b9c41f8f067b2 Mon Sep 17 00:00:00 2001 From: miha Date: Mon, 12 Apr 2021 13:57:03 +0200 Subject: [PATCH 5/7] replace .find with .find_each --- db/migrate/20210410100006_fix_last_changed_by_on_steps.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb index 1e5e2302d..2d6adb46c 100644 --- a/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb +++ b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb @@ -15,7 +15,7 @@ class FixLastChangedByOnSteps < ActiveRecord::Migration[6.1] # because we're not recording the changes to last_modified_by_id def up - Step.where(last_modified_by_id: nil).each do |step| + Step.where(last_modified_by_id: nil).find_each do |step| # Try to find corresponding activity, if step was changed activity should exist # we cannot completely rely on that, so for the fallback we use the step user activity = Activity From 4c29de62bebb90d72056beac31cdba229d030485 Mon Sep 17 00:00:00 2001 From: miha Date: Tue, 13 Apr 2021 10:25:01 +0200 Subject: [PATCH 6/7] [SCI-5635] Use update_column in the migration in order to avoid callbacks --- db/migrate/20210410100006_fix_last_changed_by_on_steps.rb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb index 2d6adb46c..6f289b8be 100644 --- a/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb +++ b/db/migrate/20210410100006_fix_last_changed_by_on_steps.rb @@ -24,9 +24,9 @@ class FixLastChangedByOnSteps < ActiveRecord::Migration[6.1] .order(created_at: :desc) .first if activity && user_id = activity.values.dig('message_items', 'user', 'id') - step.update!(last_modified_by_id: user_id) + step.update_column(:last_modified_by_id, user_id) else - step.update!(last_modified_by_id: step.user_id) + step.update_column(:last_modified_by_id, step.user_id) end end end From a6b2b23800d9d196ba9f5eafde1af016ee524780 Mon Sep 17 00:00:00 2001 From: Miha Mencin Date: Mon, 19 Apr 2021 08:53:34 +0200 Subject: [PATCH 7/7] Bump version to 1.21.8 --- VERSION | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/VERSION b/VERSION index 8819d012c..428abfd24 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -1.21.7 +1.21.8