app/controllers/concerns/token_auth.rb

Summary

Maintainability
A
0 mins
Test Coverage
module TokenAuth
  extend ActiveSupport::Concern

  included do
    rescue_from Pundit::NotAuthorizedError, with: :auth_errors
    rescue_from ActionController::InvalidAuthenticityToken, with: :authenticity_token_exception
  end

  # As a security precaution, certain actions must be POSTed to.
  def needs_token_on_get
    if request.get?
      authorize(:api_token, :valid!, params)
    end
  end

  def validate_access
    validate_signed_in

    current_user_tokens.where(used_at: nil).update_all(used_at: Time.zone.now)
    authorize(proposal, :can_complete!)

    if params[:version] && params[:version] != proposal.version.to_s
      raise Pundit::NotAuthorizedError, I18n.t("errors.policies.api_token.version_mismatch")
    end
  end

  def validate_signed_in
    if not_signed_in?
      authorize(:api_token, :valid_and_not_delegate!, params)
      token = ApiToken.find_by(access_token: params[:cch])
      sign_in(token.user)
    end
  end

  def auth_errors(exception)
    case exception.record
    when :api_token
      render_api_token_exception(exception)
    when Proposal
      if exception.message == I18n.t("errors.policies.proposal.step_complete")
        flash[:error] = exception.message
        redirect_to proposal
      else
        render "authorization_error", status: 403, locals: { msg: I18n.t("errors.policies.proposal.show_permission") }
      end
    else
      render_other_exception(exception)
    end
  end

  def authenticity_token_exception
    flash[:error] = "Your session has expired. Please sign in."
    redirect_to root_path
  end

  def render_api_token_exception(exception)
    if signed_in?
      flash[:error] = exception.message
      render "authentication_error", status: 403
    else
      return_to_param = make_return_to("Previous", request.fullpath)
      session[:return_to] = return_to_param
      redirect_to root_path(return_to: return_to_param), alert: I18n.t("errors.policies.api_token.not_delegate")
    end
  end

  def render_other_exception(exception)
    if exception.message == "Client is disabled"
      render_disabled_client_message(exception.message)
    else
      flash[:error] = exception.message
      redirect_to proposal
    end
  end

  private

  def current_user_tokens
    ApiToken.joins(:step).where(
      steps: { user_id: current_user, proposal_id: proposal }
    )
  end
end