2

私はコントローラーを持っています:

class SessionTimeoutController < ApplicationController
  # snip

  skip_before_filter :authenticate_user!, only: [:has_user_timed_out]

  #snip

  # Determines whether the user has been logged out due to
  # inactivity or not.
  def has_user_timed_out
    @has_timed_out = (!user_signed_in?) or (current_user.timedout? (user_session["last_request_at"]))

    respond_to do |format|
      format.json { render json: @has_timed_out, status: :created}
    end
  end

  #snip
end

session_timeout/has_user_timed_outAJAXを使用してリクエストを送信します。

$.ajax({
  url: '/session_timeout/has_user_timed_out',
  type: 'GET',
  dataType: 'json',
  success: function( logged_out ) {
    if(logged_out === true)
    {
      window.location = '/users/sign_in?timedout=1'
    }
    else
    {
      checkTimeLeft();
    }
  }
});

この AJAX 呼び出しが行われると、ユーザーがログアウトした場合でも、常に応答を受け取ることが期待されます。skip_before_filter :authenticate_user!コントローラーで設定したので、これを期待しています。ただし、AJAX を送信すると、サーバーは401 Unauthorized応答を返します。どうしてこれなの?

さらに、この JS をブラウザ コンソールから実行すると、期待どおりに動作します。AJAX で問題が発生するのは、ビューから呼び出されたときだけです。

$.ajax({url: '/session_timeout/has_user_timed_out', type: 'GET', dataType:'JSON', success: function(logged_out) {console.log(logged_out + ' ' + typeof logged_out);}});

問題を引き起こしている可能性のあることの 1 つは、非アクティブなためにユーザーが自動的にログアウトされてから数秒以内に AJAX 要求を送信することです。これはDeviseをまったく混乱させますか?


編集: これは、401 応答になるリクエストを送信したときの Rails ログです。

Started GET "/session_timeout/has_user_timed_out" for 10.0.2.2 at 2013-07-25 21:25:51 +0000
Processing by SessionTimeoutController#has_user_timed_out as JSON
  ConfigVar Load (0.2ms)  SELECT `config_vars`.* FROM `config_vars` WHERE `config_vars`.`name` = 'maintenance_mode' ORDER BY `config_vars`.`id` ASC LIMIT 1
  User Load (0.3ms)  SELECT `users`.* FROM `users` WHERE `users`.`id` = 14 ORDER BY `users`.`id` ASC LIMIT 1
Completed 401 Unauthorized in 4ms

これは、ユーザーがログアウトしていても機能するリクエストを送信したときの Rails ログです。

Started GET "/session_timeout/has_user_timed_out" for 10.0.2.2 at 2013-07-25 21:29:39 +0000
Processing by SessionTimeoutController#has_user_timed_out as JSON
  ConfigVar Load (0.1ms)  SELECT `config_vars`.* FROM `config_vars` WHERE `config_vars`.`name` = 'maintenance_mode' ORDER BY `config_vars`.`id` ASC LIMIT 1
Completed 201 Created in 3ms (Views: 0.1ms | ActiveRecord: 0.1ms)

もう少しいじってみるとsession_timeout/has_user_timed_out、ユーザーがログアウトした後に初めてリクエストを送信すると、どれだけ待っても失敗するようです。その後、後続のすべての要求が成功します。

4

1 に答える 1

1

回避策を見つけました。応答コードに基づいてサーバー応答をインターセプトできるオプションを使用できます$.ajaxstatus私の場合、気になるのはユーザーがログインしているかどうかです。401エラーはユーザーがログアウトしていることを教えてくれるので、応答を傍受してそれに応じて行動できます。

$.ajax({
  url: '/session_timeout/has_user_timed_out',
  type: 'GET',
  dataType: 'json',
  success: function( logged_out ) {
    if(logged_out === true)
    {
      window.location = '/users/timedout'
    }
    else
    {
      checkTimeLeft();
    }
  },
  // THIS IS THE CODE I ADDED FOR MY WORKAROUND:
  statusCode: {
    // If we get back an authentication error, it's a
    // pretty safe bet we're not logged in anymore
    401: function() {
      window.location = '/users/timedout'
    }
  }
});
于 2013-07-25T22:27:46.747 に答える