Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
484 changes: 484 additions & 0 deletions .gitignore

Large diffs are not rendered by default.

13 changes: 13 additions & 0 deletions .idea/.idea.InvestmentPerformanceWebApi/.idea/.gitignore

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions .idea/.idea.InvestmentPerformanceWebApi/.idea/.name

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions .idea/.idea.InvestmentPerformanceWebApi/.idea/encodings.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

8 changes: 8 additions & 0 deletions .idea/.idea.InvestmentPerformanceWebApi/.idea/indexLayout.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

7 changes: 7 additions & 0 deletions .idea/.idea.InvestmentPerformanceWebApi/.idea/vcs.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

35 changes: 0 additions & 35 deletions InvestmentPerformanceWebAPI.md

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,181 @@
using System.Threading;
using System.Threading.Tasks;
using InvestmentPerformanceWebApi.Controllers;
using InvestmentPerformanceWebApi.Models;
using InvestmentPerformanceWebApi.Services;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Logging;
using Moq;
using Xunit;

namespace InvestmentPerformanceWebApi.Tests.Controllers;

public class InvestmentControllerTests
{
private readonly InvestmentController _controller;
private readonly Mock<IInvestmentService> _serviceMock;

public InvestmentControllerTests()
{
_serviceMock = new Mock<IInvestmentService>();
_controller = new InvestmentController(Mock.Of<ILogger<InvestmentController>>(), _serviceMock.Object);
}

[Fact]
public async Task Get_WithNonPositiveUserId_ReturnsBadRequest()
{
var result = await _controller.Get(0, CancellationToken.None);

var badRequest = Assert.IsType<BadRequestObjectResult>(result.Result);
Assert.Equal("UserId must be a positive integer.", badRequest.Value);
_serviceMock.VerifyNoOtherCalls();
}

[Fact]
public async Task Get_WithValidUserId_ReturnsOkWithInvestments()
{
var userId = 42;
var expected = new List<InvestmentDto>
{
new(1, "ETF A"),
new(2, "Stock B")
};

_serviceMock
.Setup(s => s.GetInvestmentsByUserIdAsync(userId, It.IsAny<CancellationToken>()))
.ReturnsAsync(expected);

var result = await _controller.Get(userId, CancellationToken.None);

var ok = Assert.IsType<OkObjectResult>(result.Result);
var actual = Assert.IsAssignableFrom<IReadOnlyList<InvestmentDto>>(ok.Value);
Assert.Equal(expected.Count, actual.Count);
_serviceMock.VerifyAll();
}

[Fact]
public async Task Get_WhenCancelled_Returns499()
{
var userId = 7;

_serviceMock
.Setup(s => s.GetInvestmentsByUserIdAsync(userId, It.IsAny<CancellationToken>()))
.ThrowsAsync(new OperationCanceledException());

var result = await _controller.Get(userId, CancellationToken.None);

var objectResult = Assert.IsType<ObjectResult>(result.Result);
Assert.Equal(499, objectResult.StatusCode);
Assert.Equal("Client Closed Request", objectResult.Value);
_serviceMock.VerifyAll();
}

[Fact]
public async Task Get_WhenUnhandledException_Returns500()
{
var userId = 9;

_serviceMock
.Setup(s => s.GetInvestmentsByUserIdAsync(userId, It.IsAny<CancellationToken>()))
.ThrowsAsync(new InvalidOperationException("boom"));

var result = await _controller.Get(userId, CancellationToken.None);

var objectResult = Assert.IsType<ObjectResult>(result.Result);
Assert.Equal(500, objectResult.StatusCode);
Assert.Equal("An error occurred while retrieving investments.", objectResult.Value);
_serviceMock.VerifyAll();
}

[Fact]
public async Task GetDetails_WithInvalidIds_ReturnsBadRequest()
{
var result = await _controller.GetDetails(0, -1, CancellationToken.None);

var badRequest = Assert.IsType<BadRequestObjectResult>(result.Result);
Assert.Equal("UserId and InvestmentId must be positive integers.", badRequest.Value);
_serviceMock.VerifyNoOtherCalls();
}

[Fact]
public async Task GetDetails_NotFound_ReturnsNotFound()
{
var userId = 5;
var investmentId = 123;

_serviceMock
.Setup(s => s.GetInvestmentDetailsAsync(userId, investmentId, It.IsAny<CancellationToken>()))
.ReturnsAsync((InvestmentDetailDto?)null);

var result = await _controller.GetDetails(userId, investmentId, CancellationToken.None);

Assert.IsType<NotFoundResult>(result.Result);
_serviceMock.VerifyAll();
}

[Fact]
public async Task GetDetails_Found_ReturnsOkWithDetails()
{
var userId = 5;
var investmentId = 123;

var expected =
new InvestmentDetailDto(investmentId, userId, "Bond C", 1000m, 100m, 1500m, 1000m, "2022-01-01", 10m);

_serviceMock
.Setup(s => s.GetInvestmentDetailsAsync(userId, investmentId, It.IsAny<CancellationToken>()))
.ReturnsAsync(expected);

var result = await _controller.GetDetails(userId, investmentId, CancellationToken.None);

var ok = Assert.IsType<OkObjectResult>(result.Result);
var actual = Assert.IsType<InvestmentDetailDto>(ok.Value);
Assert.Equal(expected.Id, actual.Id);
_serviceMock.VerifyAll();
}

[Fact]
public async Task GetDetails_WhenCancelled_Returns499()
{
var userId = 2;
var investmentId = 3;

_serviceMock
.Setup(s => s.GetInvestmentDetailsAsync(userId, investmentId, It.IsAny<CancellationToken>()))
.ThrowsAsync(new OperationCanceledException());

var result = await _controller.GetDetails(userId, investmentId, CancellationToken.None);

var objectResult = Assert.IsType<ObjectResult>(result.Result);
Assert.Equal(499, objectResult.StatusCode);
Assert.Equal("Client Closed Request", objectResult.Value);
_serviceMock.VerifyAll();
}

[Fact]
public async Task GetDetails_WhenUnhandledException_Returns500()
{
var userId = 2;
var investmentId = 3;

_serviceMock
.Setup(s => s.GetInvestmentDetailsAsync(userId, investmentId, It.IsAny<CancellationToken>()))
.ThrowsAsync(new Exception("unexpected"));

var result = await _controller.GetDetails(userId, investmentId, CancellationToken.None);

var objectResult = Assert.IsType<ObjectResult>(result.Result);
Assert.Equal(500, objectResult.StatusCode);
Assert.Equal("An error occurred while retrieving investment details.", objectResult.Value);
_serviceMock.VerifyAll();
}

[Fact]
public void GetHealth_ReturnsOk()
{
var result = _controller.Get();

var ok = Assert.IsType<OkObjectResult>(result);
Assert.Equal("Investment API is running.", ok.Value);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<TargetFramework>net8.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>

<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
</PropertyGroup>

<ItemGroup>
<PackageReference Include="coverlet.collector" Version="6.0.0"/>
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.8.0"/>
<PackageReference Include="Moq" Version="4.20.72" />
<PackageReference Include="xunit" Version="2.9.3" />
<PackageReference Include="xunit.runner.visualstudio" Version="3.1.4">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
</ItemGroup>

<ItemGroup>
<Using Include="Xunit"/>
</ItemGroup>

<ItemGroup>
<ProjectReference Include="..\InvestmentPerformanceWebApi\InvestmentPerformanceWebApi.csproj" />
</ItemGroup>

</Project>
9 changes: 9 additions & 0 deletions InvestmentPerformanceWebApi.Tests/UnitTest1.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
namespace InvestmentPerformanceWebApi.Tests;

public class UnitTest1
{
[Fact]
public void Test1()
{
}
}
38 changes: 38 additions & 0 deletions InvestmentPerformanceWebApi.sln
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@

Microsoft Visual Studio Solution File, Format Version 12.00
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "InvestmentPerformanceWebApi", "InvestmentPerformanceWebApi\InvestmentPerformanceWebApi.csproj", "{F47C3C24-2A75-4F75-901B-4678D686AC7F}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{CB7FBD03-01E6-4FB1-90FE-30E8D957AD4C}"
ProjectSection(SolutionItems) = preProject
README.md = README.md
EndProjectSection
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "InvestmentPerformanceWebApi", "InvestmentPerformanceWebApi\InvestmentPerformanceWebApi.csproj", "{7ABA96F3-82E3-4FE4-848A-66C21CF821F1}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Src", "Src", "{14289EB3-560B-4809-862B-D409D59D4175}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Tests", "Tests", "{270CFF4E-0879-4A11-8853-2B5A0D6A003D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "InvestmentPerformanceWebApi.Tests", "InvestmentPerformanceWebApi.Tests\InvestmentPerformanceWebApi.Tests.csproj", "{67657AED-A457-4EAC-BF4F-C221B0E03A7B}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{F47C3C24-2A75-4F75-901B-4678D686AC7F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{F47C3C24-2A75-4F75-901B-4678D686AC7F}.Debug|Any CPU.Build.0 = Debug|Any CPU
{F47C3C24-2A75-4F75-901B-4678D686AC7F}.Release|Any CPU.ActiveCfg = Release|Any CPU
{F47C3C24-2A75-4F75-901B-4678D686AC7F}.Release|Any CPU.Build.0 = Release|Any CPU
{67657AED-A457-4EAC-BF4F-C221B0E03A7B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{67657AED-A457-4EAC-BF4F-C221B0E03A7B}.Debug|Any CPU.Build.0 = Debug|Any CPU
{67657AED-A457-4EAC-BF4F-C221B0E03A7B}.Release|Any CPU.ActiveCfg = Release|Any CPU
{67657AED-A457-4EAC-BF4F-C221B0E03A7B}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{F47C3C24-2A75-4F75-901B-4678D686AC7F} = {7ABA96F3-82E3-4FE4-848A-66C21CF821F1}
{7ABA96F3-82E3-4FE4-848A-66C21CF821F1} = {14289EB3-560B-4809-862B-D409D59D4175}
{67657AED-A457-4EAC-BF4F-C221B0E03A7B} = {270CFF4E-0879-4A11-8853-2B5A0D6A003D}
EndGlobalSection
EndGlobal
25 changes: 25 additions & 0 deletions InvestmentPerformanceWebApi/.dockerignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
**/.dockerignore
**/.env
**/.git
**/.gitignore
**/.project
**/.settings
**/.toolstarget
**/.vs
**/.vscode
**/.idea
**/*.*proj.user
**/*.dbmdl
**/*.jfm
**/azds.yaml
**/bin
**/charts
**/docker-compose*
**/Dockerfile*
**/node_modules
**/npm-debug.log
**/obj
**/secrets.dev.yaml
**/values.dev.yaml
LICENSE
README.md
Loading