Skip to content

Commit

Permalink
Merge branch 'main' into cleanup
Browse files Browse the repository at this point in the history
  • Loading branch information
trulshj authored Jan 6, 2025
2 parents 301b1b7 + 3443fb5 commit 394eba3
Show file tree
Hide file tree
Showing 32 changed files with 1,055 additions and 67 deletions.
2 changes: 1 addition & 1 deletion backend/Api/Agreements/AgreementController.cs
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ public class AgreementController(
IAgreementsRepository agreementsRepository) : ControllerBase
{
private const string SelectedOrganizationNotFound = "Selected organization not found";


[HttpGet]
[Route("{agreementId:int}")]
Expand Down
24 changes: 19 additions & 5 deletions backend/Api/Common/StorageService.cs
Original file line number Diff line number Diff line change
Expand Up @@ -199,7 +199,20 @@ public async Task<Consultant> CreateConsultant(Organization org, ConsultantWrite
return consultant;
}

public Customer UpdateOrCreateCustomer(Organization org, string customerName, string orgUrlKey)
public Customer DeactivateOrActivateCustomer(int customerId, Organization org, bool active, string orgUrlKey)
{
var customer = GetCustomerFromId(orgUrlKey, customerId);
if (customer is null) return null;

customer.IsActive = active;
_dbContext.Customer.Update(customer);
_dbContext.SaveChanges();
ClearConsultantCache(orgUrlKey);

return customer;
}

public Customer FindOrCreateCustomer(Organization org, string customerName, string orgUrlKey)
{
var customer = context.Customer.Where(c => c.OrganizationId == org.Id)
.SingleOrDefault(c => c.Name == customerName);
Expand All @@ -211,14 +224,15 @@ public Customer UpdateOrCreateCustomer(Organization org, string customerName, st
Name = customerName,
Organization = org,
OrganizationId = org.Id,
Projects = []
Projects = [],
IsActive = true
};

context.Customer.Add(customer);
_dbContext.Customer.Add(customer);
_dbContext.SaveChanges();
ClearConsultantCache(orgUrlKey);
}

context.SaveChanges();
ClearConsultantCache(orgUrlKey);

return customer;
}
Expand Down
73 changes: 68 additions & 5 deletions backend/Api/Projects/ProjectController.cs
Original file line number Diff line number Diff line change
@@ -1,10 +1,14 @@
using Api.Common;
using Api.StaffingController;
using Core.Consultants;
using Core.Customers;
using Core.DomainModels;
using Core.Engagements;
using Core.Organizations;
using Core.Weeks;
using Infrastructure.DatabaseContext;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Http.HttpResults;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Caching.Memory;
Expand Down Expand Up @@ -45,10 +49,9 @@ public async Task<ActionResult<List<EngagementPerCustomerReadModel>>> Get(
var selectedOrgId = await organisationRepository.GetOrganizationByUrlKey(orgUrlKey, cancellationToken);
if (selectedOrgId is null) return BadRequest();

var absenceReadModels = new EngagementPerCustomerReadModel(-1, AbsenceCustomerName,
await context.Absence.Where(a => a.Organization.UrlKey == orgUrlKey).Select(absence =>
new EngagementReadModel(absence.Id, absence.Name, EngagementState.Absence, false))
.ToListAsync(cancellationToken));
var absenceReadModels = new EngagementPerCustomerReadModel(-1, AbsenceCustomerName, true,
context.Absence.Where(a => a.Organization.UrlKey == orgUrlKey).Select(absence =>
new EngagementReadModel(absence.Id, absence.Name, EngagementState.Absence, false)).ToList());

var projectReadModels = await context.Project.Include(project => project.Customer)
.Where(project =>
Expand All @@ -58,6 +61,7 @@ await context.Absence.Where(a => a.Organization.UrlKey == orgUrlKey).Select(abse
new EngagementPerCustomerReadModel(
a.Key.Id,
a.Key.Name,
a.Key.IsActive,
a.Select(e =>
new EngagementReadModel(e.Id, e.Name, e.State, e.IsBillable)).ToList()))
.ToListAsync(cancellationToken);
Expand Down Expand Up @@ -94,6 +98,7 @@ CancellationToken cancellationToken
return new CustomersWithProjectsReadModel(
customer.Id,
customer.Name,
customer.IsActive,
customer.Projects.Where(p =>
p.Staffings.Any(s => s.Week.CompareTo(thisWeek) >= 0) && p.State != EngagementState.Closed).Select(e =>
new EngagementReadModel(e.Id, e.Name, e.State, e.IsBillable)).ToList(),
Expand Down Expand Up @@ -286,6 +291,64 @@ private Engagement MergeProjects(int id)
.Single(p => p.Id == engagementToKeep.Id);
}

[HttpPut]
[Route("customer/{customerId}/activate")]
public async Task<Results<Ok, NotFound<string>>> Put([FromRoute] int customerId, [FromQuery] bool activate, string orgUrlKey, CancellationToken ct)
{

var service = new StorageService(cache, context);
var selectedOrg = await organisationRepository.GetOrganizationByUrlKey(orgUrlKey, ct);
if (selectedOrg is null) return TypedResults.NotFound("Selected org not found");
var customer = service.DeactivateOrActivateCustomer(customerId, selectedOrg, activate, orgUrlKey);
if (customer is null) return TypedResults.NotFound("Selected customer not found");
return TypedResults.Ok();
}


[HttpPut]
public async Task<ActionResult<ProjectWithCustomerModel>> Put([FromRoute] string orgUrlKey,
[FromBody] EngagementWriteModel body, CancellationToken ct)
{
var service = new StorageService(cache, context);

var selectedOrg = await organisationRepository.GetOrganizationByUrlKey(orgUrlKey, ct);
if (selectedOrg is null) return BadRequest("Selected org not found");

if (body.CustomerName == AbsenceCustomerName)
return Ok(HandleAbsenceChange(body, orgUrlKey));

var customer = service.FindOrCreateCustomer(selectedOrg, body.CustomerName, orgUrlKey);

var project = context.Project
.Include(p => p.Customer)
.SingleOrDefault(p => p.Customer.Id == customer.Id
&& p.Name == body.ProjectName
);

if (project is null)
{
project = new Engagement
{
Customer = customer,
State = body.BookingType,
Staffings = new List<Staffing>(),
Consultants = new List<Consultant>(),
Name = body.ProjectName,
IsBillable = body.IsBillable
};

context.Project.Add(project);
}

await context.SaveChangesAsync(ct);
service.ClearConsultantCache(orgUrlKey);

var responseModel =
new ProjectWithCustomerModel(project.Name, customer.Name, project.State, project.IsBillable, project.Id);

return Ok(responseModel);
}


private ProjectWithCustomerModel HandleAbsenceChange(EngagementWriteModel body, string orgUrlKey)
{
Expand All @@ -298,7 +361,7 @@ private CustomersWithProjectsReadModel HandleGetAbsenceWithAbsences(string orgUr
{
var vacation = new EngagementReadModel(-1, "Ferie", EngagementState.Absence, false);

var readModel = new CustomersWithProjectsReadModel(-1, AbsenceCustomerName + " og Ferie",
var readModel = new CustomersWithProjectsReadModel(-1, AbsenceCustomerName + " og Ferie", true,
context.Absence.Where(a => a.Organization.UrlKey == orgUrlKey).Select(absence =>
new EngagementReadModel(absence.Id, absence.Name, EngagementState.Absence, false)).ToList(),
new List<EngagementReadModel>());
Expand Down
28 changes: 17 additions & 11 deletions backend/Api/Projects/ProjectModels.cs
Original file line number Diff line number Diff line change
@@ -1,18 +1,23 @@
using System.ComponentModel.DataAnnotations;
using System.Runtime.InteropServices;
using Core.Customers;
using Core.Engagements;

namespace Api.Projects;

public record EngagementPerCustomerReadModel(
[property: Required] int CustomerId,
[property: Required] string CustomerName,
[property: Required] List<EngagementReadModel> Engagements);
int CustomerId,
string CustomerName,
bool IsActive,
List<EngagementReadModel> Engagements);



public record EngagementReadModel(
[property: Required] int EngagementId,
[property: Required] string EngagementName,
[property: Required] EngagementState BookingType,
[property: Required] bool IsBillable);
int EngagementId,
string EngagementName,
EngagementState BookingType,
bool IsBillable);

public record EngagementWriteModel(
EngagementState BookingType,
Expand Down Expand Up @@ -43,7 +48,8 @@ public record UpdateProjectWriteModel(
public record UpdateEngagementNameWriteModel(int EngagementId, string EngagementName);

public record CustomersWithProjectsReadModel(
[property: Required] int CustomerId,
[property: Required] string CustomerName,
[property: Required] List<EngagementReadModel> ActiveEngagements,
[property: Required] List<EngagementReadModel> InactiveEngagements);
int CustomerId,
string CustomerName,
bool IsActive,
List<EngagementReadModel> ActiveEngagements,
List<EngagementReadModel> InactiveEngagements);
8 changes: 4 additions & 4 deletions backend/Core/Customers/Customer.cs
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
using System.ComponentModel;
using System.ComponentModel.DataAnnotations.Schema;
using Core.Agreements;
using Core.Engagements;
Expand All @@ -13,10 +14,9 @@ public class Customer
[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public int Id { get; init; }
public required string Name { get; set; }
public required Organization Organization { get; set; }
public required List<Engagement> Projects { get; set; }

public ICollection<Agreement> Agreements { get; init; } = new List<Agreement>();
public required List<Engagement> Projects { get; init; }
public bool IsActive { get; set; } = true;

public required string OrganizationId { get; init; }
public required Organization Organization { get; init; }
}
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ protected override void OnModelCreating(ModelBuilder modelBuilder)
.HasForeignKey(customer => customer.OrganizationId);

Check failure on line 56 in backend/Infrastructure/DatabaseContext/ApplicationContext.cs

View workflow job for this annotation

GitHub Actions / dotnet_core_project_tests (8.x.x)

'Customer' does not contain a definition for 'OrganizationId' and no accessible extension method 'OrganizationId' accepting a first argument of type 'Customer' could be found (are you missing a using directive or an assembly reference?)

modelBuilder.Entity<Customer>()
.HasIndex(customer => new { customer.OrganizationId, customer.Name })
.HasIndex(customer => new { customer.OrganizationId, customer.Name, customer.IsActive })

Check failure on line 59 in backend/Infrastructure/DatabaseContext/ApplicationContext.cs

View workflow job for this annotation

GitHub Actions / dotnet_core_project_tests (8.x.x)

'Customer' does not contain a definition for 'OrganizationId' and no accessible extension method 'OrganizationId' accepting a first argument of type 'Customer' could be found (are you missing a using directive or an assembly reference?)
.IsUnique();

modelBuilder.Entity<Customer>()
Expand Down
Loading

0 comments on commit 394eba3

Please sign in to comment.