Functional testing is the next level above the unit testing and gives us confidence that the code we wrote works as intended. If we are talking about functional testing in the scope of ASP.NET, we mean how we test our API.
What actually functional testing is?
Functional testing is the process through which we determine if a piece of software is acting following pre-determined requirements. It uses black-box testing techniques, in which the tester does not know the internal system logic.
In the scope of API, we need to prepare possible requests that we expect to get and be sure that endpoints return a proper response for any payload.
Should we even spend time on it?
Definitely! If you don`t write integrational tests, then functional testing is a solution to ensure that your code has fewer bugs than it could have. When you can write integrational tests as well, it should be simple to reuse a shared codebase for functional and integrational tests. One difference will be in the proper environmental setup.
Let`s define a simple API
Our controller contains only single action to create some entity for test purposes. Our business rules may require any entity, which does not matter for our test example. Besides, we should remember about authentication nowadays. Security compliance requires protecting API with any authentication. Let`s assume that the current API requires an Authorization header with a Bearer token.
What will the functional test look like for this controller?
In our test project, I used NUnit as a test framework, but it is up to you to choose which better suits your needs. NUnit has all the required functionality to write good tests. Also, it`s pretty popular on the market.
Let`s add the following packages to the test project:
[Test] publicasync Task CreateAsync_Should_ReturnCreateResponse() { // Arrange var request = new CreateRequest { PropertyName = "propertyValue" }; var json = JsonConvert.SerializeObject(request); usingvar data = new StringContent(json, Encoding.UTF8, "application/json"); var token = await _tokenBuilder.BuildJWTToken(); _client?.DefaultRequestHeaders.Add("Authorization", "Bearer " + token);
var expected = new CreateResponse { PropertyName = "propertyValue" };
// Act var response = await _client?.PostAsync(CreateUrl, data)!;
// Assert response.StatusCode.Should().Be(HttpStatusCode.OK); var responseBody = await response.Content.ReadAsStringAsync(); var responseObject = JsonConvert.DeserializeObject<CreateResponse>(responseBody); responseObject.Should().BeEquivalentTo(expected); } }
We should cover that API endpoint protected by authentication, and clients can not request it without a proper token. To do this, we need to prepare a test server with a proper authentication setup. To do it in Startup, we need to override JwtBearerOptions with our test authentication, which relies on auth token signed by SymmetricSecurityKey. To comply with security rules, we should avoid storing any secrets in code, and I don`t suggest you violate this rule even for test projects.
When the authentication configuration is ready, we can generate the proper token to execute business logic in test cases. We can separate token generation into TokenBuilder and use it to get auth token.
Conclusion
I hope you feel that functional tests are pretty simple and allow you to verify your API each time something changes. We often include functional tests as a step in your build pipeline, and it gives you confidence that everything works well before deploying to the production environment.