forked from fixin.me/fixin.me
defaults_diff returns base Units where needed
This commit is contained in:
parent
1790f2e7f2
commit
6c678b6560
@ -14,48 +14,57 @@ class Unit < ApplicationRecord
|
|||||||
validates :multiplier, numericality: {other_than: 0}, if: :base
|
validates :multiplier, numericality: {other_than: 0}, if: :base
|
||||||
|
|
||||||
scope :defaults, ->{ where(user: nil) }
|
scope :defaults, ->{ where(user: nil) }
|
||||||
scope :with_defaults, ->{ self.or(Unit.where(user: nil)) }
|
|
||||||
scope :defaults_diff, ->{
|
scope :defaults_diff, ->{
|
||||||
|
actionable_units = Arel::Table.new('actionable_units')
|
||||||
|
units = actionable_units.alias('units')
|
||||||
bases_units = arel_table.alias('bases_units')
|
bases_units = arel_table.alias('bases_units')
|
||||||
other_units = arel_table.alias('other_units')
|
other_units = arel_table.alias('other_units')
|
||||||
other_bases_units = arel_table.alias('other_bases_units')
|
other_bases_units = arel_table.alias('other_bases_units')
|
||||||
sub_units = arel_table.alias('sub_units')
|
sub_units = arel_table.alias('sub_units')
|
||||||
|
|
||||||
Unit.with(units: self.with_defaults).left_joins(:base)
|
Unit.with_recursive(actionable_units: [
|
||||||
# Exclude Units that are/have default counterpart
|
Unit.with(units: self.or(Unit.defaults)).left_joins(:base)
|
||||||
.where.not(
|
.where.not(
|
||||||
Arel::SelectManager.new.project(1).from(other_units)
|
# Exclude Units that are/have default counterpart
|
||||||
.outer_join(other_bases_units)
|
Arel::SelectManager.new.project(1).from(other_units)
|
||||||
.on(other_units[:base_id].eq(other_bases_units[:id]))
|
.outer_join(other_bases_units)
|
||||||
.where(
|
.on(other_units[:base_id].eq(other_bases_units[:id]))
|
||||||
other_bases_units[:symbol].is_not_distinct_from(bases_units[:symbol])
|
.where(
|
||||||
.and(other_units[:symbol].eq(arel_table[:symbol]))
|
other_bases_units[:symbol].is_not_distinct_from(bases_units[:symbol])
|
||||||
.and(other_units[:user_id].is_distinct_from(arel_table[:user_id]))
|
.and(other_units[:symbol].eq(arel_table[:symbol]))
|
||||||
).exists
|
.and(other_units[:user_id].is_distinct_from(arel_table[:user_id]))
|
||||||
# Decide if Unit can be im-/exported based on existing hierarchy:
|
).exists
|
||||||
# * same base unit symbol has to exist
|
)
|
||||||
# * unit with subunits can only be ported to root
|
.select(
|
||||||
).select(
|
arel_table[Arel.star],
|
||||||
arel_table[Arel.star],
|
# Decide if Unit can be im-/exported based on existing hierarchy:
|
||||||
arel_table[:base_id].eq(nil).or(
|
# * same base unit symbol has to exist
|
||||||
(
|
# * unit with subunits can only be ported to root
|
||||||
Arel::SelectManager.new.project(1).from(other_units)
|
arel_table[:base_id].eq(nil).or(
|
||||||
.join(sub_units).on(other_units[:id].eq(sub_units[:base_id]))
|
(
|
||||||
.where(
|
Arel::SelectManager.new.project(1).from(other_units)
|
||||||
other_units[:symbol].eq(arel_table[:symbol])
|
.join(sub_units).on(other_units[:id].eq(sub_units[:base_id]))
|
||||||
.and(other_units[:user_id].is_distinct_from(arel_table[:user_id]))
|
.where(
|
||||||
)
|
other_units[:symbol].eq(arel_table[:symbol])
|
||||||
.exists.not
|
.and(other_units[:user_id].is_distinct_from(arel_table[:user_id]))
|
||||||
).and(
|
).exists.not
|
||||||
Arel::SelectManager.new.project(1).from(other_bases_units)
|
).and(
|
||||||
.where(
|
Arel::SelectManager.new.project(1).from(other_bases_units)
|
||||||
other_bases_units[:symbol].is_not_distinct_from(bases_units[:symbol])
|
.where(
|
||||||
.and(other_bases_units[:user_id].is_distinct_from(bases_units[:user_id]))
|
other_bases_units[:symbol].is_not_distinct_from(bases_units[:symbol])
|
||||||
)
|
.and(other_bases_units[:user_id].is_distinct_from(bases_units[:user_id]))
|
||||||
.exists
|
).exists
|
||||||
)
|
)
|
||||||
).as('portable')
|
).as('portable')
|
||||||
)
|
),
|
||||||
|
# TODO: replace AS and MIN with Arel
|
||||||
|
# TODO: turn off ONLY_FULL_GROUP_BY
|
||||||
|
# Add missing base Units. Duplicates will be removed by final group(), as
|
||||||
|
# actionable Units will differ on 'portable' column and can't be UNION-ed.
|
||||||
|
arel_table.join(actionable_units).on(actionable_units[:base_id].eq(arel_table[:id]))
|
||||||
|
.project(arel_table[Arel.star], 'NULL AS portable')
|
||||||
|
]).select(units: column_names)#, 'MIN(units.portable)' => :portable)
|
||||||
|
.from(units).group(Unit.column_names)
|
||||||
}
|
}
|
||||||
scope :ordered, ->{
|
scope :ordered, ->{
|
||||||
left_outer_joins(:base)
|
left_outer_joins(:base)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user