it-swarm.com.ru

Как мне "ожидать" что-то, что вызывает исключение в RSpec?

У меня есть метод, который выполняет некоторые действия на модели Cat и в случае неправильного ввода вызывает исключение:

context "hungry cat" do
  it { expect { eat(what: nil) }.to raise_error } 
end

Что я хочу сделать, это проверить, меняет ли этот метод статус кошек, например:

context "hungry cat" do
  it { expect { eat(what: nil) }.to raise_error } 
  it { expect { eat(what: nil) }.not_to change(cat, :status) } 
end

Проблема в том, что, так как eat(what: nil) вызовет исключение, второй it потерпит неудачу, несмотря ни на что. Итак, возможно ли игнорировать исключение и проверить некоторые условия?

Я знаю, что можно сделать что-то вроде:

it do 
  expect do
    begin
      eat(what: nil)
    rescue
    end
  end.not_to change(cat, :status)
end

Но это слишком уродливо.

26
Andrew

Вы можете использовать идиому «Rescue Nil», чтобы сократить то, что у вас уже есть:

it { expect { eat(what: nil) rescue nil }.not_to change(cat, :status) }
19
awendt

Вы можете цепочка положительные утверждения с and. Если вы хотите объединить в цепочку отрицательный элемент, RSpec 3.1 представил define_negated_matcher .

Вы могли бы сделать что-то вроде:

RSpec::Matchers.define_negated_matcher :avoid_changing, :change

expect { eat(what: nil) }
  .to raise_error
  .and avoid_changing(cat, :status)

Вдохновленный этот комментарий .

20
Jan Klimo

В RSpec 3 вы можете объединить два теста в один. Я считаю, что это более читабельно, чем подход rescue nil

it { expect { eat(what: nil) }.to raise_error.and not_to change(cat, :status)}

12
Sean

Звучит странно, что код eat(what: nil) не выполняется отдельно для каждого из ваших тестов и влияет на другие. Я не уверен, но, возможно, переписывание тестов немного или решит проблему, или более точно определит, где проблема (выберите ваш вкус ниже).

Использование should:

context "hungry cat" do
  context "when not given anything to eat" do
    subject { -> { eat(what: nil) } }
    it { should raise_error } 
    it { should_not change(cat, :status) }
  end
end

Использование expect:

context "hungry cat" do
  context "when not given anything to eat" do
    let(:eating_nothing) { -> { eat(what: nil) } }
    it "raises an error" do
      expect(eating_nothing).to raise_error
    end
    it "doesn't change cat's status" do
      expect(eating_nothing).to_not change(cat, :status)
    end 
  end
end
0
Paul Fioravanti