Skip to content

Commit

Permalink
Merge branch 'master' into feature/function/scheduled-weekly-report
Browse files Browse the repository at this point in the history
# Conflicts:
#	src/backend/api/Fusion.Resources.Api/Controllers/Utilities/UtilitiesController.cs
  • Loading branch information
BouVid committed Oct 24, 2023
2 parents 92cf532 + cc1975f commit e847769
Show file tree
Hide file tree
Showing 72 changed files with 549 additions and 231 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/test-api.yml
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ jobs:
- name: Setup .NET
uses: actions/setup-dotnet@v1
with:
dotnet-version: 6.0.x
dotnet-version: 7.0.x
- name: Restore dependencies
run: dotnet restore
- name: Build
Expand Down
4 changes: 3 additions & 1 deletion pipelines/function-pr-pipeline.yml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ pr:
variables:
subscriptionService: 'PROJECT_PORTAL (63b791ae-b2bc-41a1-ac66-806c4e69bffe)'
webPackage: '$(Pipeline.Workspace)/drop/Fusion.Resources.Functions.zip'
prNumber: $(System.PullRequest.PullRequestNumber)

stages:
- template: templates/build-function.yml
Expand All @@ -43,10 +44,11 @@ stages:
runOnce:
deploy:
steps:
- template: templates/deploy-function-template.yml
- template: templates/deploy-function-pr-template.yml
parameters:
envName: $(environment)
clientId: '5a842df8-3238-415d-b168-9f16a6a6031b'
pullRequestNumber: $(prNumber)

- task: AzureFunctionApp@1
inputs:
Expand Down
94 changes: 94 additions & 0 deletions pipelines/templates/deploy-function-pr-template.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
#
# Template for deploying pull request function app.
#
# Done this instead of making one template to include all ifs-and-buts.
#


parameters:
envName: ''
clientId: ''
pullRequestNumber: ''
fusionResource: '5a842df8-3238-415d-b168-9f16a6a6031b'
templateFile: $(Build.SourcesDirectory)/src/backend/function/Fusion.Resources.Functions/Deployment/function.template.json
disabledFunctionsFile: $(Build.SourcesDirectory)/src/backend/function/Fusion.Resources.Functions/Deployment/disabled-functions.json

