# frozen_string_literal: true

require 'spec_helper'

module RailsBestPractices
  module Reviews
    describe UseBeforeFilterReview do
      let(:runner) { Core::Runner.new(reviews: described_class.new(customize_count: 2)) }

      it 'uses before_filter' do
        content = <<-EOF
        class PostsController < ApplicationController
          def show
            @post = current_user.posts.find(params[:id])
          end

          def edit
            @post = current_user.posts.find(params[:id])
          end

          def update
            @post = current_user.posts.find(params[:id])
            @post.update_attributes(params[:post])
          end

          def destroy
            @post = current_user.posts.find(params[:id])
            @post.destroy
          end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(1)
        expect(runner.errors[0].to_s).to eq(
          'app/controllers/posts_controller.rb:2,6,10,15 - use before_filter for show,edit,update,destroy'
        )
      end

      it 'does not use before_filter when equal to customize count' do
        content = <<-EOF
        class PostsController < ApplicationController
          def show
            @post = Post.find(params[:id])
          end

          def edit
            @post = Post.find(params[:id])
          end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(0)
      end

      it 'does not use before_filter' do
        content = <<-EOF
        class PostsController < ApplicationController
          before_filter :find_post, only: [:show, :edit, :update, :destroy]

          def update
            @post.update_attributes(params[:post])
          end

          def destroy
            @post.destroy
          end

          protected
          def find_post
            @post = current_user.posts.find(params[:id])
          end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(0)
      end

      it 'does not use before_filter by nil' do
        content = <<-EOF
        class PostsController < ApplicationController
          def show; end
          def edit; end
          def update; end
          def destroy; end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(0)
      end

      it 'does not use before_filter for protected/private methods' do
        content = <<-EOF
        class PostsController < ApplicationController
          protected
          def load_comments
            load_post
            @comments = @post.comments
          end

          def load_user
            load_post
            @user = @post.user
          end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(0)
      end

      it 'does not check ignored files' do
        runner = Core::Runner.new(reviews: described_class.new(customize_count: 2, ignored_files: /posts_controller/))
        content = <<-EOF
        class PostsController < ApplicationController
          def show
            @post = current_user.posts.find(params[:id])
          end

          def edit
            @post = current_user.posts.find(params[:id])
          end

          def update
            @post = current_user.posts.find(params[:id])
            @post.update_attributes(params[:post])
          end

          def destroy
            @post = current_user.posts.find(params[:id])
            @post.destroy
          end
        end
        EOF
        runner.review('app/controllers/posts_controller.rb', content)
        expect(runner.errors.size).to eq(1)
      end
    end
  end
end