mirror of
https://github.com/bitwarden/server.git
synced 2025-04-06 05:28:15 -05:00
Added Create/Delete for OrganizationIntegrationConfigurations
This commit is contained in:
parent
1f2d049b10
commit
25b150d560
@ -0,0 +1,61 @@
|
|||||||
|
using Bit.Api.AdminConsole.Models.Request.Organizations;
|
||||||
|
using Bit.Api.AdminConsole.Models.Response.Organizations;
|
||||||
|
using Bit.Core.Context;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
|
namespace Bit.Api.AdminConsole.Controllers;
|
||||||
|
|
||||||
|
[Route("organizations/{organizationId:guid}/integrations/{integrationId:guid}/configurations")]
|
||||||
|
[Authorize("Application")]
|
||||||
|
public class OrganizationIntegrationConfigurationController(
|
||||||
|
ICurrentContext currentContext,
|
||||||
|
IOrganizationIntegrationRepository integrationRepository,
|
||||||
|
IOrganizationIntegrationConfigurationRepository integrationConfigurationRepository) : Controller
|
||||||
|
{
|
||||||
|
[HttpPost("")]
|
||||||
|
public async Task<OrganizationIntegrationConfigurationResponseModel> PostAsync(
|
||||||
|
Guid organizationId,
|
||||||
|
Guid integrationId,
|
||||||
|
[FromBody] OrganizationIntegrationConfigurationCreateRequestModel model)
|
||||||
|
{
|
||||||
|
if (!await currentContext.OrganizationOwner(organizationId))
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
var integration = await integrationRepository.GetByIdAsync(integrationId);
|
||||||
|
if (integration == null || integration.OrganizationId != organizationId)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
var organizationIntegrationConfiguration = model.ToOrganizationIntegrationConfiguration();
|
||||||
|
var configuration = await integrationConfigurationRepository.CreateAsync(organizationIntegrationConfiguration);
|
||||||
|
return new OrganizationIntegrationConfigurationResponseModel(configuration);
|
||||||
|
}
|
||||||
|
|
||||||
|
[HttpDelete("{configurationId:guid}")]
|
||||||
|
[HttpPost("{configurationId:guid}/delete")]
|
||||||
|
public async Task DeleteAsync(Guid organizationId, Guid integrationId, Guid configurationId)
|
||||||
|
{
|
||||||
|
if (!await currentContext.OrganizationOwner(organizationId))
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
var integration = await integrationRepository.GetByIdAsync(integrationId);
|
||||||
|
if (integration == null || integration.OrganizationId != organizationId)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
var configuration = await integrationConfigurationRepository.GetByIdAsync(configurationId);
|
||||||
|
if (configuration is null)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
await integrationConfigurationRepository.DeleteAsync(configuration);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,36 @@
|
|||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
|
||||||
|
#nullable enable
|
||||||
|
|
||||||
|
namespace Bit.Api.AdminConsole.Models.Request.Organizations;
|
||||||
|
|
||||||
|
public class OrganizationIntegrationConfigurationCreateRequestModel : IValidatableObject
|
||||||
|
{
|
||||||
|
public string? Configuration { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
public Guid OrganizationIntegrationId { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
public EventType EventType { get; set; }
|
||||||
|
|
||||||
|
public string? Template { get; set; }
|
||||||
|
|
||||||
|
public OrganizationIntegrationConfiguration ToOrganizationIntegrationConfiguration()
|
||||||
|
{
|
||||||
|
return new OrganizationIntegrationConfiguration()
|
||||||
|
{
|
||||||
|
OrganizationIntegrationId = OrganizationIntegrationId,
|
||||||
|
Configuration = Configuration,
|
||||||
|
EventType = EventType,
|
||||||
|
Template = Template
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
|
||||||
|
{
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,31 @@
|
|||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Api;
|
||||||
|
|
||||||
|
#nullable enable
|
||||||
|
|
||||||
|
namespace Bit.Api.AdminConsole.Models.Response.Organizations;
|
||||||
|
|
||||||
|
public class OrganizationIntegrationConfigurationResponseModel : ResponseModel
|
||||||
|
{
|
||||||
|
public OrganizationIntegrationConfigurationResponseModel(OrganizationIntegrationConfiguration organizationIntegrationConfiguration, string obj = "organizationIntegrationConfiguration")
|
||||||
|
: base(obj)
|
||||||
|
{
|
||||||
|
if (organizationIntegrationConfiguration == null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(organizationIntegrationConfiguration));
|
||||||
|
}
|
||||||
|
|
||||||
|
Id = organizationIntegrationConfiguration.Id;
|
||||||
|
Configuration = organizationIntegrationConfiguration.Configuration;
|
||||||
|
CreationDate = organizationIntegrationConfiguration.CreationDate;
|
||||||
|
EventType = organizationIntegrationConfiguration.EventType;
|
||||||
|
Template = organizationIntegrationConfiguration.Template;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Guid Id { get; set; }
|
||||||
|
public string? Configuration { get; set; }
|
||||||
|
public DateTime CreationDate { get; set; }
|
||||||
|
public EventType EventType { get; set; }
|
||||||
|
public string? Template { get; set; }
|
||||||
|
}
|
@ -0,0 +1,196 @@
|
|||||||
|
using Bit.Api.AdminConsole.Controllers;
|
||||||
|
using Bit.Api.AdminConsole.Models.Request.Organizations;
|
||||||
|
using Bit.Api.AdminConsole.Models.Response.Organizations;
|
||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.Context;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using NSubstitute;
|
||||||
|
using NSubstitute.ReturnsExtensions;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Api.Test.AdminConsole.Controllers;
|
||||||
|
|
||||||
|
[ControllerCustomize(typeof(OrganizationIntegrationConfigurationController))]
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class OrganizationIntegrationsConfigurationControllerTests
|
||||||
|
{
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task DeleteAsync_AllParamsProvided_Succeeds(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId,
|
||||||
|
OrganizationIntegration organizationIntegration,
|
||||||
|
OrganizationIntegrationConfiguration organizationIntegrationConfiguration)
|
||||||
|
{
|
||||||
|
organizationIntegration.OrganizationId = organizationId;
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegration);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegrationConfiguration);
|
||||||
|
|
||||||
|
await sutProvider.Sut.DeleteAsync(organizationId, organizationIntegration.Id, organizationIntegrationConfiguration.Id);
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IOrganizationIntegrationRepository>().Received(1)
|
||||||
|
.GetByIdAsync(organizationIntegration.Id);
|
||||||
|
await sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>().Received(1)
|
||||||
|
.GetByIdAsync(organizationIntegrationConfiguration.Id);
|
||||||
|
await sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>().Received(1)
|
||||||
|
.DeleteAsync(organizationIntegrationConfiguration);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task DeleteAsync_IntegrationConfigurationDoesNotExist_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId,
|
||||||
|
OrganizationIntegration organizationIntegration)
|
||||||
|
{
|
||||||
|
organizationIntegration.OrganizationId = organizationId;
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegration);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.DeleteAsync(organizationId, Guid.Empty, Guid.Empty));
|
||||||
|
}
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task DeleteAsync_IntegrationDoesNotExist_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.DeleteAsync(organizationId, Guid.Empty, Guid.Empty));
|
||||||
|
}
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task DeleteAsync_IntegrationDoesNotBelongToOrganization_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId,
|
||||||
|
OrganizationIntegration organizationIntegration)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegration);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.DeleteAsync(organizationId, organizationIntegration.Id, Guid.Empty));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task DeleteAsync_UserIsNotOrganizationAdmin_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.DeleteAsync(organizationId, Guid.Empty, Guid.Empty));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task PostAsync_AllParamsProvided_Succeeds(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId,
|
||||||
|
OrganizationIntegration organizationIntegration,
|
||||||
|
OrganizationIntegrationConfiguration organizationIntegrationConfiguration,
|
||||||
|
OrganizationIntegrationConfigurationCreateRequestModel model)
|
||||||
|
{
|
||||||
|
organizationIntegration.OrganizationId = organizationId;
|
||||||
|
model.OrganizationIntegrationId = organizationIntegration.Id;
|
||||||
|
|
||||||
|
var expected = new OrganizationIntegrationConfigurationResponseModel(organizationIntegrationConfiguration);
|
||||||
|
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegration);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>()
|
||||||
|
.CreateAsync(Arg.Any<OrganizationIntegrationConfiguration>())
|
||||||
|
.Returns(organizationIntegrationConfiguration);
|
||||||
|
var requestAction = await sutProvider.Sut.PostAsync(organizationId, organizationIntegration.Id, model);
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>().Received(1)
|
||||||
|
.CreateAsync(Arg.Any<OrganizationIntegrationConfiguration>());
|
||||||
|
Assert.IsType<OrganizationIntegrationConfigurationResponseModel>(requestAction);
|
||||||
|
Assert.Equal(expected.Id, requestAction.Id);
|
||||||
|
Assert.Equal(expected.Configuration, requestAction.Configuration);
|
||||||
|
Assert.Equal(expected.EventType, requestAction.EventType);
|
||||||
|
Assert.Equal(expected.Template, requestAction.Template);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task PostAsync_IntegrationDoesNotExist_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.PostAsync(
|
||||||
|
organizationId,
|
||||||
|
Guid.Empty,
|
||||||
|
new OrganizationIntegrationConfigurationCreateRequestModel()));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task PostAsync_IntegrationDoesNotBelongToOrganization_ThrowsNotFound(
|
||||||
|
SutProvider<OrganizationIntegrationConfigurationController> sutProvider,
|
||||||
|
Guid organizationId,
|
||||||
|
OrganizationIntegration organizationIntegration)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<IOrganizationIntegrationRepository>()
|
||||||
|
.GetByIdAsync(Arg.Any<Guid>())
|
||||||
|
.Returns(organizationIntegration);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.PostAsync(
|
||||||
|
organizationId,
|
||||||
|
organizationIntegration.Id,
|
||||||
|
new OrganizationIntegrationConfigurationCreateRequestModel()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task PostAsync_UserIsNotOrganizationAdmin_ThrowsNotFound(SutProvider<OrganizationIntegrationConfigurationController> sutProvider, Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.PostAsync(organizationId, Guid.Empty, new OrganizationIntegrationConfigurationCreateRequestModel()));
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user