require_relative '../spec_helper'

describe Order do
  let!(:ftt) { create :financial_transaction_type }
  let(:user) { create :user, groups: [create(:ordergroup)] }

  it 'automaticly finishes ended' do
    create :order, created_by: user, starts: Date.yesterday, ends: 1.hour.from_now
    create :order, created_by: user, starts: Date.yesterday, ends: 1.hour.ago
    create :order, created_by: user, starts: Date.yesterday, ends: 1.hour.from_now, end_action: :auto_close
    order = create :order, created_by: user, starts: Date.yesterday, ends: 1.hour.ago, end_action: :auto_close

    Order.finish_ended!
    order.reload

    expect(Order.open.count).to eq 3
    expect(Order.finished.count).to eq 1
    expect(order).to be_finished
  end

  describe 'state scopes and boolean getters' do
    let!(:open_order)     { create :order, state: 'open'     }
    let!(:finished_order) { create :order, state: 'finished' }
    let!(:received_order) { create :order, state: 'received' }
    let!(:closed_order)   { create :order, state: 'closed'   }

    it 'retrieves open orders in the "open" scope' do
      expect(Order.open.count).to eq(1)
      expect(Order.open.where(id: open_order.id)).to exist
    end

    it 'retrieves finished, received and closed orders in the "finished" scope' do
      expect(Order.finished.count).to eq(3)
      expect(Order.finished.where(id: finished_order.id)).to exist
      expect(Order.finished.where(id: received_order.id)).to exist
      expect(Order.finished.where(id: closed_order.id)).to exist
    end

    it 'retrieves finished and received orders in the "finished_not_closed" scope' do
      expect(Order.finished_not_closed.count).to eq(2)
      expect(Order.finished_not_closed.where(id: finished_order.id)).to exist
      expect(Order.finished_not_closed.where(id: received_order.id)).to exist
    end

    it 'returns valid boolean states for open orders' do
      expect(open_order.open?).to be(true)
      expect(open_order.finished?).to be(false)
      expect(open_order.received?).to be(false)
      expect(open_order.closed?).to be(false)
    end

    it 'returns valid boolean states for finished orders' do
      expect(finished_order.open?).to be(false)
      expect(finished_order.finished?).to be(true)
      expect(finished_order.received?).to be(false)
      expect(finished_order.closed?).to be(false)
    end

    it 'returns valid boolean states for received orders' do
      expect(received_order.open?).to be(false)
      expect(received_order.finished?).to be(true)
      expect(received_order.received?).to be(true)
      expect(received_order.closed?).to be(false)
    end

    it 'returns valid boolean states for closed orders' do
      expect(closed_order.open?).to be(false)
      expect(closed_order.finished?).to be(false)
      expect(closed_order.received?).to be(false)
      expect(closed_order.closed?).to be(true)
    end
  end

  it 'sends mail if min_order_quantity has been reached' do
    create :user, groups: [create(:ordergroup)]
    create :order, created_by: user, starts: Date.yesterday, ends: 1.hour.ago, end_action: :auto_close_and_send_min_quantity

    Order.finish_ended!
    expect(ActionMailer::Base.deliveries.count).to eq 1
  end

  it 'needs a supplier' do
    expect(build(:order, supplier: nil)).to be_invalid
  end

  it 'needs order articles' do
    supplier = create :supplier, article_count: 0
    expect(build(:order, supplier: supplier)).to be_invalid
  end

  it 'can be created' do
    expect(build(:order, article_count: 1)).to be_valid
  end

  describe 'with articles' do
    let(:order) { create :order }

    it 'is open by default'         do expect(order).to be_open end
    it 'is not finished by default' do expect(order).to_not be_finished end
    it 'is not closed by default'   do expect(order).to_not be_closed end

    it 'has valid order articles' do
      order.order_articles.each { |oa| expect(oa).to be_valid }
    end

    it 'can be finished' do
      # TODO randomise user
      order.finish!(user)
      expect(order).to_not be_open
      expect(order).to be_finished
      expect(order).to_not be_closed
    end

    it 'can be closed' do
      # TODO randomise user
      order.finish!(user)
      order.close!(user)
      expect(order).to_not be_open
      expect(order).to be_closed
    end
  end

  describe 'with a default end date' do
    let(:order) { create :order }

    before do
      FoodsoftConfig[:order_schedule] = { ends: { recurr: 'FREQ=WEEKLY;BYDAY=MO', time: '9:00' } }
      order.init_dates
    end

    it 'to have a correct date' do
      expect(order.ends.to_date).to eq Date.today.next_week.at_beginning_of_week(:monday)
    end

    it 'to have a correct time' do
      expect(order.ends.strftime('%H:%M')).to eq '09:00'
    end
  end

  describe 'mapped to GroupOrders' do
    let!(:user) { create :user, groups: [create(:ordergroup)] }
    let!(:order) { create :order }
    let!(:order2) { create :order }
    let!(:go) { create :group_order, order: order, ordergroup: user.ordergroup }

    it 'to map a user\'s GroupOrders to a list of Orders' do
      orders = Order.ordergroup_group_orders_map(user.ordergroup)

      expect(orders.length).to be 2
      expect(orders[0][:order]).to have_attributes(id: order.id)
      expect(orders[0][:group_order]).to have_attributes(id: go.id)
      expect(orders[1][:order]).to have_attributes(id: order2.id)
      expect(orders[1][:group_order]).to be_nil
    end
  end

  describe 'balancing charges correct amounts' do
    let!(:transport) { rand(0.1..26.0).round(2) }
    let!(:order) { create :order, article_count: 1 }
    let!(:oa) { order.order_articles.first }
    let!(:go) { create :group_order, order: order, transport: transport }
    let!(:goa) { create :group_order_article, group_order: go, order_article: oa, quantity: 1 }

    before do
      goa.update_quantities(1, 0)
      go.reload
      go.update_price!
      user.ordergroup.update_stats!
      oa.update_results!

      order.finish!(user)
      order.reload
      order.close!(user, ftt)
    end

    it 'creates financial transaction with correct amount' do
      expect(goa.result).to be > 0
      expect(go.ordergroup.financial_transactions.last.amount).to eq(-go.total)
    end
  end
end