From 149156a534df4acdfcf1d5371febb2260e7fc4f0 Mon Sep 17 00:00:00 2001 From: Florian Lentsch Date: Sun, 1 Sep 2024 21:47:08 +0200 Subject: [PATCH] Fixes #99 --- app/models/article_version.rb | 26 ++++++----- app/models/order_article.rb | 1 + config/locales/de.yml | 2 + config/locales/en.yml | 2 + spec/factories/article.rb | 2 + spec/models/order_article_spec.rb | 71 +++++++++++++++++++++++++++++++ 6 files changed, 93 insertions(+), 11 deletions(-) diff --git a/app/models/article_version.rb b/app/models/article_version.rb index c447bd55..9bed9aa3 100644 --- a/app/models/article_version.rb +++ b/app/models/article_version.rb @@ -40,14 +40,13 @@ class ArticleVersion < ApplicationRecord validates :price, numericality: { greater_than_or_equal_to: 0 } validates :group_order_granularity, numericality: { greater_than_or_equal_to: 0 } validates :deposit, :tax, numericality: true - validates :minimum_order_quantity, - numericality: { allow_nil: true, only_integer: false, if: :supplier_order_unit_is_si_convertible } - validates :minimum_order_quantity, - numericality: { allow_nil: true, only_integer: true, unless: :supplier_order_unit_is_si_convertible } + validates :minimum_order_quantity, numericality: { allow_nil: true } + # validates_uniqueness_of :name, :scope => [:supplier_id, :deleted_at, :type], if: Proc.new {|a| a.supplier.shared_sync_method.blank? or a.supplier.shared_sync_method == 'import' } # validates_uniqueness_of :name, :scope => [:supplier_id, :deleted_at, :type, :unit, :unit_quantity] validate :uniqueness_of_name validate :only_one_unit_type + validate :minimum_order_quantity_as_integer, unless: :supplier_order_unit_is_si_convertible # Replace numeric seperator with database format localize_input_of :price, :tax, :deposit @@ -117,13 +116,7 @@ def minimum_order_quantity=(value) if value.blank? self[:minimum_order_quantity] = nil else - value = value.gsub(I18n.t('number.format.separator'), '.') if value.is_a?(String) - begin - value = value.to_i if Float(value) % 1 == 0 - rescue ArgumentError - # not any number -> let validation handle this - end - super(value) + super end end @@ -190,6 +183,17 @@ def uniqueness_of_name end end + def minimum_order_quantity_as_integer + begin + return if Float(minimum_order_quantity) % 1 == 0 + rescue ArgumentError, TypeError + # not any number -> let numericality validation handle this + return + end + + errors.add(:minimum_order_quantity, :only_integer) + end + def only_one_unit_type return if unit.blank? || supplier_order_unit.blank? diff --git a/app/models/order_article.rb b/app/models/order_article.rb index 0b29c4d4..a176675f 100644 --- a/app/models/order_article.rb +++ b/app/models/order_article.rb @@ -81,6 +81,7 @@ def update_results! # 4 | 5 | 4 | 2 # def calculate_units_to_order(quantity, tolerance = 0) + return 0 if !price.minimum_order_quantity.nil? && quantity + tolerance < price.minimum_order_quantity return price.minimum_order_quantity if quantity > 0 && !price.minimum_order_quantity.nil? && quantity < price.minimum_order_quantity && quantity + tolerance >= price.minimum_order_quantity unit_size = price.convert_quantity(1, price.supplier_order_unit, price.group_order_unit) diff --git a/config/locales/de.yml b/config/locales/de.yml index bf0cdd7c..6a699ea6 100644 --- a/config/locales/de.yml +++ b/config/locales/de.yml @@ -272,6 +272,8 @@ de: name: taken: Name ist bereits vergeben taken_with_unit: Name und Einheit sind bereits vergeben + minimum_order_quantity: + only_integer: Muss eine Ganzzahl sein, wenn die Liefereinheit eine Stückeinheit ist. supplier: attributes: shared_sync_method: diff --git a/config/locales/en.yml b/config/locales/en.yml index 480243cb..6129e6dc 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -274,6 +274,8 @@ en: name: taken: name is already taken taken_with_unit: name and unit are already taken + minimum_order_quantity: + only_integer: Must be an integer if supplier order unit is a piece unit supplier: attributes: shared_sync_method: diff --git a/spec/factories/article.rb b/spec/factories/article.rb index f64cfab1..ce6e6b39 100644 --- a/spec/factories/article.rb +++ b/spec/factories/article.rb @@ -10,6 +10,7 @@ order_number { nil } unit_quantity { nil } unit { nil } + minimum_order_quantity { nil } supplier_order_unit { 'XPK' } group_order_unit { 'XPK' } billing_unit { 'XPK' } @@ -23,6 +24,7 @@ order_number: evaluator.order_number, unit_quantity: evaluator.unit_quantity, unit: evaluator.unit, + minimum_order_quantity: evaluator.minimum_order_quantity, supplier_order_unit: evaluator.supplier_order_unit, group_order_unit: evaluator.group_order_unit, billing_unit: evaluator.billing_unit, diff --git a/spec/models/order_article_spec.rb b/spec/models/order_article_spec.rb index a0bc9671..b1c71a1a 100644 --- a/spec/models/order_article_spec.rb +++ b/spec/models/order_article_spec.rb @@ -213,4 +213,75 @@ def goa_reload # end end end + + describe 'minimum order quantity' do + let(:order) { create(:order, article_ids: [article.id], starts: 1.week.ago) } + let(:oa) { order.order_articles.first } + let(:go) { create(:group_order, order: order) } + let(:goa) { create(:group_order_article, group_order: go, order_article: oa) } + + describe 'without unit_quantity' do + let(:article) { create(:article, minimum_order_quantity: 10) } + + it 'doesn\'t order anything, if the minimum order quantity hasn\'t been reached' do + goa.update_quantities(4, 0) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 0 + end + + it 'orders the minimum order quantity, if tolerance allows for it' do + goa.update_quantities(9, 1) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 10 + end + + it 'orders the desired amount, if minimum order quantity has been surpassed' do + goa.update_quantities(12, 1) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 12 + end + end + + describe 'with unit_quantity' do + let(:article) { create(:article, minimum_order_quantity: 2, unit_quantity: 5) } + + it 'doesn\'t order anything if the minimum order quantity hasn\'t been reached' do + goa.update_quantities(4, 0) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 0 + end + + it 'orders the minimum order quantity if tolerance allows for it' do + goa.update_quantities(9, 1) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 2 + end + + it 'orders a lower amount, even if minimum order quantity has been surpassed, but unit_quantity demands it' do + goa.update_quantities(12, 1) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 2 + end + + it 'orders the exact amount, if minimum order quantity has been surpassed and unit_quantity allows for it' do + goa.update_quantities(14, 1) + oa.update_results! + oa.reload + + expect(oa.units_to_order).to eq 3 + end + end + end end