AyuntamientoMadrid/participacion

View on GitHub
spec/lib/user_segments_spec.rb

Summary

Maintainability
C
7 hrs
Test Coverage
require "rails_helper"

describe UserSegments do
  let(:user1) { create(:user) }
  let(:user2) { create(:user) }
  let(:user3) { create(:user) }

  describe ".segment_name" do
    it "returns a readable name of the segment" do
      expect(UserSegments.segment_name("all_users")).to eq "All users"
      expect(UserSegments.segment_name("administrators")).to eq "Administrators"
      expect(UserSegments.segment_name("proposal_authors")).to eq "Proposal authors"
    end

    it "accepts symbols as parameters" do
      expect(UserSegments.segment_name(:all_users)).to eq "All users"
    end

    it "returns nil for invalid segments" do
      expect(UserSegments.segment_name("invalid")).to be nil
    end

    context "with geozones in the database" do
      before do
        create(:geozone, name: "Lands and Borderlands")
        create(:geozone, name: "Lowlands and Highlands")
      end

      it "returns geozone names when the geozone exists" do
        expect(UserSegments.segment_name("lands_and_borderlands")).to eq "Lands and Borderlands"
        expect(UserSegments.segment_name("lowlands_and_highlands")).to eq "Lowlands and Highlands"
      end

      it "supports international alphabets" do
        create(:geozone, name: "Česká republika")
        create(:geozone, name: "България")
        create(:geozone, name: "日本")

        expect(UserSegments.segment_name("ceska_republika")).to eq "Česká republika"
        expect(UserSegments.segment_name("България")).to eq "България"
        expect(UserSegments.segment_name("日本")).to eq "日本"
      end

      it "returns regular segments when the geozone doesn't exist" do
        expect(UserSegments.segment_name("all_users")).to eq "All users"
      end

      it "returns nil for invalid segments" do
        expect(UserSegments.segment_name("invalid")).to be nil
      end
    end
  end

  describe ".valid_segment?" do
    it "returns true when the segment exists" do
      expect(UserSegments.valid_segment?("all_proposal_authors")).to be true
      expect(UserSegments.valid_segment?("investment_authors")).to be true
      expect(UserSegments.valid_segment?("feasible_and_undecided_investment_authors")).to be true
    end

    it "accepts symbols as parameters" do
      expect(UserSegments.valid_segment?(:selected_investment_authors)).to be true
      expect(UserSegments.valid_segment?(:winner_investment_authors)).to be true
      expect(UserSegments.valid_segment?(:not_supported_on_current_budget)).to be true
    end

    it "is falsey when the segment doesn't exist" do
      expect(UserSegments.valid_segment?("imaginary_segment")).to be_falsey
    end

    it "is falsey when nil is passed" do
      expect(UserSegments.valid_segment?(nil)).to be_falsey
    end

    context "with geozones in the database" do
      before do
        create(:geozone, name: "Lands and Borderlands")
        create(:geozone, name: "Lowlands and Highlands")
      end

      it "returns true when the geozone exists" do
        expect(UserSegments.valid_segment?("lands_and_borderlands")).to be true
        expect(UserSegments.valid_segment?("lowlands_and_highlands")).to be true
      end

      it "returns true when the segment exists" do
        expect(UserSegments.valid_segment?("all_users")).to be true
      end

      it "is falsey when the segment doesn't exist" do
        expect(UserSegments.valid_segment?("imaginary_segment")).to be_falsey
      end

      it "is falsey when nil is passed" do
        expect(UserSegments.valid_segment?(nil)).to be_falsey
      end
    end
  end

  describe ".all_users" do
    it "returns all active users enabled" do
      active_user = create(:user)
      erased_user = create(:user, erased_at: Time.current)

      expect(UserSegments.all_users).to eq [active_user]
      expect(UserSegments.all_users).not_to include erased_user
    end
  end

  describe ".administrators" do
    it "returns all active administrators users" do
      active_user = create(:user)
      active_admin = create(:administrator).user
      erased_user = create(:user, erased_at: Time.current)

      expect(UserSegments.administrators).to eq [active_admin]
      expect(UserSegments.administrators).not_to include active_user
      expect(UserSegments.administrators).not_to include erased_user
    end
  end

  describe ".all_proposal_authors" do
    it "returns users that have created a proposal even if is archived or retired" do
      create(:proposal, author: user1)
      create(:proposal, :archived, author: user2)
      create(:proposal, :retired, author: user3)

      all_proposal_authors = UserSegments.all_proposal_authors

      expect(all_proposal_authors).to match_array [user1, user2, user3]
    end

    it "does not return duplicated users" do
      create(:proposal, author: user1)
      create(:proposal, :archived, author: user1)
      create(:proposal, :retired, author: user1)

      all_proposal_authors = UserSegments.all_proposal_authors

      expect(all_proposal_authors).to eq [user1]
    end
  end

  describe ".proposal_authors" do
    it "returns users that have created a proposal" do
      create(:proposal, author: user1)

      proposal_authors = UserSegments.proposal_authors

      expect(proposal_authors).to eq [user1]
    end

    it "does not return duplicated users" do
      create(:proposal, author: user1)
      create(:proposal, author: user1)

      proposal_authors = UserSegments.proposal_authors
      expect(proposal_authors).to contain_exactly(user1)
    end
  end

  describe ".investment_authors" do
    it "returns users that have created a budget investment" do
      investment = create(:budget_investment, author: user1)
      budget = create(:budget)
      investment.update!(budget: budget)

      investment_authors = UserSegments.investment_authors

      expect(investment_authors).to eq [user1]
    end

    it "does not return duplicated users" do
      investment1 = create(:budget_investment, author: user1)
      investment2 = create(:budget_investment, author: user1)
      budget = create(:budget)
      investment1.update!(budget: budget)
      investment2.update!(budget: budget)

      investment_authors = UserSegments.investment_authors
      expect(investment_authors).to contain_exactly(user1)
    end
  end

  describe ".feasible_and_undecided_investment_authors" do
    it "returns authors of a feasible or an undecided budget investment" do
      user4 = create(:user)
      user5 = create(:user)
      user6 = create(:user)

      feasible_investment_finished = create(:budget_investment, :feasible, :finished, author: user1)
      undecided_investment_finished = create(:budget_investment, :undecided, :finished, author: user2)
      feasible_investment_unfinished = create(:budget_investment, :feasible, author: user3)
      undecided_investment_unfinished = create(:budget_investment, :undecided, author: user4)
      unfeasible_investment_unfinished = create(:budget_investment, :unfeasible, author: user5)
      unfeasible_investment_finished = create(:budget_investment, :unfeasible, :finished, author: user6)

      budget = create(:budget)
      feasible_investment_finished.update!(budget: budget)
      undecided_investment_finished.update!(budget: budget)
      feasible_investment_unfinished.update!(budget: budget)
      undecided_investment_unfinished.update!(budget: budget)
      unfeasible_investment_unfinished.update!(budget: budget)
      unfeasible_investment_finished.update!(budget: budget)

      investment_authors = UserSegments.feasible_and_undecided_investment_authors
      expect(investment_authors).to match_array [user1, user2, user3, user4, user5]
      expect(investment_authors).not_to include user6
    end

    it "does not return duplicated users" do
      feasible_investment = create(:budget_investment, :feasible, author: user1)
      undecided_investment = create(:budget_investment, :undecided, author: user1)
      budget = create(:budget)
      feasible_investment.update!(budget: budget)
      undecided_investment.update!(budget: budget)

      investment_authors = UserSegments.feasible_and_undecided_investment_authors
      expect(investment_authors).to contain_exactly(user1)
    end
  end

  describe ".selected_investment_authors" do
    it "returns authors of selected budget investments" do
      selected_investment = create(:budget_investment, :selected, author: user1)
      unselected_investment = create(:budget_investment, :unselected, author: user2)
      budget = create(:budget)
      selected_investment.update!(budget: budget)
      unselected_investment.update!(budget: budget)

      investment_authors = UserSegments.selected_investment_authors

      expect(investment_authors).to eq [user1]
    end

    it "does not return duplicated users" do
      selected_investment1 = create(:budget_investment, :selected, author: user1)
      selected_investment2 = create(:budget_investment, :selected, author: user1)
      budget = create(:budget)
      selected_investment1.update!(budget: budget)
      selected_investment2.update!(budget: budget)

      investment_authors = UserSegments.selected_investment_authors
      expect(investment_authors).to contain_exactly(user1)
    end
  end

  describe ".winner_investment_authors" do
    it "returns authors of winner budget investments" do
      winner_investment = create(:budget_investment, :winner, author: user1)
      selected_investment = create(:budget_investment, :selected, author: user2)
      budget = create(:budget)
      winner_investment.update!(budget: budget)
      selected_investment.update!(budget: budget)

      investment_authors = UserSegments.winner_investment_authors

      expect(investment_authors).to eq [user1]
    end

    it "does not return duplicated users" do
      winner_investment1 = create(:budget_investment, :winner, author: user1)
      winner_investment2 = create(:budget_investment, :winner, author: user1)
      budget = create(:budget)
      winner_investment1.update!(budget: budget)
      winner_investment2.update!(budget: budget)

      investment_authors = UserSegments.winner_investment_authors
      expect(investment_authors).to contain_exactly(user1)
    end
  end

  describe ".current_budget_investments" do
    it "only returns investments from the current budget" do
      investment1 = create(:budget_investment, author: create(:user))
      investment2 = create(:budget_investment, author: create(:user))
      budget = create(:budget)
      investment1.update!(budget: budget)

      current_budget_investments = UserSegments.current_budget_investments

      expect(current_budget_investments).to eq [investment1]
      expect(current_budget_investments).not_to include investment2
    end
  end

  describe ".not_supported_on_current_budget" do
    it "only returns users that haven't supported investments on current budget" do
      investment1 = create(:budget_investment)
      investment2 = create(:budget_investment)
      budget = create(:budget)
      investment1.vote_by(voter: user1, vote: "yes")
      investment2.vote_by(voter: user2, vote: "yes")
      investment1.update!(budget: budget)
      investment2.update!(budget: budget)

      not_supported_on_current_budget = UserSegments.not_supported_on_current_budget
      expect(not_supported_on_current_budget).to include user3
      expect(not_supported_on_current_budget).not_to include user1
      expect(not_supported_on_current_budget).not_to include user2
    end
  end

  describe ".user_segment_emails" do
    it "returns list of emails sorted by user creation date" do
      create(:user, email: "first@email.com", created_at: 1.day.ago)
      create(:user, email: "last@email.com")

      emails = UserSegments.user_segment_emails(:all_users)
      expect(emails).to eq ["first@email.com", "last@email.com"]
    end
  end

  context "Geozones" do
    let!(:new_york) { create(:geozone, name: "New York") }
    let!(:california) { create(:geozone, name: "California") }
    let!(:user1) { create(:user, geozone: new_york) }
    let!(:user2) { create(:user, geozone: new_york) }
    let!(:user3) { create(:user, geozone: california) }

    before do
      create(:geozone, name: "Mars")
      create(:user, geozone: nil)
    end

    it "includes geozones in available segments" do
      expect(UserSegments.segments).to include("new_york")
      expect(UserSegments.segments).to include("california")
      expect(UserSegments.segments).to include("mars")
      expect(UserSegments.segments).not_to include("jupiter")
    end

    it "returns users of a geozone" do
      expect(UserSegments.recipients("new_york")).to match_array [user1, user2]
      expect(UserSegments.recipients("california")).to eq [user3]
    end

    it "accepts symbols as parameters" do
      expect(UserSegments.recipients(:new_york)).to match_array [user1, user2]
      expect(UserSegments.recipients(:california)).to eq [user3]
    end

    it "only returns active users of a geozone" do
      user2.update!(erased_at: Time.current)

      expect(UserSegments.recipients("new_york")).to eq [user1]
    end
  end
end