xUnit ClaimsPrincipal mock and passing into controller User is null

473 Views Asked by At

I have been reading articles trying to figure this one out. Structured off of this article How to add claims in a mock ClaimsPrincipal. I am still getting a null user from my controller and test fails on a null object of User inside the controller.

BusinessController

    [Route("api/[controller]")]
    [ApiController]
    public class BusinessController : ControllerBase
    {
        private readonly IGBusinessRepository businessRepository;
        private readonly IPersonRepository personRepository;
        private readonly IUserClaims userClaims;

        public BusinessController(IGBusinessRepository businessRepository,
                                    IPersonRepository personRepository,
                                    IUserClaims userClaims)
        {
            this.businessRepository = businessRepository;
            this.personRepository = personRepository;
            this.userClaims = userClaims;
        }

        // GET api/<BusinessController>/5
        [HttpGet("{id}")]
        [Authorize]
        public async Task<IActionResult> GetBusiness(Guid businessId)
        {
            var userGuid = userClaims.GetUserGuid(User.Claims);
            var ownerId = await personRepository.GetPersonIdByGUID(userGuid);
            var business = await businessRepository.GetBusinessById(businessId);
            if(business != null && business.OwnerId == businessId)
            {
                return Ok(business);
            }

            return BadRequest("Bad business id or your not the owner");            
        }

UserClaims

    public class UserClaims : IUserClaims
    {
        public string GetUserGuid(IEnumerable<Claim> claims)
        {
            var claimType = "http://schemas.xmlsoap.org/ws/2005/05/identity/claims/nameidentifier";
            var guidClaim = claims.Where(c => c.Type == claimType).Select(s => s.Value).SingleOrDefault();


            return guidClaim;
        }
    }

TestIdentity

    public class TestIdentity : ClaimsIdentity
    {
        public TestIdentity(params Claim[] claims) : base(claims)
        {

        }
    }

TestPrincipal

    public class TestPrincipal : ClaimsPrincipal
    {
        public TestPrincipal(params Claim[] claims) : base(new TestIdentity(claims))
        {

        }
    }

BusinessControllerTests

    public class BusinessControllerTests
    {
        //Controller 
        private readonly Mock<IGBusinessRepository> mockBusinessRepository;
        private readonly Mock<IPersonRepository> mockPersonRepository;
        private readonly Mock<IUserClaims> mockUserClaims;
        private BusinessController controller;

        //Objects
        private Guid id = Guid.NewGuid(); 

        public BusinessControllerTests()
        {
            mockBusinessRepository = new Mock<IGBusinessRepository>();
            mockPersonRepository = new Mock<IPersonRepository>();
            mockUserClaims = new Mock<IUserClaims>();
            controller = new BusinessController(mockBusinessRepository.Object, mockPersonRepository.Object, mockUserClaims.Object);
        }

        [Fact]
        public async Task GetBussiness_NotBusinessOwner_ReturnsBadRequest()
        {
            //Arrange
            var userGuidString = Guid.NewGuid().ToString();
            var ownerId = Guid.NewGuid();
            var userClaim = new TestPrincipal(new Claim("name", "[email protected]"));
            Thread.CurrentPrincipal = userClaim;
            
            //mockUserClaims.Setup(repo => repo.GetUserGuid(userClaim)).Returns(userGuidString);
            mockPersonRepository.Setup(repo => repo.GetPersonIdByGUID(userGuidString));
            mockBusinessRepository.Setup(repo => repo.GetBusinessById(id)).ReturnsAsync(business);

            //Act
            var result = await controller.GetBusiness(id);

            //Assert
            Assert.IsType<BadRequestResult>(result);
        }

        private GobiezBusiness business = new GobiezBusiness()
        {
            Id = new MongoDB.Bson.ObjectId(),
            BusinessId = Guid.NewGuid(),
            Name = "Test",
            Email = "[email protected]",
            Address = "123 A street",
            State = "WA",
            ZipCode = "12345",
            PhoneNumber = "123-456-7890",
            OwnerId = Guid.NewGuid()
        };   
    }
1

There are 1 best solutions below

1
On BEST ANSWER

The controller was not arranged correctly to be able to access the principal

// ... omitted for brevity

var userClaim = new TestPrincipal(new Claim("name", "[email protected]"));
var httpContext = new DefaultHttpContext() {
    User = userClaim;
};

//Controller needs a controller context to access HttpContext
var controllerContext = new ControllerContext() {
    HttpContext = httpContext
};

//assign context to controller    
BusinessController controller = new BusinessController(
    mockBusinessRepository.Object, 
    mockPersonRepository.Object, 
    mockUserClaims.Object)
{
    ControllerContext = controllerContext
};

//Act
var result = await controller.GetBusiness(id);

// ... omitted for brevity

The controller should also be created within the scope of the test being executed.

This line in the subject under test

//...

var userGuid = userClaims.GetUserGuid(User.Claims);

//...

Should now return the created TestPrincipal arranged in the test