require 'rails_helper'


RSpec.describe Licitacao::ContratosController, type: :controller do
	sign_in_admin
	cria_configuracao_default
	set_licitacao_na_sessao


	let(:atributos_validos) {
		FactoryBot.attributes_for( :licitacao_contrato, :com_lotes_do_contrato )
	}

	let(:atributos_invalidos) {
		atributos_validos.merge!(tipo_de_contrato: "")
	}

	let(:projeto) {
		FactoryBot.attributes_for(:licitacao_projeto, forma_de_agrupamento: "por_lote")
	}

	let(:projeto_de_dispensa) {
		FactoryBot.attributes_for(:licitacao_projeto, modalidade_do_processo: "dispensa_de_licitacao", tipo_de_totalizacao: 'menor_valor', modalidade_de_licitacao: 'dispensa_ou_inexigibilidade' )
	}

	let(:projeto_registro_de_preco) {
		FactoryBot.attributes_for(:licitacao_projeto, forma_de_agrupamento: "por_lote", registro_de_preco: 'false')
	}

	let(:contrato_aberto) {
		Licitacao::Contrato.find_by(FactoryBot.attributes_for :licitacao_contrato, status: 'aberto') || FactoryBot.create(:licitacao_contrato, :com_lotes_do_contrato, status: 'aberto')
	}

	context 'quando o formato é JS' do
		it 'retorna um JS' do
			modelo = Licitacao::ModeloDeDocumento.find_or_create_by(FactoryBot.attributes_for :licitacao_modelo_de_documento)
			contrato_aberto
			get :retorna_modelo, params: {projeto_id: contrato_aberto.projeto.id, modelo_id: modelo.id, format: :js}
			expect(response.status).to eq 200
			expect(response.headers["Content-Type"]).to eq "text/javascript; charset=utf-8"
		end
	end

	describe "PATCH #cancelar" do
		context "com parâmetros válidos" do
			let(:novos_atributos) {
				{motivo_do_cancelamento: "motivo convincente"}
			}

			it "atualiza o(a) contrato requisitado(a)" do
				patch :cancelar, params: {id: contrato_aberto.to_param, licitacao_contrato: novos_atributos}
				contrato_aberto.reload
				expect(contrato_aberto.motivo_do_cancelamento).to eq "motivo convincente"
			end

			it "muda status do contrato_aberto para 'cancelado'" do
				patch :cancelar, params: {id: contrato_aberto.to_param, licitacao_contrato: novos_atributos}
				contrato_aberto.reload
				expect(contrato_aberto.cancelado?).to be_truthy
			end

			it "redireciona para o(a) contrato" do
				patch :cancelar, params: {id: contrato_aberto.to_param, licitacao_contrato: novos_atributos}
				expect(response).to redirect_to(contrato_aberto)
			end
		end
	end

	describe "PATCH #finalizar" do
		context "finaliza contrato" do

			it "muda status do contrato_aberto para 'fechado'" do
				patch :finalizar, params: {id: contrato_aberto.to_param}
				contrato_aberto.reload
				expect(contrato_aberto.fechado?).to be_truthy
			end

			it "redireciona para o(a) contrato" do
				patch :finalizar, params: {id: contrato_aberto.to_param}
				expect(response).to redirect_to(contrato_aberto)
			end
		end
	end

	describe "GET #index" do
		before do
			projeto = Licitacao::Projeto.create!(FactoryBot.attributes_for(:licitacao_projeto, forma_de_agrupamento: "por_lote", registro_de_preco: true))
			@contrato = FactoryBot.create( :licitacao_contrato, :com_lotes_do_contrato, projeto_id: projeto.id )
			@outro_contrato = FactoryBot.create( :licitacao_contrato, :com_lotes_do_contrato, :unidade_orcamentaria_diferente, projeto_id: projeto.id )
		end
		context 'quando o usuário acessa todas as unidades' do
			it "atribui a lista completa de contratos solicitados à @contratos" do
				sign_in FactoryBot.create(:usuario_comum)
				get :index
				expect(assigns(:contratos)).to include(@contrato, @outro_contrato)
			end
		end
		context 'quando o usuário possui unidades selecionadas' do
			sign_in_secretario
			it 'retorna apenas contratos solicitados das unidades do usuário' do
				get :index
				expect(assigns(:contratos)).to eq([@contrato])
			end
		end
	end

	describe "GET #show" do
		sign_in_secretario
		context 'e o usuário pode acessar unidade do contrato' do
			it 'atribui a contrato requisitado(a) à @contrato' do
				contrato = FactoryBot.create( :licitacao_contrato, :com_lotes_do_contrato )
				get :show, params: { id: contrato.to_param }
				expect(assigns(:contrato)).to eq(contrato)
			end
		end
	end

	describe "GET #new" do
		before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
		end

		it "atribui um(a) novo(a) contrato à @contrato" do
			get :new, params: {projeto_id: @projeto.to_param}
			expect(assigns(:contrato)).to be_a_new(Licitacao::Contrato)
		end
	end

	describe "GET #edit" do
		before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
		end

		it "atribui o(a) contrato requisitado à @contrato" do
			contrato = Licitacao::Contrato.create! atributos_validos
			get :edit, params: {id: contrato.to_param, projeto_id: @projeto.to_param}
			expect(assigns(:contrato)).to eq(contrato)
		end
	end

	describe "POST #create" do
		context "com parâmetros válidos" do
			before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
				@projeto.update_column(:registro_de_preco, true)
			end

			it "cria um(a) novo(a) Licitacao::Contrato" do
				expect {
					pessoa_do_projeto = FactoryBot.create(:licitacao_pessoa_do_projeto, projeto_id: @projeto.id)
					post :create, params: {licitacao_contrato: atributos_validos.merge(pessoa_do_projeto_id: pessoa_do_projeto.id), projeto_id: @projeto.to_param}
				}.to change(Licitacao::Contrato, :count).by(1)
			end

			it "atribui o(a) contrato recém criado(a) à @contrato" do
				pessoa_do_projeto = FactoryBot.create(:licitacao_pessoa_do_projeto, projeto_id: @projeto.id)
				post :create, params: {licitacao_contrato: atributos_validos.merge(pessoa_do_projeto_id: pessoa_do_projeto.id), projeto_id: @projeto.to_param}
				expect(assigns(:contrato)).to be_a(Licitacao::Contrato)
				expect(assigns(:contrato)).to be_persisted
			end

			it "redireciona para o processo da publicação criado(a)" do
				pessoa_do_projeto = FactoryBot.create(:licitacao_pessoa_do_projeto, projeto_id: @projeto.id)
				post :create, params: {:licitacao_contrato => atributos_validos.merge(pessoa_do_projeto_id: pessoa_do_projeto.id), projeto_id: @projeto.to_param}
				expect(response).to redirect_to(Licitacao::Contrato.last)
			end

		end

		context "Tentando criar dois contratos para projeto de dispensa para o mesmo licitante" do
			it "atribui um(a) contrato recém criado(a), mas não salva o segundo @contrato" do
				projeto = FactoryBot.create(:licitacao_projeto, :dispensa)
				pessoa_do_projeto = FactoryBot.create(:licitacao_pessoa_do_projeto, projeto_id: projeto.id)
				projeto.reload
				_contrato = FactoryBot.create(:licitacao_contrato, :com_lotes_do_contrato, pessoa_do_projeto_id: pessoa_do_projeto.id)
				post :create, params: {licitacao_contrato: atributos_validos.merge(pessoa_do_projeto_id: pessoa_do_projeto.id), projeto_id: projeto.id}
				expect(response).to redirect_to(Licitacao::Contrato.last)
			end
		end

		context "Tentando criar dois contratos para projeto que não é registro de preço" do
			it "atribui um(a) contrato recém criado(a), mas não salva o segundo @contrato" do
				projeto = Licitacao::Projeto.create! projeto_registro_de_preco
				pessoa_do_projeto = FactoryBot.create(:licitacao_pessoa_do_projeto, projeto_id: projeto.id)
				_contrato = FactoryBot.create(:licitacao_contrato, :com_lotes_do_contrato, pessoa_do_projeto_id: pessoa_do_projeto.id)
				post :create, params: {licitacao_contrato: atributos_validos.merge(pessoa_do_projeto_id: pessoa_do_projeto.id), projeto_id: projeto.id}
				expect(response).to redirect_to(Licitacao::Contrato.last)
			end
		end


		context "com parâmetros inválidos" do
			before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
				@projeto.update_column(:registro_de_preco, true)
			end

			it "atribui um(a) contrato recém criado(a), mas não salvo(a), à @contrato" do
				post :create, params: {licitacao_contrato: atributos_invalidos, projeto_id: @projeto.to_param}
				expect(assigns(:contrato)).to be_a_new(Licitacao::Contrato)
			end

			it "re-renderiza o template 'new'" do
				post :create, params: {licitacao_contrato: atributos_invalidos, projeto_id: @projeto.to_param}
				expect(response).to render_template("new")
			end
		end
	end

	describe "PUT #update" do
		context "com parâmetros válidos" do
			let(:novos_atributos) {
				{ tipo_de_contrato: "outros_contratos"}
			}

			before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
			end

			it "atualiza o(a) contrato requisitado(a)" do
				contrato = Licitacao::Contrato.create! atributos_validos
				put :update, params: {id: contrato.to_param, licitacao_contrato: novos_atributos, projeto_id: @projeto.id}
				contrato.reload
				expect(contrato.tipo_de_contrato).to eq("outros_contratos")
			end

			it "atribui o(a) contrato requisitado(a) à @contrato" do
				contrato = Licitacao::Contrato.create! atributos_validos
				put :update, params: {id: contrato.to_param, licitacao_contrato: atributos_validos, projeto_id: @projeto.id}
				expect(assigns(:contrato)).to eq(contrato)
			end

			it "redireciona para o contrato" do
				contrato = Licitacao::Contrato.create! atributos_validos
				put :update, params: {id: contrato.to_param, licitacao_contrato: atributos_validos, projeto_id: @projeto.id}
				expect(response).to redirect_to(contrato)
			end
		end

		context "com parâmetros inválidos" do

			before(:each) do
				@projeto = Licitacao::Projeto.create! projeto
			end

			it "atribui o(a) contrato à @contrato" do
				contrato = Licitacao::Contrato.create! atributos_validos
				put :update, params: {id: contrato.to_param, licitacao_contrato: atributos_invalidos, projeto_id: @projeto.id}
				expect(assigns(:contrato)).to eq(contrato)
			end

			it "re-renderiza o template 'edit'" do
				contrato = Licitacao::Contrato.create! atributos_validos.merge!(status: :aberto)
				put :update, params: {id: contrato.to_param, licitacao_contrato: atributos_invalidos, projeto_id: @projeto.id}
				expect(response).to render_template("edit")
			end
		end
	end

	describe "DELETE #destroy" do
		it "destrói o(a) contrato requisitado(a)" do
			contrato = Licitacao::Contrato.create! atributos_validos
			expect {
				delete :destroy, params: {:id => contrato.to_param}
			}.to change(Licitacao::Contrato, :count).by(-1)
		end

		it "redireciona para a lista de contratos" do
			contrato = Licitacao::Contrato.create! atributos_validos
			delete :destroy, params: {:id => contrato.to_param}
			expect(response).to redirect_to(licitacao_contratos_path)
		end
	end

	describe 'PATCH #atualiza_itens' do
		context "com parâmetros válidos" do

			it 'atualiza o contrato com os itens' do
				contrato = Licitacao::Contrato.create! atributos_validos
				contrato.itens_do_contrato.destroy_all
				atributos_itens_do_contrato = {
					itens_do_contrato_attributes: {
						'0': {
							contrato_id: contrato.id,
							item_do_lote_id: Licitacao::ItemDoLote.find_or_create_by!( FactoryBot.attributes_for(:licitacao_item_do_lote)).id,
							quantidade: 1
						}
					}
				}
				patch :atualiza_itens, params: {id: contrato.to_param, licitacao_contrato: atributos_itens_do_contrato}
				contrato.reload
				contrato.itens_do_contrato.reload
				expect(contrato.itens_do_contrato.count).to eq 1
			end
		end

		context "com parâmetros inválidos" do
			let(:pessoa_do_projeto) { FactoryBot.create(:licitacao_pessoa_do_projeto) }
			let(:contrato) { FactoryBot.create :licitacao_contrato, :com_lotes_do_contrato, numero: '3442' }
			let(:atributos_itens_do_contrato) {
				{
						itens_do_contrato_attributes: {
								'0': {
									contrato_id: contrato.id,
									quantidade: "5"
									}
								}
						}
				}

			it 're-renderiza o template :editar_itens' do
				patch :atualiza_itens, params: {id: contrato.to_param, licitacao_contrato: atributos_itens_do_contrato.merge(pessoa_do_projeto_id: pessoa_do_projeto.id)}
				expect(response).to render_template("editar_itens")
			end
		end
	end

  describe "PATCH #nao_enviar_para_sim" do
		it "nao_enviar_para_sim" do
			contrato = FactoryBot.create :licitacao_contrato, :com_lotes_do_contrato, envia_pro_sim: true
      contrato.update_column(:envia_pro_sim, false)
      expect redirect_to(contrato)
		end
	end

  describe "PATCH #enviar_para_sim" do
		it "enviar_para_sim" do
			contrato = FactoryBot.create :licitacao_contrato, :com_lotes_do_contrato, envia_pro_sim: false
      contrato.update_column(:envia_pro_sim, true)
      expect redirect_to(contrato)
		end
	end

	describe "GET #editar_data_de_envio_pro_sim" do
		it "atribui o contrato requisitado à @contrato" do
			contrato = FactoryBot.create :licitacao_contrato, :com_lotes_do_contrato
			get :editar_data_de_envio_pro_sim, params: {id: contrato.id}
			expect(assigns(:contrato)).to eq(contrato)
		end
	end

	describe "PATCH #atualizar_data_de_envio_pro_sim" do
		context "com parâmetros válidos" do
			let(:data_valida) {
				{ envia_pro_sim: true }
				{ data_de_envio_pro_sim: Date.today }
			}

			it "atualiza o processo requisitado e redireciona pro @processo" do
				contrato = Licitacao::Contrato.create! atributos_validos
				patch :atualizar_data_de_envio_pro_sim, params: { id: contrato.to_param, licitacao_contrato: data_valida }
				contrato.reload
				expect(contrato.data_de_envio_pro_sim).to eq Date.today
			end
		end

		context "com parâmetros inválidos" do
			let(:data_invalida) {
				{ envia_pro_sim: true }
				{ data_de_envio_pro_sim: nil }
			}

			it "não atualiza o processo e re-renderiza o template :editar_data_de_envio_pro_sim" do
				contrato = Licitacao::Contrato.create! atributos_validos
				patch :atualizar_data_de_envio_pro_sim, params: {id: contrato.to_param, licitacao_contrato: data_invalida}
				expect(response).to render_template("editar_data_de_envio_pro_sim")
			end
		end
	end
end