steps:
- checkout: self
- task: AzurePowerShell@4
displayName: 'Deploy Function ARM template'
inputs:
azureSubscription: 'PROJECT_PORTAL (63b791ae-b2bc-41a1-ac66-806c4e69bffe)'
ScriptType: 'InlineScript'
FailOnStandardError: true
azurePowerShellVersion: 'LatestVersion'
Inline: |
#
# Hardcoded to PR env
#
$environment = "pr"
$pullRequestNumber = "${{ parameters.pullRequestNumber }}"
$fusionEnvironment = "ci"
$functionAppName = "func-fap-resources-$environment"
$envVaultName = "kv-fap-resources-$environment"
$envResourceGroup = Get-AzResourceGroup -Tag @{ "fusion-app-component" = "resources-rg-$environment" }
if ($envResourceGroup -eq $null) { throw "Cannot locate resource group for environment '$environment'" }
$resourceGroup = $envResourceGroup.ResourceGroupName
Write-Host "Using resource group $resourceGroup"
#
# Generate the app settings
#
# Sett correct resources URI based on environment
$resourcesFunctionUri = "https://resources-api-pr-$pullRequestNumber.fusion-dev.net/"
$settings = @{
clientId = "${{ parameters.clientId }}"
secretIds = @{
clientSecret = "https://$envVaultName.vault.azure.net:443/secrets/AzureAd--ClientSecret"
serviceBus = "https://$envVaultName.vault.azure.net:443/secrets/Connectionstrings--ServiceBus"
}
endpoints = @{
lineorg = "https://fusion-s-lineorg-$fusionEnvironment.azurewebsites.net"
org = "https://fusion-s-org-$fusionEnvironment.azurewebsites.net"
people = "https://fusion-s-people-$fusionEnvironment.azurewebsites.net"
resources = "$resourcesFunctionUri"
notifications = "https://fusion-s-notification-$fusionEnvironment.azurewebsites.net"
context = "https://fusion-s-context-$fusionEnvironment.azurewebsites.net"
portal = "https://fusion-s-portal-$fusionEnvironment.azurewebsites.net"
}
resources = @{
fusion = "${{ parameters.fusionResource }}"
}
queues = @{
provisionPosition = "provision-position"
}
}
New-AzResourceGroupDeployment -Mode Incremental -Name "resources-function" -ResourceGroupName $resourceGroup -TemplateFile "${{ parameters.templateFile }}" `
-env-name $environment `
-settings $settings
$functionApp = Get-AzWebApp -ResourceGroupName $resourceGroup -Name $functionAppName
Set-AzKeyVaultAccessPolicy -VaultName $envVaultName -ResourceGroupName $resourceGroup -ObjectId $functionApp.Identity.PrincipalId -PermissionsToSecrets get
## Load disabled functions
$disabledFunctionConfig = ConvertFrom-Json (Get-Content "${{ parameters.disabledFunctionsFile }}" -Raw)
$disabledFunctions = $disabledFunctionConfig | where -Property environment -eq $environment | Select -expandproperty disabledFunctions
Write-Host "Disabled functions"
$disabledFunctions
$settings = @{}
ForEach ($kvp in $functionApp.SiteConfig.AppSettings) {
$settings[$kvp.Name] = $kvp.Value
}
## Mark functions as disabled
$disabledFunctions | ForEach-Object { $settings["AzureWebJobs.$_.Disabled"] = "true" }
## Update web app settings for function app
Set-AzWebApp -ResourceGroupName $resourceGroup -Name $functionAppName -AppSettings $settings
2 changes: 1 addition & 1 deletion src/backend/.config/dotnet-tools.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
"isRoot": true,
"tools": {
"dotnet-ef": {
"version": "5.0.2",
"version": "7.0.12",
"commands": [
"dotnet-ef"
]
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<TargetFramework>net7.0</TargetFramework>
<Nullable>enable</Nullable>
</PropertyGroup>

Expand All @@ -13,7 +13,7 @@
</Content>
</ItemGroup>
<ItemGroup>
<PackageReference Include="Fusion.Integration.Authorization" Version="6.10.6" />
<PackageReference Include="Fusion.Integration.Authorization" Version="7.0.4" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\api\Fusion.Resources.Database\Fusion.Resources.Database.csproj" />
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,8 @@
using Fusion.Resources.Domain;
using Fusion.AspNetCore.FluentAuthorization;
using Fusion.Authorization;
using Fusion.Integration.Profile.Internal;
using Fusion.Resources.Domain;
using MediatR;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using System.Threading.Tasks;
Expand All @@ -12,22 +16,69 @@ public class AdminController : ResourceControllerBase
{

private readonly IOrgUnitCache orgUnitCache;
private readonly IMediator mediator;

public AdminController(IOrgUnitCache orgUnitCache)
public AdminController(IOrgUnitCache orgUnitCache, IMediator mediator)
{

this.orgUnitCache = orgUnitCache;
this.mediator = mediator;
}

[HttpGet("admin/cache/org-units")]

public async Task<ActionResult> CleareCache()
{
#region Authorization

var authResult = await Request.RequireAuthorizationAsync(r =>
{
r.AlwaysAccessWhen().FullControl().FullControlInternal();
r.AnyOf(or =>
{
or.FullControl();
or.FullControlInternal();
or.BeTrustedApplication();
});
});

if (authResult.Unauthorized)
return authResult.CreateForbiddenResponse();

#endregion Authorization

await orgUnitCache.ClearOrgUnitCacheAsync();

return Ok();
}


[HttpPost("admin/cache/reset-internal-cache")]
public async Task<ActionResult> ClearInternalCache()
{
#region Authorization

var authResult = await Request.RequireAuthorizationAsync(r =>
{
r.AlwaysAccessWhen().FullControl().FullControlInternal();
r.AnyOf(or =>
{
or.FullControl();
or.FullControlInternal();
or.BeTrustedApplication();
});
});

if (authResult.Unauthorized)
return authResult.CreateForbiddenResponse();

#endregion Authorization

await mediator.Publish(new DistributedEvents.ResetCacheNotification());

return new OkObjectResult(new { message = "Cache reset has been queued for all instances."});
}


}
}
Original file line number Diff line number Diff line change
Expand Up @@ -166,7 +166,7 @@ public async Task<ActionResult<ApiDepartmentResponsible>> AddDelegatedResourceOw
UpdatedByAzureUniqueId = User.GetAzureUniqueId() ?? User.GetApplicationId()
}.WithReason(request.Reason);

