package org.initialde.yakasave.Integration;

import org.initialde.yakasave.Api.Requests.LoginRequest;
import org.initialde.yakasave.Domain.Entities.SavingsFund;

import org.initialde.yakasave.UserFactory;
import org.initialde.yakasave.Domain.Enums.TypeSavingsFund;
import org.initialde.yakasave.Domain.ValueObject.Amount;
import org.initialde.yakasave.Infrastructure.Persistence.SavingsFundRepository;
import org.initialde.yakasave.Infrastructure.Persistence.UserRepository;
import org.initialde.yakasave.Infrastructure.authentication.AuthenticationGateway;
import org.initialde.yakasave.Unit.SavingsFundFactory;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.http.MediaType;
import org.springframework.test.web.servlet.MockMvc;

import java.time.LocalDate;
import java.util.UUID;

import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.patch;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;

@SpringBootTest
@AutoConfigureMockMvc
public class CloseSavingsFundControllerTest {
    private final String ENDPOINT = "/savings-fund/{reference}/close";
    @Autowired
    private SavingsFundRepository savingsFundRepository;

    @Autowired
    private MockMvc mockMvc;

    @Autowired
    private AuthenticationGateway authenticationGateway;

    @Autowired
    private UserRepository userRepository;

    @BeforeEach
    public void setup() {
        savingsFundRepository.deleteAll();
        userRepository.deleteAll();
    }

    @Test
    public void shouldCloseSavingsFundWhenBalanceIsZero() throws Exception {
        var reference = UUID.randomUUID().toString();
        int maxAllowedMembers = 2;

        final var ownerCredentials = new LoginRequest("owner", "1234");
        final var owner = UserFactory.create( "owner", "1234");
        userRepository.save(owner);

        var savingsFund = SavingsFundFactory.createCollectiveSavingsFund(reference, owner, maxAllowedMembers);
        savingsFundRepository.save(savingsFund);

        userRepository.save(owner);
        String token = authenticationGateway.authenticate(ownerCredentials);
        this.mockMvc.perform(patch(ENDPOINT, reference)
                        .header("Authorization", "Bearer " + token)
                        .contentType(MediaType.APPLICATION_JSON))                .andExpect(status().isOk())
                .andReturn();
    }

    @Test
    public void shouldFailCloseSavingsFundWhenBalanceIsGreaterThanZero() throws Exception {
        var reference = UUID.randomUUID().toString();

        final var ownerCredentials = new LoginRequest("owner", "1234");
        var owner = UserFactory.create( "owner", "1234");
        userRepository.save(owner);

        var savingsFund = SavingsFund
                .builder()
                .reference(reference)
                .owner(owner)
                .balance(3000)
                .isActive(true)
                .goalAmount(Amount.MINIMUM_GOAL_AMOUNT)
                .type(TypeSavingsFund.PERSONAL)
                .deadline(LocalDate.now().plusDays(1))
                .build();
        savingsFundRepository.save(savingsFund);

        userRepository.save(owner);
        String token = authenticationGateway.authenticate(ownerCredentials);
        this.mockMvc.perform(patch(ENDPOINT, reference)
                        .header("Authorization", "Bearer " + token)
                        .contentType(MediaType.APPLICATION_JSON))                .andExpect(status().isConflict())
                .andReturn();
    }

    @Test
    public void shouldFailCloseSavingsFundWhenAlreadyClosed() throws Exception {
        var reference = UUID.randomUUID().toString();

        final var ownerCredentials = new LoginRequest("owner", "1234");
        var owner = UserFactory.create( "owner", "1234");
        userRepository.save(owner);

        var savingsFund = SavingsFund
                .builder()
                .reference(reference)
                .owner(owner)
                .balance(0)
                .isActive(false)
                .goalAmount(Amount.MINIMUM_GOAL_AMOUNT)
                .type(TypeSavingsFund.PERSONAL)
                .deadline(LocalDate.now().plusDays(1))
                .build();
        savingsFundRepository.save(savingsFund);

        userRepository.save(owner);
        String token = authenticationGateway.authenticate(ownerCredentials);
        this.mockMvc.perform(patch(ENDPOINT, reference)
                        .header("Authorization", "Bearer " + token)
                        .contentType(MediaType.APPLICATION_JSON))                .andExpect(status().isConflict())
                .andReturn();
    }
}
