web-dev-qa-db-ja.com

ノードでmocha / supertestを使用してリダイレクトするリクエストをテストする

mochasupertest 、および should (およびcoffeescript)を使用して、次の統合テストをExpressプロジェクトに渡すことができないようです。


テスト

_should  = require('should')
request = require('supertest')
app     = require('../../app')

describe 'authentication', ->
  describe 'POST /sessions', ->
    describe 'success', (done) ->
      it 'displays a flash', (done) ->
        request(app)
          .post('/sessions')
          .type('form')
          .field('user', 'username')
          .field('password', 'password')
          .end (err, res) ->
            res.text.should.include('logged in')
            done()
_

関連アプリケーションコード

_app.post '/sessions', (req, res) ->
  req.flash 'info', "You are now logged in as #{req.body.user}"
  res.redirect '/login'
_

失敗

_1) authentication POST /sessions success displays a flash:
   AssertionError: expected 'Moved Temporarily. Redirecting to //127.0.0.1:3456/login' to include 'logged in'
_

明らかに、アプリケーションコードは何も役に立たない。私はテストに合格しようとしています。

期待値(res.text.should.include('logged in'))をend関数の外側とexpect関数の内側に置いても、同じ結果が得られます。たとえば、.type('form')呼び出しを削除し、2つの.send(user: 'username', password: 'password')呼び出しの代わりに.field()を使用するなど、さまざまな関数呼び出しを試しました。

それが何かを意味する場合、ローカルで実行しているときにアプリにcurl POSTリクエストを送信すると、同じ出力が生成されます(_Moved Temporarily. Redirecting to //127.0.0.1:3456/login_)

これは些細な間違いだと思います。おそらく、アプリケーションコードまたはテストコードで忘れているものです。

助言がありますか?

EDIT 1:また、ブラウザの送信ボタンをクリックすると、期待どおりの結果(フラッシュメッセージ)が表示されることにも注意してください。

EDIT 2:さらに調査すると、anyリダイレクトの出力が_Moved Temporarily. Redirecting to ..._応答本文に出力されています。これにより、app.jsでアプリをエクスポートする方法に問題があるのではないかと思います。

_var express = require('express')
var app = express();

module.exports = app;
_
36
Feech

このページに出くわした人にとって、この質問への答えは非常に簡単です。 Moved Temporarily.レスポンスボディは、スーパーテストから返されるものです。詳細は 問題 を参照してください。

要約すると、私はこのようなことをすることになりました。

should  = require('should')
request = require('supertest')
app     = require('../../app')

describe 'authentication', ->
  describe 'POST /sessions', ->
    describe 'success', ->
      it 'redirects to the right path', (done) ->
        request(app)
          .post('/sessions')
          .send(user: 'username', password: 'password')
          .end (err, res) ->
            res.header['location'].should.include('/home')
            done()

応答ヘッダーlocationが想定どおりであることを確認してください。フラッシュメッセージのテストと特定の統合テストの表示は、別の方法を使用して行う必要があります。

25
Feech

supertestには、このための組み込みのアサーションがあります。

should  = require('should')
request = require('supertest')
app     = require('../../app')

describe 'authentication', ->
  describe 'POST /sessions', ->
    describe 'success', ->
      it 'redirects to the right path', (done) ->
        request(app)
          .post('/sessions')
          .send(user: 'username', password: 'password')
          .expect(302)
          .expect('Location', '/home')
          .end(done)
25
Blacksonic
_describe 'authentication', ->
  describe 'POST /sessions', ->
    describe 'success', (done) ->
      it 'displays a flash', (done) ->
        request(app)
          .post('/sessions')
          .type('form')
          .field('user', 'username')
          .field('password', 'password')
          .redirects(1)
          .end (err, res) ->
            res.text.should.include('logged in')
            done()
_

redirects()はリダイレクトに従って、通常のビューテストを実行できます。

3
super_noobling

私もリダイレクトするリクエストのいくつかの統合テストを書こうとしていて、モジュールの作者自身が良い例を見つけた here

TJの例では、彼はチェーンを使用しているので、私もそのようなものを使用しました。

ログインしているユーザーがログアウトしたときに、ホームページにリダイレクトするシナリオを考えてみます。

it('should log the user out', function (done) {
  request(app)
    .get('/logout')
    .end(function (err, res) {
      if (err) return done(err);
      // Logging out should have redirected you...
      request(app)
        .get('/')
        .end(function (err, res) {
          if (err) return done(err);
          res.text.should.not.include('Testing Tester');
          res.text.should.include('login');
          done();
        });
    });
});

リダイレクトの数によっては、いくつかのコールバックをネストする必要があるかもしれませんが、TJの例で十分です。

2
JohnnyCoder