await DispatchAsync(command);
await DispatchCommandAsync(command);

}
catch (RoleDelegationExistsError ex)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -352,7 +352,7 @@ public async Task<ActionResult> DeletePersonAbsence([FromRoute] string personId,
#endregion


await DispatchAsync(new DeletePersonAbsence(id, absenceId));
await DispatchCommandAsync(new DeletePersonAbsence(id, absenceId));

return NoContent();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -249,7 +249,7 @@ public async Task<ActionResult> DeletePersonalNote(string personId, Guid noteId)
if (!notes.Any(n => n.Id == noteId))
return ApiErrors.NotFound("Could not locate note for user");

await DispatchAsync(new Domain.Commands.DeletePersonNote(noteId, user.azureId));
await DispatchCommandAsync(new Domain.Commands.DeletePersonNote(noteId, user.azureId));

return NoContent();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -270,7 +270,7 @@ public async Task<ActionResult> ResetAllocationState(string fullDepartmentString
return ApiErrors.NotFound("Could not locate allocation on person");


await DispatchAsync(new Domain.Commands.ResetAllocationState(allocation.Project.OrgProjectId, allocation.PositionId, instanceId));
await DispatchCommandAsync(new Domain.Commands.ResetAllocationState(allocation.Project.OrgProjectId, allocation.PositionId, instanceId));

return NoContent();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -786,7 +786,7 @@ public async Task<ActionResult<ApiResourceAllocationRequest>> StartProjectReques

await using var eventTransaction = await notificationClient.BeginTransactionAsync();
await using var transaction = await BeginTransactionAsync();
await DispatchAsync(new Logic.Commands.ResourceAllocationRequest.Initialize(requestId));
await DispatchCommandAsync(new Logic.Commands.ResourceAllocationRequest.Initialize(requestId));
await transaction.CommitAsync();
await eventTransaction.CommitAsync();
}
Expand Down Expand Up @@ -833,7 +833,7 @@ public async Task<ActionResult<ApiResourceAllocationRequest>> StartResourceOwner
{
await using var eventTransaction = await notificationClient.BeginTransactionAsync();
await using var transaction = await BeginTransactionAsync();
await DispatchAsync(new Logic.Commands.ResourceAllocationRequest.Initialize(requestId));
await DispatchCommandAsync(new Logic.Commands.ResourceAllocationRequest.Initialize(requestId));
await transaction.CommitAsync();
await eventTransaction.CommitAsync();
}
Expand Down Expand Up @@ -881,7 +881,7 @@ public async Task<ActionResult> DeleteAllocationRequest(Guid requestId)

await using var eventTransaction = await notificationClient.BeginTransactionAsync();
await using var transaction = await BeginTransactionAsync();
await DispatchAsync(new DeleteInternalRequest(requestId));
await DispatchCommandAsync(new DeleteInternalRequest(requestId));

await transaction.CommitAsync();
await eventTransaction.CommitAsync();
Expand Down Expand Up @@ -921,7 +921,7 @@ public async Task<ActionResult<ApiResourceAllocationRequest>> ProvisionProjectAl
await using var eventTransaction = await notificationClient.BeginTransactionAsync();
await using var scope = await BeginTransactionAsync();

await DispatchAsync(new Logic.Commands.ResourceAllocationRequest.Provision(requestId)
await DispatchCommandAsync(new Logic.Commands.ResourceAllocationRequest.Provision(requestId)
{
ForceProvision = force
});
Expand Down Expand Up @@ -956,7 +956,7 @@ public async Task<ActionResult<ApiResourceAllocationRequest>> ApproveProjectAllo

try
{
await DispatchAsync(new Logic.Commands.ResourceAllocationRequest.Approve(requestId));
await DispatchCommandAsync(new Logic.Commands.ResourceAllocationRequest.Approve(requestId));
await scope.CommitAsync();
await eventTransaction.CommitAsync();
}
Expand Down Expand Up @@ -995,7 +995,7 @@ public async Task<ActionResult<ApiResourceAllocationRequest>> ApproveProjectAllo

try
{
await DispatchAsync(new Logic.Commands.ResourceAllocationRequest.Approve(requestId));
await DispatchCommandAsync(new Logic.Commands.ResourceAllocationRequest.Approve(requestId));
await scope.CommitAsync();
await eventTransaction.CommitAsync();
}
Expand Down Expand Up @@ -1281,7 +1281,7 @@ public async Task<ActionResult<ApiRequestComment>> UpdateRequestComment(Guid req

#endregion Authorization

await DispatchAsync(new UpdateComment(commentId, update.Content));
await DispatchCommandAsync(new UpdateComment(commentId, update.Content));

comment = await DispatchAsync(new GetRequestComment(commentId));
return new ApiRequestComment(comment!);
Expand Down Expand Up @@ -1321,7 +1321,7 @@ public async Task<ActionResult> DeleteRequestComment(Guid requestId, Guid commen

#endregion Authorization

await DispatchAsync(new DeleteComment(commentId));
await DispatchCommandAsync(new DeleteComment(commentId));

return NoContent();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,7 @@ protected Task<IDbContextTransaction> BeginTransactionAsync()
return scope.BeginTransactionAsync();
}

protected Task DispatchAsync(IRequest command)
protected Task DispatchCommandAsync(IRequest command)
{
var mediator = HttpContext.RequestServices.GetRequiredService<IMediator>();
return mediator.Send(command);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -226,7 +226,7 @@ public async Task<ActionResult> DeleteResponsibilityMatrix(Guid matrixId)
if (responsibilityMatrix == null)
return FusionApiError.NotFound(matrixId, "Could not locate responsibility matrix");

await DispatchAsync(new DeleteResponsibilityMatrix(matrixId));
await DispatchCommandAsync(new DeleteResponsibilityMatrix(matrixId));

return NoContent();
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
using Fusion.Infrastructure.MediatR.Distributed;
using Fusion.Integration.Profile.Internal;
using Fusion.Resources.Domain;
using MediatR;
using System.Threading;
using System.Threading.Tasks;

namespace Fusion.Resources.Api.DistributedEvents
{
/// <summary>
/// Trigger a reset of all internal caches.
/// </summary>
public class ResetCacheNotification : DistributedNotification
{

public class Handler : INotificationHandler<ResetCacheNotification>
{
private readonly IOrgUnitCache orgUnitCache;
private readonly IProfileCache fusionProfileResolverCache;

public Handler(IOrgUnitCache orgUnitCache, IProfileCache fusionProfileResolverCache)
{
this.orgUnitCache = orgUnitCache;
this.fusionProfileResolverCache = fusionProfileResolverCache;
}

public async Task Handle(ResetCacheNotification notification, CancellationToken cancellationToken)
{
await fusionProfileResolverCache.ClearAsync();
await orgUnitCache.ClearOrgUnitCacheAsync();

}
}
}
}
Loading

0 comments on commit e847769

Please sign in to comment.