app/controllers/proposals_controller.rb

Summary

Maintainability
A
3 hrs
Test Coverage
class ProposalsController < ApplicationController
  include TokenAuth

  skip_before_action :authenticate_user!, only: [:approve, :complete]
  skip_before_action :check_disabled_client, only: [:approve, :complete]
  # TODO use Policy for all actions
  before_action -> { authorize proposal }, only: [:show, :cancel, :cancel_form, :history]
  before_action :needs_token_on_get, only: [:approve, :complete]
  before_action :validate_access, only: [:approve, :complete]
  add_template_helper ProposalsHelper
  rescue_from Pundit::NotAuthorizedError, with: :auth_errors

  def show
    @proposal = proposal.decorate

    if detail_beta?
      show_next
    end
  end

  def show_next
    @client_data_instance ||= proposal.client_data

    @client_data_display = !@client_data_instance.nil? ? PrepareDisplayFields.new(@client_data_instance).run : @client_data_instance # Needed for tests to not fail
    @subscriber_list = SubscriberList.new(@proposal).triples
    @events = HistoryList.new(proposal).filtered_approvals
    render "show_next"
  end

  def index
    @closed_proposal_limit = ENV.fetch("CLOSED_PROPOSAL_LIMIT", 10).to_i
    @pending_data = listing.pending
    @pending_review_data = listing.pending_review
    @completed_data = listing.completed.apply_limit(@closed_proposal_limit)
    @canceled_data = listing.canceled.apply_limit(@closed_proposal_limit)
    beta_index_setup
  end

  def index_redesign
    @list_view_config = current_user.list_view_config || "nil"
    render "index_next"
  end

  def archive
    redirect_to query_proposals_path(text: "status:completed")
  end

  def beta_index_setup
    if list_beta?
      @unfiltered_data = listing.all
      index_redesign
    end
  end

  def cancel_form
    @proposal = proposal.decorate
  end

  def cancel
    if params[:reason_input].present?
      cancel_proposal
    else
      redirect_to(
        cancel_form_proposal_path(params[:id]),
        alert: "A reason for cancelation is required. Please indicate why this request needs to be canceled."
      )
    end
  end

  def approve
    complete
  end

  def complete
    step = proposal.existing_or_delegated_actionable_step_for(current_user)
    if step
      complete_step(step)
    else
      flash[:alert] = I18n.t("errors.policies.proposal.step_complete")
    end
    redirect_to proposal
  end

  def query
    check_search_params
    query_listing = listing
    unless @proposals_data = try_search(query_listing)
      redirect_to proposals_path
    end
    @start_date = query_listing.start_date
    @end_date = query_listing.end_date
  end

  def query_count
    set_search_params
    if !valid_search_params?
      render json: { total: 0 }
    elsif @proposals_data = try_search(listing)
      render json: { total: @proposals_data.es_response.results.total }
    else
      render json: { error: flash[:error] }
    end
  end

  def download
    params[:size] = :all
    params.delete(:page)
    begin
      build_csv_download
    rescue SearchBadQuery, SearchUnavailable => error
      flash[:error] = error.message
      redirect_to proposals_path
    end
  end

  def history
    @container = ProposalVersionsQuery.new(proposal).container
    @container.state_from_params = params
  end

  def update_approvals_card
    @proposal = proposal.decorate
    render partial: "proposals/details/status", locals: { proposal: @proposal }
  end

  protected

  def try_search(query_listing)
    begin
      @proposals_data = query_listing.query
    rescue SearchBadQuery, SearchUnavailable => error
      flash[:error] = error.message
      false
    end
  end

  def build_csv_download
    query_listing = listing
    @proposals_data = query_listing.query
    timestamp = Time.current.utc.strftime("%Y-%m-%d-%H-%M-%S")
    headers["Content-Disposition"] = %(attachment; filename="C2-Proposals-#{timestamp}.csv")
    headers["Content-Type"] = "text/csv"
  end

  def cancel_proposal_and_send_cancelation_emails
    comments = "Request canceled with comments: " + params[:reason_input]
    proposal.cancel!
    proposal.comments.create!(comment_text: comments, user: current_user)
    DispatchFinder.run(proposal).deliver_cancelation_emails(current_user, params[:reason_input])
  end

  def proposal
    @cached_proposal ||= Proposal.find(params[:id])
  end

  def auth_errors(exception)
    if %w(cancel cancel_form).include?(params[:action])
      redirect_to proposal_path, alert: exception.message
    else
      super
    end
  end

  def listing
    ProposalListingQuery.new(current_user, params)
  end

  def set_search_params
    @dsl = build_search_dsl
    @text = params[:text]
    @adv_search = @dsl.client_query
    build_search_query
    find_search_report
  end

  def check_search_params
    set_search_params
    unless valid_search_params?
      flash[:alert] = "Please enter one or more search criteria"
      redirect_to proposals_path
    end
  end

  def valid_search_params?
    @text.present? ||
      @adv_search.present? ||
      (params[:start_date].present? && params[:end_date].present?) ||
      params[:status].present?
  end

  def find_search_report
    if params[:report]
      @report = Report.find params[:report]
    end
  end

  def build_search_dsl
    ProposalSearchDsl.new(
      params: params,
      current_user: current_user,
      query: params[:text],
      client_data_type: current_user.client_model.to_s
    )
  end

  def build_search_query
    @search_query = { "humanized" => @dsl.humanized_query_string }
    if @text.present?
      @search_query["text"] = @text
    end
    if @adv_search.present?
      @search_query[current_user.client_model_slug] = @adv_search.to_h
    end
  end

  def complete_step(step)
    step.update_attributes!(completer: current_user)
    step.complete!
    flash[:success] = "You have approved #{proposal.public_id}."
  end

  def detail_beta?
    current_user.should_see_beta?("BETA_FEATURE_DETAIL_VIEW")
  end

  def list_beta?
    current_user.should_see_beta?("BETA_FEATURE_LIST_VIEW")
  end

  def cancel_proposal
    cancel_proposal_and_send_cancelation_emails
    flash[:success] = "You've canceled the request."
    redirect_path = detail_beta? ? proposals_path : proposal_path(proposal)
    redirect_to redirect_path
  end
end