scinote-web/app/models/my_module.rb
2024-08-28 11:19:27 +02:00

598 lines
22 KiB
Ruby

# frozen_string_literal: true
class MyModule < ApplicationRecord
ID_PREFIX = 'TA'
include PrefixedIdModel
SEARCHABLE_ATTRIBUTES = ['my_modules.name', 'my_modules.description', PREFIXED_ID_SQL,
'comments.message', 'tags.name', 'users.full_name', 'users.email'].freeze
include ArchivableModel
include SearchableModel
include SearchableByNameModel
include TinyMceImages
include PermissionCheckableModel
include Assignable
include Cloneable
attr_accessor :transition_error_rollback
enum state: Extends::TASKS_STATES
enum provisioning_status: { done: 0, in_progress: 1, failed: 2 }
before_validation :archiving_and_restoring_extras, on: :update, if: :archived_changed?
before_save -> { report_elements.destroy_all }, if: -> { !new_record? && experiment_id_changed? }
before_save :reset_due_date_notification_sent, if: -> { due_date_changed? }
around_save :exec_status_consequences, if: :my_module_status_id_changed?
before_create :create_blank_protocol
before_create :assign_default_status_flow
after_save -> { experiment.workflowimg.purge },
if: -> { (saved_changes.keys & %w(x y experiment_id my_module_group_id input_id output_id archived)).any? }
auto_strip_attributes :name, :description, nullify: false, if: proc { |mm| mm.name_changed? || mm.description_changed? }
validates :name,
length: { minimum: Constants::NAME_MIN_LENGTH,
maximum: Constants::NAME_MAX_LENGTH }
validates :description, length: { maximum: Constants::RICH_TEXT_MAX_LENGTH }
validates :x, :y, :workflow_order, presence: true
validates :experiment, presence: true
validates :my_module_group, presence: true, if: proc { |mm| !mm.my_module_group_id.nil? }
validate :coordinates_uniqueness_check, if: :active?
validates :completed_on, presence: true, if: proc { |mm| mm.completed? }
validate :check_status, if: :my_module_status_id_changed?
validate :check_status_conditions, if: :my_module_status_id_changed?
validate :check_status_implications
belongs_to :created_by, foreign_key: 'created_by_id', class_name: 'User', optional: true
belongs_to :last_modified_by, foreign_key: 'last_modified_by_id', class_name: 'User', optional: true
belongs_to :archived_by, foreign_key: 'archived_by_id', class_name: 'User', optional: true
belongs_to :restored_by, foreign_key: 'restored_by_id', class_name: 'User', optional: true
belongs_to :experiment, inverse_of: :my_modules, touch: true
has_one :project, through: :experiment, autosave: false
has_one :shareable_link, as: :shareable, dependent: :destroy
delegate :team, to: :project
belongs_to :my_module_group, inverse_of: :my_modules, optional: true
belongs_to :my_module_status, optional: true
belongs_to :changing_from_my_module_status, optional: true, class_name: 'MyModuleStatus'
delegate :my_module_status_flow, to: :my_module_status, allow_nil: true
has_many :results, inverse_of: :my_module, dependent: :destroy
has_many :results_include_discarded, -> { with_discarded }, class_name: 'Result', inverse_of: :my_module
has_many :my_module_tags, inverse_of: :my_module, dependent: :destroy
has_many :tags, through: :my_module_tags, dependent: :destroy
has_many :task_comments, foreign_key: :associated_id, dependent: :destroy
has_many :inputs, class_name: 'Connection', foreign_key: 'input_id', inverse_of: :to, dependent: :destroy
has_many :outputs, class_name: 'Connection', foreign_key: 'output_id', inverse_of: :from, dependent: :destroy
has_many :my_modules, through: :outputs, source: :to, class_name: 'MyModule'
has_many :my_module_antecessors, through: :inputs, source: :from, class_name: 'MyModule'
has_many :my_module_repository_rows, inverse_of: :my_module, dependent: :destroy
has_many :repository_rows, through: :my_module_repository_rows
has_many :repository_snapshots, dependent: :destroy, inverse_of: :my_module
has_many :user_my_modules, inverse_of: :my_module, dependent: :destroy
has_many :users, through: :user_assignments
has_many :designated_users, through: :user_my_modules, source: :user
has_many :report_elements, inverse_of: :my_module, dependent: :destroy
has_many :protocols, inverse_of: :my_module, dependent: :destroy
has_many :steps, through: :protocols
has_many :assets_in_steps, class_name: 'Asset', source: :assets, through: :steps
has_many :assets_in_results, class_name: 'Asset', source: :assets, through: :results
# Associations for old activity type
has_many :activities, inverse_of: :my_module
scope :overdue, -> { where('my_modules.due_date < ?', Time.current.utc) }
scope :without_group, -> { active.where(my_module_group: nil) }
scope :one_day_prior, (lambda do
where('my_modules.due_date > ? AND my_modules.due_date < ?',
Time.current.utc,
Time.current.utc + 1.day)
end)
scope :workflow_ordered, -> { order(workflow_order: :asc) }
scope :uncomplete, -> { where(state: 'uncompleted') }
scope :my_module_search_scope, lambda { |experiment_ids, user|
joins(:user_assignments).where(
experiment: experiment_ids,
user_assignments: { user: user }
).distinct
}
scope :repository_row_assignable_by_user, lambda { |user|
active
.joins(user_assignments: :user_role)
.where(user_assignments: { user: user })
.where('? = ANY(user_roles.permissions)', MyModulePermissions::REPOSITORY_ROWS_ASSIGN)
}
# A module takes this much space in canvas (x, y) in database
WIDTH = 30
HEIGHT = 14
def self.search(
user,
include_archived,
query = nil,
current_team = nil,
options = {}
)
teams = options[:teams] || current_team || user.teams.select(:id)
new_query = distinct.left_joins(:task_comments, my_module_tags: :tag, user_my_modules: :user)
.with_granted_permissions(user, MyModulePermissions::READ)
.where(user_assignments: { team: teams })
.where_attributes_like_boolean(SEARCHABLE_ATTRIBUTES, query, options)
unless include_archived
new_query = new_query.joins(experiment: :project)
.active
.where(experiments: { archived: false }, projects: { archived: false })
end
new_query
end
def self.viewable_by_user(user, teams)
with_granted_permissions(user, MyModulePermissions::READ)
.where(experiment: Experiment.viewable_by_user(user, teams))
end
def self.filter_by_teams(teams = [])
return self if teams.blank?
joins(experiment: :project).where(experiment: { projects: { team: teams } })
end
def self.approaching_due_dates
joins(experiment: :project)
.active
.where(
due_date_notification_sent: false,
projects: { archived: false },
experiments: { archived: false }
)
.where('my_modules.due_date > ? AND my_modules.due_date <= ?', DateTime.current, DateTime.current + 1.day)
end
def parent
experiment
end
def results_count(view_mode = 'active')
return results.size if archived_branch?
return results.archived.size if view_mode == 'archived'
results.active.size
end
def navigable?
!experiment.archived? && experiment.navigable?
end
def archived_branch?
archived? || experiment.archived_branch?
end
def repository_rows_count(repository)
my_module_repository_rows.joins(repository_row: :repository)
.where('repositories.id': repository.id)
.count
end
def assigned_repositories
Repository.joins(repository_rows: :my_module_repository_rows)
.where(my_module_repository_rows: { my_module_id: id })
.group(:id)
end
def readable_live_and_snapshot_repositories_list(user, team = user.current_team)
snapshots = repository_snapshots.left_outer_joins(:original_repository)
selected_snapshots = snapshots.where(selected: true)
.or(snapshots.where(original_repositories_repositories: { id: nil }))
.or(snapshots.where.not(parent_id: assigned_repositories.select(:id)))
.select('DISTINCT ON ("repositories"."parent_id") "repositories".*')
.select('COUNT(repository_rows.id) AS assigned_rows_count')
.joins(:repository_rows)
.group(:parent_id, :id)
.order(:parent_id, updated_at: :desc)
live_repositories = assigned_repositories
.viewable_by_user(user, team)
.select('repositories.*, COUNT(DISTINCT repository_rows.id) AS assigned_rows_count')
.where.not(id: repository_snapshots.where(selected: true).select(:parent_id))
(live_repositories + selected_snapshots).sort_by { |r| r.name.downcase }
end
def live_and_snapshot_repositories_list
snapshots = repository_snapshots.left_outer_joins(:original_repository)
selected_snapshots = snapshots.where(selected: true)
.or(snapshots.where(original_repositories_repositories: { id: nil }))
.or(snapshots.where.not(parent_id: assigned_repositories.select(:id)))
.select('DISTINCT ON ("repositories"."parent_id") "repositories".*')
.select('COUNT(repository_rows.id) AS assigned_rows_count')
.joins(:repository_rows)
.group(:parent_id, :id)
.order(:parent_id, updated_at: :desc)
live_repositories = assigned_repositories
.select('repositories.*, COUNT(DISTINCT repository_rows.id) AS assigned_rows_count')
.where.not(id: repository_snapshots.where(selected: true).select(:parent_id))
(live_repositories + selected_snapshots).sort_by { |r| r.name.downcase }
end
def update_report_repository_references(repository)
ids = if repository.is_a?(Repository)
RepositorySnapshot.where(parent_id: repository.id).pluck(:id)
else
Repository.where(id: repository.parent_id).pluck(:id) +
RepositorySnapshot.where(parent_id: repository.parent_id).pluck(:id)
end
report_elements.where(repository_id: ids).update(repository: repository)
end
def undesignated_users
User.joins(:user_assignments)
.joins(
"LEFT OUTER JOIN user_my_modules ON user_my_modules.user_id = users.id "\
"AND user_my_modules.my_module_id = #{id}"
)
.where(user_assignments: { assignable: self })
.where(user_my_modules: { id: nil })
.distinct
end
def unassigned_tags
experiment.project.tags.where.not(id: tags)
end
def last_activities(count = Constants::ACTIVITY_AND_NOTIF_SEARCH_LIMIT)
Activity.where(my_module_id: id).order(:created_at).last(count)
end
# Get module comments ordered by created_at time. Results are paginated
# using last comment id and per_page parameters.
def last_comments(last_id = 1, per_page = Constants::COMMENTS_SEARCH_LIMIT)
last_id = Constants::INFINITY if last_id <= 1
comments = TaskComment.joins(:my_module)
.where(my_modules: { id: id })
.where('comments.id < ?', last_id)
.order(created_at: :desc)
.limit(per_page)
TaskComment.from(comments, :comments).order(created_at: :asc)
end
def last_activities(last_id = 1,
count = Constants::ACTIVITY_AND_NOTIF_SEARCH_LIMIT)
last_id = Constants::INFINITY if last_id <= 1
Activity.joins(:my_module)
.where(my_module_id: id)
.where('activities.id < ?', last_id)
.order(created_at: :desc)
.limit(count)
.uniq
end
def protocol
# Temporary function (until we fully support
# multiple protocols per module)
protocols.count > 0 ? protocols.first : nil
end
def is_overdue?(datetime = DateTime.current)
due_date.present? && datetime.utc > due_date.utc
end
def overdue_for_days(datetime = DateTime.current)
if due_date.blank? || due_date.utc > datetime.utc
0
else
((datetime.utc.to_i - due_date.utc.to_i) / 1.day.to_f).ceil
end
end
def is_one_day_prior?(datetime = DateTime.current)
is_due_in?(datetime, 1.day)
end
def is_due_in?(datetime, diff)
due_date.present? &&
datetime.utc < due_date.utc &&
datetime.utc > (due_date.utc - diff)
end
def space_taken
st = 0
protocol.steps.find_each do |step|
st += step.space_taken
end
results
.includes(:result_asset)
.find_each do |result|
st += result.space_taken
end
st
end
def archived_results
results
.select('results.*')
.select('ra.id AS result_asset_id')
.select('rt.id AS result_table_id')
.select('rx.id AS result_text_id')
.joins('LEFT JOIN result_assets AS ra ON ra.result_id = results.id')
.joins('LEFT JOIN result_tables AS rt ON rt.result_id = results.id')
.joins('LEFT JOIN result_texts AS rx ON rx.result_id = results.id')
.where(:archived => true)
end
# Treat this module as root, get all modules of that subtree
def downstream_modules
final = []
modules = [self]
until modules.blank?
my_module = modules.shift
final << my_module unless final.include?(my_module)
modules.push(*my_module.my_modules)
end
final
end
# Treat this module as inversed root, get all modules of that inversed subtree
def upstream_modules
final = []
modules = [self]
until modules.blank?
my_module = modules.shift
final << my_module unless final.include?(my_module)
modules.push(*my_module.my_module_antecessors)
end
final
end
# Generate the repository rows belonging to this module
# in JSON form, suitable for display in handsontable.js
def repository_json_hot(repository, order)
# Prepare column headers
headers = [
I18n.t('repositories.table.id'),
I18n.t('repositories.table.row_name'),
I18n.t('repositories.table.added_on'),
I18n.t('repositories.table.added_by')
]
data = []
rows = repository.assigned_rows(self).includes(:created_by).order(created_at: order)
if repository.has_stock_management? && repository.has_stock_consumption?
headers.push(I18n.t('repositories.table.row_consumption'))
rows = rows.left_joins(my_module_repository_rows: :repository_stock_unit_item)
.select(
'repository_rows.*',
'my_module_repository_rows.stock_consumption'
)
end
rows.find_each do |row|
row_json = []
row_json << row.code
row_json << (row.archived ? "#{escape_script_tag(row.name)} [#{I18n.t('general.archived')}]" : escape_script_tag(row.name))
row_json << I18n.l(row.created_at, format: :full)
row_json << escape_script_tag(row.created_by.full_name)
if repository.has_stock_management? && repository.has_stock_consumption?
if repository.is_a?(RepositorySnapshot)
consumed_stock = escape_script_tag(row.repository_stock_consumption_cell&.value&.formatted)
row_json << (consumed_stock || 0)
else
row_json << escape_script_tag(row.row_consumption(row.stock_consumption))
end
end
data << row_json
end
{ data: data, headers: headers }
end
def repository_docx_json(repository)
headers = [
I18n.t('repositories.table.id'),
I18n.t('repositories.table.row_name'),
I18n.t('repositories.table.added_on'),
I18n.t('repositories.table.added_by')
]
custom_columns = []
return false unless repository
repository.repository_columns.order(:id).each do |column|
if column.data_type == 'RepositoryStockValue'
if repository.has_stock_consumption?
headers.push(I18n.t('repositories.table.row_consumption'))
custom_columns.push(column.id)
end
elsif column.data_type != 'RepositoryStockConsumptionValue' &&
!(repository.is_a?(RepositorySnapshot) && column.data_type == 'RepositoryStockConsumptionValue')
headers.push(column.name)
custom_columns.push(column.id)
end
end
records = repository.assigned_rows(self)
.select(:id, :name, :created_at, :created_by_id, :repository_id, :parent_id, :archived)
{ headers: headers, rows: records, custom_columns: custom_columns }
end
def deep_clone(current_user)
deep_clone_to_experiment(current_user, experiment)
end
def deep_clone_to_experiment(current_user, experiment_dest)
# Copy the module
clone = MyModule.new(
name: name,
experiment: experiment_dest,
description: description,
x: x,
y: y,
created_by: current_user
)
# set new position if cloning in the same experiment
clone.attributes = get_new_position if clone.experiment == experiment
clone.save!
clone.assign_user(current_user)
copy_content(current_user, clone)
clone
end
def copy_content(current_user, target_my_module)
# Remove the automatically generated protocol,
# & clone the protocol instead
target_my_module.protocol.destroy
target_my_module.reload
# Update the cloned protocol if neccesary
clone_tinymce_assets(target_my_module, target_my_module.experiment.project.team)
protocol.deep_clone_my_module(target_my_module, current_user)
target_my_module.reload
end
# Find an empty position for the restored module. It's
# basically a first empty row with empty space inside x=[0, 32).
def get_new_position
return { x: 0, y: 0 } if experiment.blank?
# Get all modules position that overlap with first column, [0, WIDTH) and
# sort them by y coordinate.
positions = experiment.my_modules.active.collect { |m| [m.x, m.y] }
.select { |x, _| x >= 0 && x < WIDTH }
.sort_by { |_, y| y }
return { x: 0, y: 0 } if positions.blank? || positions.first[1] >= HEIGHT
# It looks we'll have to find a gap between the modules if it exists (at
# least 2*HEIGHT wide
ind = positions.each_cons(2).map { |f, s| s[1] - f[1] }
.index { |y| y >= 2 * HEIGHT }
return { x: 0, y: positions[ind][1] + HEIGHT } if ind
# We lucked out, no gaps, therefore we need to add it after the last element
{ x: 0, y: positions.last[1] + HEIGHT }
end
def assign_user(user, assigned_by = nil)
user_my_modules.create(
assigned_by: assigned_by || user,
user: user
)
Activities::CreateActivityService
.call(activity_type: :designate_user_to_my_module,
owner: assigned_by || user,
team: experiment.project.team,
project: experiment.project,
subject: self,
message_items: { my_module: id,
user_target: user.id })
end
def shared?
team.shareable_links_enabled? && shareable_link.present?
end
def comments
task_comments
end
def permission_parent
experiment
end
private
def create_blank_protocol
protocols << Protocol.new_blank_for_module(self)
end
def escape_script_tag(value)
value&.gsub(/\</, '&lt;')&.gsub(/\>/, '&gt;')
end
def coordinates_uniqueness_check
if experiment && experiment.my_modules.active.where(x: x, y: y).where.not(id: id).any?
errors.add(:position, I18n.t('activerecord.errors.models.my_module.attributes.position.not_unique'))
end
end
def assign_default_status_flow
return if my_module_status.present? || MyModuleStatusFlow.global.blank?
self.my_module_status = MyModuleStatusFlow.global.last.initial_status
end
def check_status_conditions
return if my_module_status.blank?
my_module_status.my_module_status_conditions.each do |condition|
condition.call(self)
end
end
def check_status_implications
return if my_module_status.blank?
my_module_status.my_module_status_implications.each do |implication|
implication.call(self)
end
end
def check_status
return unless my_module_status_id_was
original_status = MyModuleStatus.find_by(id: my_module_status_id_was)
unless my_module_status && [original_status.next_status, original_status.previous_status].include?(my_module_status)
errors.add(:my_module_status_id,
I18n.t('activerecord.errors.models.my_module.attributes.my_module_status_id.not_correct_order'))
end
end
def exec_status_consequences
return if my_module_status.blank? || status_changing
self.changing_from_my_module_status_id = my_module_status_id_was if my_module_status_id_was.present?
self.status_changing = true
status_changing_direction = my_module_status.previous_status_id == my_module_status_id_was ? :forward : :backward
yield
if my_module_status.my_module_status_consequences.any?(&:runs_in_background?)
MyModuleStatusConsequencesJob
.perform_later(self, my_module_status.my_module_status_consequences.to_a, status_changing_direction)
else
MyModuleStatusConsequencesJob
.perform_now(self, my_module_status.my_module_status_consequences.to_a, status_changing_direction)
end
end
def reset_due_date_notification_sent
self.due_date_notification_sent = false
end
def archiving_and_restoring_extras
if archived?
# Removes connections with other modules
self.x = 0
self.y = 0
# Remove association with module group.
self.my_module_group = nil
# Remove all connection between modules.
Connection.where(input_id: id).destroy_all
Connection.where(output_id: id).destroy_all
else
# Calculate new module position
new_pos = get_new_position
self.x = new_pos[:x]
self.y = new_pos[:y]
end
end
end