Domanda Come verificare lo stato del rendering: 404 con Rails4 e RSpec quando si utilizza rescue_from


Ho un'applicazione Rails4 con un 'PagesController'.

Il metodo show genera un'eccezione personalizzata 'PageNotFoundError' quando una pagina non viene trovata.

In cima al controller che ho definito rescue_from PageNotFoundError, with: :render_not_found

render not found è un metodo privato di PagesController e assomiglia a:

def render_not_found
  flash[:alert]=t(:page_does_not_exists, title: params[:id])
  @pages = Page.all
  render :index, status: :not_found #404
end

Il rails-log in modalità di sviluppo mostra:

Started GET "/pages/readmef" for 127.0.0.1 at 2013-08-02 23:11:35 +0200
Processing by PagesController#show as HTML
  Parameters: {"id"=>"readmef"}
  ..
  Completed 404 Not Found in 14ms (Views: 12.0ms)

Quindi, rispecchia il mio: status =>: not_found funziona, finora.

Quando io faccio curl -v http://0.0.0.0:3000/pages/readmef log di arricciatura

curl -v http://localhost:3000/pages/readmef
* About to connect() to localhost port 3000 (#0)
*   Trying 127.0.0.1...
* connected
* Connected to localhost (127.0.0.1) port 3000 (#0)
> GET /pages/readmef HTTP/1.1
> User-Agent: curl/7.24.0 (x86_64-apple-darwin12.0) libcurl/7.24.0 OpenSSL/0.9.8x zlib/1.2.5
> Host: localhost:3000
> Accept: */*
>
< HTTP/1.1 404 Not Found
< X-Frame-Options: SAMEORIGIN

Ma il seguente test con RSpec fallisce:

 it 'renders an error if page not found' do
    visit page_path('not_existing_page_321')
    expect(response.status).to eq(404)
    within( '.alert-error' ) do
      page.should have_content('Page not_existing_page_321 doesn\'t exist')
    end
  end

  1) PagesController renders an error if page not found
     Failure/Error: expect(response.status).to eq(404)

       expected: 404
            got: 200

Tutto sembra a posto e anche il test.log dice 404

$ tail -f log/test.log
Started GET "/pages/not_existing_page_321" for 127.0.0.1 at 2013-08-03 09:48:13 +0200
Processing by PagesController#show as HTML
  Parameters: {"id"=>"not_existing_page_321"}
  Rendered pages/_page.haml (0.8ms)
  Rendered layouts/_navigation.haml (0.6ms)
  Rendered layouts/_messages.haml (0.2ms)
  Rendered layouts/_locales.haml (0.3ms)
  Rendered layouts/_footer.haml (0.6ms)
Completed 404 Not Found in 6ms (Views: 4.5ms)

Ho provato diversi server, WebRICK, Thin, unicorno. Tutto funziona come previsto in modalità di sviluppo e produzione. Anche il test.log è corretto ma il test fallisce.

Qualcuno può dirmi perché il test dice 200 invece di 404?


17
2017-08-02 21:46


origine


risposte:


Anche se non sono molto contento di questa soluzione, almeno questa è una soluzione:

Ho diviso il test in due specifiche separate. Uno per testare il codice di risposta 404 (con GET invece di visitare) e un secondo per testare l'avviso. Il secondo test è necessario perché get non rende la vista, anche se render_views è definito in cima al file spec.

  it 'response with 404 if page not found' do
    get :show, { controller: 'pages', id: 'not_existing_page_321' }
    expect(response.status).to eq(404)
  end

  it 'renders an error-message if page not found and shows index' do
    visit page_path('page_not_found')
    within '.alert-error' do
      page.should have_content("Page page_not_found doesn't exist")
    end
  end

13
2017-08-04 20:01



Un altro approccio con RSpec 3+ potrebbe essere quello di verificare un'eccezione:

it 'respond with 404 if page not found' do
  expect{ get :show, :id => 'bad_id' }.to raise_error(ActionController::RoutingError)
end

13
2017-12-11 10:52



Il problema qui è che si confondono i test delle funzionalità di Capybara e i test del controller RSpec. visit è un metodo fornito da Capybara e get / response sono forniti dai test del controller RSpec - non puoi usarli insieme.

Per testare questo come un singolo test del controller RSpec puoi fare:

it "returns a not found response" do
  get :show, { id: 'not_existing_page_321' }
  expect(response.status).to eq(404)
  expect(response.text).to match(/Page page_not_found doesn't exist/)
end

(N.b. La linea get è diversa da quella che hai pubblicato - Non ho incluso il controller param come se lo mettessi dentro spec/controllers/pages_controller_spec.rb dov'è che non hai bisogno di questo)

O come singolo test di funzionalità di Capybara:

it "renders a not found response" do
  visit page_path('page_not_found')
  expect(page.status_code).to eq(404)
  within '.alert-error' do
    expect(page).to have_content("Page page_not_found doesn't exist")
  end
end

8
2017-08-24 06:41