-
Notifications
You must be signed in to change notification settings - Fork 55
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: data versioning #2068
base: main
Are you sure you want to change the base?
feat: data versioning #2068
Conversation
@@ -0,0 +1,98 @@ | |||
require 'open-uri' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/FrozenStringLiteralComment: Missing frozen string literal comment.
@@ -0,0 +1,2 @@ | |||
class ContainerHierarchy < ApplicationRecord |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/FrozenStringLiteralComment: Missing frozen string literal comment.
@@ -0,0 +1,60 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Fetchers::ReactionFetcher |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
@@ -0,0 +1,34 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Fetchers::ScreenFetcher |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
new(**args).call | ||
end | ||
|
||
def call |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/AbcSize: Assignment Branch Condition size for call is too high. [<14, 33, 4> 36.07/25]
versions += Versioning::Serializers::ContainerSerializer.call(analysis, ["Analysis: #{analysis.name}"]) | ||
|
||
analysis.children.with_deleted.with_log_data.each do |dataset| | ||
versions += Versioning::Serializers::ContainerSerializer.call(dataset, ["Analysis: #{analysis.name}", "Dataset: #{dataset.name}"]) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [138/120]
versions += Versioning::Serializers::ContainerSerializer.call(dataset, ["Analysis: #{analysis.name}", "Dataset: #{dataset.name}"]) | ||
|
||
dataset.attachments.with_log_data.each do |attachment| | ||
versions += Versioning::Serializers::AttachmentSerializer.call(attachment, ["Analysis: #{analysis.name}", "Dataset: #{dataset.name}", "Attachment: #{attachment.filename}"]) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [182/120]
end | ||
|
||
screen.research_plans.each do |research_plan| | ||
versions += Versioning::Fetchers::ResearchPlanFetcher.call(research_plan: research_plan, prefix: "Research Plan: #{research_plan.name}") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [142/120]
@@ -0,0 +1,38 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Merger |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
|
||
merged_versions = [] | ||
grouped_versions.each_with_index do |(_, v), index| | ||
changes = v.map do |v| |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Lint/ShadowingOuterLocalVariable: Shadowing outer local variable - v
.
@@ -0,0 +1,90 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Serializers::ContainerSerializer < Versioning::Serializers::BaseSerializer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
new(record: record, name: name).call | ||
end | ||
|
||
def field_definitions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/AbcSize: Assignment Branch Condition size for field_definitions is too high. [<14, 31, 5> 34.38/25]
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I made a few suggestions to revert the changes to semantically unchanged files.
This would make the PR a bit smaller which would make working with the PR in the GitHub UI a bit easier. Atm, the UI is quite unresponsive due to to its size (large diffs in many files).
I'll continue the review over the coming days.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why do we need this stub?
import ScreensFetcher from 'src/fetchers/ScreensFetcher'; | ||
import WellplatesFetcher from 'src/fetchers/WellplatesFetcher'; | ||
|
||
export default class VersionsTable extends Component { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe convert this to a functional component? I feel like introducing new components might be an opportunity to move the codebase away from class components.
const alertText = revertibleFields() < 0 | ||
? 'You cannot undo changes. You have unsaved data which would be lost.' | ||
: `You cannot undo these changes. | ||
Either the changes are up to date, it is the first version or all changes are irreversible.`; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
From a user perspective it would be great if this alert would be more specific, i.e., the three cases that are lumped together in the fallback of the ternary could be differentiated.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please revert the formatting changes, otherwise this file's diff is unnecessarily hard to review.
app/packs/src/apps/mydb/elements/details/samples/propertiesTab/SampleForm.js
Show resolved
Hide resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please revert the formatting changes, otherwise this file's diff is unnecessarily hard to review.
app/models/sample.rb
Outdated
@@ -112,45 +113,59 @@ class Sample < ApplicationRecord | |||
pg_search_scope :search_by_cas, against: { xref: 'cas' } | |||
|
|||
# scopes for suggestions | |||
scope :by_residues_custom_info, ->(info, val) { joins(:residues).where("residues.custom_info -> '#{info}' ILIKE ?", "%#{sanitize_sql_like(val)}%")} | |||
scope :by_residues_custom_info, lambda { |info, val| | |||
joins(:residues).where("residues.custom_info -> '#{info}' ILIKE ?", "%#{sanitize_sql_like(val)}%") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [134/120]
app/models/sample.rb
Outdated
scope :by_name, ->(query) { where('name ILIKE ?', "%#{sanitize_sql_like(query)}%") } | ||
scope :by_sample_xref_cas, | ||
->(query) { where("xref ? 'cas'").where("xref ->> 'cas' ILIKE ?", "%#{sanitize_sql_like(query)}%") } | ||
scope :by_exact_name, ->(query) { where('lower(name) ~* lower(?) or lower(external_label) ~* lower(?)', "^([a-zA-Z0-9]+-)?#{sanitize_sql_like(query)}(-?[a-zA-Z])$", "^([a-zA-Z0-9]+-)?#{sanitize_sql_like(query)}(-?[a-zA-Z])$") } | ||
scope :by_exact_name, lambda { |query| | ||
where('lower(name) ~* lower(?) or lower(external_label) ~* lower(?)', "^([a-zA-Z0-9]+-)?#{sanitize_sql_like(query)}(-?[a-zA-Z])$", "^([a-zA-Z0-9]+-)?#{sanitize_sql_like(query)}(-?[a-zA-Z])$") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [217/120]
app/models/sample.rb
Outdated
scope :by_reaction_solvent_ids, ->(ids) { joins(:reactions_solvent_samples).where('reactions_samples.reaction_id in (?)', ids) } | ||
scope :by_reaction_ids, ->(ids) { joins(:reactions_samples).where('reactions_samples.reaction_id in (?)', ids) } | ||
scope :by_reaction_reactant_ids, lambda { |ids| | ||
joins(:reactions_reactant_samples).where('reactions_samples.reaction_id in (?)', ids) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rails/WhereEquals: Use where(reactions_samples: { reaction_id: ids })
instead of manually constructing SQL.
app/models/sample.rb
Outdated
scope :by_reaction_solvent_ids, ->(ids) { joins(:reactions_solvent_samples).where('reactions_samples.reaction_id in (?)', ids) } | ||
scope :by_reaction_ids, ->(ids) { joins(:reactions_samples).where('reactions_samples.reaction_id in (?)', ids) } | ||
scope :by_reaction_reactant_ids, lambda { |ids| | ||
joins(:reactions_reactant_samples).where('reactions_samples.reaction_id in (?)', ids) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [122/120]
app/models/sample.rb
Outdated
joins(:reactions_reactant_samples).where('reactions_samples.reaction_id in (?)', ids) | ||
} | ||
scope :by_reaction_product_ids, lambda { |ids| | ||
joins(:reactions_product_samples).where('reactions_samples.reaction_id in (?)', ids) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rails/WhereEquals: Use where(reactions_samples: { reaction_id: ids })
instead of manually constructing SQL.
app/models/sample.rb
Outdated
self.creator.increment_counter 'samples' | ||
return unless /^#{creator.name_abbreviation}-\d+$/.match?(short_label) | ||
|
||
creator.increment_counter 'samples' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rails/SkipsModelValidations: Avoid using increment_counter
because it skips validations.
@@ -132,7 +133,7 @@ def create_subwellplate(user, collection_ids, _copy_ea = false) | |||
readouts: w.readouts, | |||
additive: w.additive, | |||
position_x: w.position_x, | |||
position_y: w.position_y | |||
position_y: w.position_y, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/AbcSize: Assignment Branch Condition size for create_subwellplate is too high. [<11, 30, 4> 32.2/25]
@@ -132,7 +133,7 @@ def create_subwellplate(user, collection_ids, _copy_ea = false) | |||
readouts: w.readouts, | |||
additive: w.additive, | |||
position_x: w.position_x, | |||
position_y: w.position_y | |||
position_y: w.position_y, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/MethodLength: Method has too many lines. [31/30]
end | ||
|
||
def call | ||
changes.each do |change| |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/BlockLength: Block has too many lines. [26/25]
@@ -0,0 +1,7 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Reverters::WellReverter < Versioning::Reverters::BaseReverter |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
@@ -0,0 +1,7 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Reverters::WellplateReverter < Versioning::Reverters::BaseReverter |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
new(record: record, name: name).call | ||
end | ||
|
||
def field_definitions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/MethodLength: Method has too many lines. [73/30]
JSON.parse(result).join("\n") | ||
}, | ||
revert: %i[extended_metadata.hyperlinks], | ||
revertible_value_formatter: ->(key, value) { JSON.parse(jsonb_formatter('hyperlinks').call(key, value) || '[]') }, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Layout/LineLength: Line is too long. [124/120]
@@ -0,0 +1,33 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Serializers::ElementalCompositionSerializer < Versioning::Serializers::BaseSerializer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
@@ -0,0 +1,88 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Serializers::ReactionSerializer < Versioning::Serializers::BaseSerializer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
end | ||
end | ||
|
||
def lookups |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/PerceivedComplexity: Perceived complexity for lookups is too high. [10/8]
@@ -0,0 +1,68 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Serializers::ResidueSerializer < Versioning::Serializers::BaseSerializer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
new(record: record, name: name).call | ||
end | ||
|
||
def field_definitions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/MethodLength: Method has too many lines. [44/30]
@@ -0,0 +1,41 @@ | |||
# frozen_string_literal: true | |||
|
|||
class Versioning::Serializers::ScreenSerializer < Versioning::Serializers::BaseSerializer |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Style/ClassAndModuleChildren: Use nested module/class definitions instead of compact style.
new(record: record, name: name).call | ||
end | ||
|
||
def field_definitions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Metrics/MethodLength: Method has too many lines. [31/30]
const isRevertible = revertibleFields(); | ||
let alertText = 'You cannot undo changes. '; | ||
if (isRevertible === 0) { | ||
alertText += 'Every change is either up to date or irreversible'; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Now the first and last conditional branch both contain the "all changes are irreversible" case.
LCOV of commit
|
afdcee9
to
89dd152
Compare
LCOV of commit
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This feature needs tests 🧐
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tests look ok. Nice that there's at least some basic coverage now.
Store a history which keeps track of every change made to certain entities and add a "History" tab to the Reaction/Sample detail pages. Related entities are displayed in the version history. E.g. Reaction -> ReactionsSample Save timestamp and author of the version with the changes. A change diff to the previous version can be displayed. Tracked entities: - Attachment - Container - ElementalComposition - Reaction - ReactionsSample - Residue - Sample Co-authored-by: VadimKeller <[email protected]>
Add revert button to versioned field reset colum information re-create schema update solvent label
Make revert function work (#780) - Update version list after reverting a change - Fix issue /w big svgs - Make reverting molfile / svg work - Fix merge issues - Replace passing local variables /w using state Add missing versioning for Screens, Wellplates, and Research plans Change the way versions are displayed in the version table. Previously we had one version per database update. Now we group versions by the request id. These changes are now merged to make it more intuitive for the user. Implement revert functionality /w fetchers, reverters, and serializers Implement a new revert UI with one button per version and one modal to select revertible values. Delete old revert button. Co-authored-by: Martin Schneider <[email protected]> Co-authored-by: VadimKeller <[email protected]>
LCOV of commit
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tested the latest implementation in the UI. I deleted a starting material from a reaction and saved the reactions. That resulted in two observations that puzzle me a bit:
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@headri and I looked into the two issues I posted earlier:
-
Seems to be an issue with the UI, not the internal logic. The "previous version column" does contain data, but it's only visible on hover-over or when enlarging the column.
-
Happens because the reaction is incorrectly marked as change / unsaved. This problem does not seem to have been introduced with this PR though.
See #1253.