mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
Moved Slack OAuth to take into account the Organization it's being stored for. Added methods to store the top level integration for Slack
This commit is contained in:
parent
61a621b04b
commit
ffda25608c
@ -1,31 +1,52 @@
|
|||||||
using Bit.Core.Services;
|
using Bit.Core.Context;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
namespace Bit.Api.AdminConsole.Controllers;
|
namespace Bit.Api.AdminConsole.Controllers;
|
||||||
|
|
||||||
[Route("slack/oauth")]
|
[Route("slack/oauth")]
|
||||||
public class SlackOAuthController(ISlackService slackService) : Controller
|
[Authorize("Application")]
|
||||||
|
public class SlackOAuthController(
|
||||||
|
ICurrentContext currentContext,
|
||||||
|
IOrganizationIntegrationConfigurationRepository integrationConfigurationRepository,
|
||||||
|
ISlackService slackService) : Controller
|
||||||
{
|
{
|
||||||
[HttpGet("redirect")]
|
[HttpGet("redirect/{id}")]
|
||||||
public IActionResult RedirectToSlack()
|
public async Task<IActionResult> RedirectToSlack(string id)
|
||||||
{
|
{
|
||||||
string callbackUrl = Url.RouteUrl(nameof(OAuthCallback));
|
var orgIdGuid = new Guid(id);
|
||||||
|
if (!await currentContext.OrganizationOwner(orgIdGuid))
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
string callbackUrl = Url.RouteUrl(nameof(OAuthCallback), new { id = id }, currentContext.HttpContext.Request.Scheme);
|
||||||
var redirectUrl = slackService.GetRedirectUrl(callbackUrl);
|
var redirectUrl = slackService.GetRedirectUrl(callbackUrl);
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(redirectUrl))
|
if (string.IsNullOrEmpty(redirectUrl))
|
||||||
{
|
{
|
||||||
return BadRequest("Slack not currently supported.");
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
return Redirect(redirectUrl);
|
return Redirect(redirectUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("callback")]
|
[HttpGet("callback/{id}", Name = nameof(OAuthCallback))]
|
||||||
public async Task<IActionResult> OAuthCallback([FromQuery] string code)
|
public async Task<IActionResult> OAuthCallback(string id, [FromQuery] string code)
|
||||||
{
|
{
|
||||||
|
var orgIdGuid = new Guid(id);
|
||||||
|
if (!await currentContext.OrganizationOwner(orgIdGuid))
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(code))
|
if (string.IsNullOrEmpty(code))
|
||||||
{
|
{
|
||||||
return BadRequest("Missing code from Slack.");
|
throw new BadRequestException("Missing code from Slack.");
|
||||||
}
|
}
|
||||||
|
|
||||||
string callbackUrl = Url.RouteUrl(nameof(OAuthCallback));
|
string callbackUrl = Url.RouteUrl(nameof(OAuthCallback));
|
||||||
@ -33,15 +54,13 @@ public class SlackOAuthController(ISlackService slackService) : Controller
|
|||||||
|
|
||||||
if (string.IsNullOrEmpty(token))
|
if (string.IsNullOrEmpty(token))
|
||||||
{
|
{
|
||||||
return BadRequest("Invalid response from Slack.");
|
throw new BadRequestException("Invalid response from Slack.");
|
||||||
}
|
}
|
||||||
|
|
||||||
SaveTokenToDatabase(token);
|
await integrationConfigurationRepository.CreateOrganizationIntegrationAsync(
|
||||||
|
orgIdGuid,
|
||||||
|
IntegrationType.Slack,
|
||||||
|
new SlackIntegration(token));
|
||||||
return Ok("Slack OAuth successful. Your bot is now installed.");
|
return Ok("Slack OAuth successful. Your bot is now installed.");
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SaveTokenToDatabase(string botToken)
|
|
||||||
{
|
|
||||||
Console.WriteLine($"Stored bot token for team: {botToken}");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -30,6 +30,7 @@ using Bit.Api.Auth.Models.Request.WebAuthn;
|
|||||||
using Bit.Core.AdminConsole.Services.NoopImplementations;
|
using Bit.Core.AdminConsole.Services.NoopImplementations;
|
||||||
using Bit.Core.Auth.Models.Data;
|
using Bit.Core.Auth.Models.Data;
|
||||||
using Bit.Core.Auth.Identity.TokenProviders;
|
using Bit.Core.Auth.Identity.TokenProviders;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Core.Tools.ImportFeatures;
|
using Bit.Core.Tools.ImportFeatures;
|
||||||
using Bit.Core.Tools.ReportFeatures;
|
using Bit.Core.Tools.ReportFeatures;
|
||||||
@ -222,10 +223,14 @@ public class Startup
|
|||||||
{
|
{
|
||||||
services.AddHttpClient(SlackService.HttpClientName);
|
services.AddHttpClient(SlackService.HttpClientName);
|
||||||
services.AddSingleton<ISlackService, SlackService>();
|
services.AddSingleton<ISlackService, SlackService>();
|
||||||
|
services.AddSingleton<IOrganizationIntegrationConfigurationRepository,
|
||||||
|
LocalOrganizationIntegrationConfigurationRepository>();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
services.AddSingleton<ISlackService, NoopSlackService>();
|
services.AddSingleton<ISlackService, NoopSlackService>();
|
||||||
|
services.AddSingleton<IOrganizationIntegrationConfigurationRepository,
|
||||||
|
LocalOrganizationIntegrationConfigurationRepository>();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
namespace Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
|
public record SlackIntegration(string token);
|
@ -1,12 +1,17 @@
|
|||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Models.Data.Integrations;
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
#nullable enable
|
|
||||||
|
|
||||||
namespace Bit.Core.Repositories;
|
namespace Bit.Core.Repositories;
|
||||||
|
|
||||||
public interface IOrganizationIntegrationConfigurationRepository
|
public interface IOrganizationIntegrationConfigurationRepository
|
||||||
{
|
{
|
||||||
Task<List<IntegrationConfiguration<T>>> GetConfigurationsAsync<T>(IntegrationType integrationType,
|
Task<List<IntegrationConfiguration<T>>> GetConfigurationsAsync<T>(
|
||||||
Guid organizationId, EventType eventType);
|
Guid organizationId,
|
||||||
|
IntegrationType integrationType,
|
||||||
|
EventType eventType);
|
||||||
|
|
||||||
|
Task CreateOrganizationIntegrationAsync<T>(
|
||||||
|
Guid organizationId,
|
||||||
|
IntegrationType integrationType,
|
||||||
|
T configuration);
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
using Bit.Core.Enums;
|
using System.Text.Json;
|
||||||
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Models.Data.Integrations;
|
using Bit.Core.Models.Data.Integrations;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
|
|
||||||
namespace Bit.Core.Repositories;
|
namespace Bit.Core.Repositories;
|
||||||
|
|
||||||
public class OrganizationIntegrationConfigurationRepository(GlobalSettings globalSettings)
|
public class LocalOrganizationIntegrationConfigurationRepository(GlobalSettings globalSettings)
|
||||||
: IOrganizationIntegrationConfigurationRepository
|
: IOrganizationIntegrationConfigurationRepository
|
||||||
{
|
{
|
||||||
public async Task<List<IntegrationConfiguration<T>>> GetConfigurationsAsync<T>(IntegrationType integrationType,
|
public async Task<List<IntegrationConfiguration<T>>> GetConfigurationsAsync<T>(Guid organizationId,
|
||||||
Guid organizationId,
|
IntegrationType integrationType,
|
||||||
EventType eventType)
|
EventType eventType)
|
||||||
{
|
{
|
||||||
var configurations = new List<IntegrationConfiguration<T>>();
|
var configurations = new List<IntegrationConfiguration<T>>();
|
||||||
@ -39,13 +40,13 @@ public class OrganizationIntegrationConfigurationRepository(GlobalSettings globa
|
|||||||
return configurations;
|
return configurations;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<IEnumerable<IntegrationConfiguration<T>>> GetAllConfigurationsAsync<T>(Guid organizationId) => throw new NotImplementedException();
|
public async Task CreateOrganizationIntegrationAsync<T>(
|
||||||
|
Guid organizationId,
|
||||||
|
IntegrationType integrationType,
|
||||||
|
T configuration)
|
||||||
|
{
|
||||||
|
var json = JsonSerializer.Serialize(configuration);
|
||||||
|
|
||||||
public async Task AddConfigurationAsync<T>(Guid organizationId, IntegrationType integrationType, EventType eventType,
|
Console.WriteLine($"Organization: {organizationId}, IntegrationType: {integrationType}, Configuration: {json}");
|
||||||
IntegrationConfiguration<T> configuration) =>
|
}
|
||||||
throw new NotImplementedException();
|
|
||||||
|
|
||||||
public async Task UpdateConfigurationAsync<T>(IntegrationConfiguration<T> configuration) => throw new NotImplementedException();
|
|
||||||
|
|
||||||
public async Task DeleteConfigurationAsync(Guid id) => throw new NotImplementedException();
|
|
||||||
}
|
}
|
@ -6,6 +6,6 @@ public interface ISlackService
|
|||||||
Task<List<string>> GetChannelIdsAsync(string token, List<string> channelNames);
|
Task<List<string>> GetChannelIdsAsync(string token, List<string> channelNames);
|
||||||
Task<string> GetDmChannelByEmailAsync(string token, string email);
|
Task<string> GetDmChannelByEmailAsync(string token, string email);
|
||||||
string GetRedirectUrl(string redirectUrl);
|
string GetRedirectUrl(string redirectUrl);
|
||||||
Task SendSlackMessageByChannelIdAsync(string token, string message, string channelId);
|
|
||||||
Task<string> ObtainTokenViaOAuth(string code, string redirectUrl);
|
Task<string> ObtainTokenViaOAuth(string code, string redirectUrl);
|
||||||
|
Task SendSlackMessageByChannelIdAsync(string token, string message, string channelId);
|
||||||
}
|
}
|
||||||
|
@ -13,10 +13,7 @@ public class SlackEventHandler(
|
|||||||
public async Task HandleEventAsync(EventMessage eventMessage)
|
public async Task HandleEventAsync(EventMessage eventMessage)
|
||||||
{
|
{
|
||||||
var organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
var organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
||||||
var configurations = await configurationRepository.GetConfigurationsAsync<SlackConfiguration>(
|
var configurations = await configurationRepository.GetConfigurationsAsync<SlackConfiguration>(organizationId, IntegrationType.Slack, eventMessage.Type);
|
||||||
IntegrationType.Slack,
|
|
||||||
organizationId, eventMessage.Type
|
|
||||||
);
|
|
||||||
|
|
||||||
foreach (var configuration in configurations)
|
foreach (var configuration in configurations)
|
||||||
{
|
{
|
||||||
|
@ -24,46 +24,6 @@ public class SlackService(
|
|||||||
return (await GetChannelIdsAsync(token, new List<string> { channelName })).FirstOrDefault();
|
return (await GetChannelIdsAsync(token, new List<string> { channelName })).FirstOrDefault();
|
||||||
}
|
}
|
||||||
|
|
||||||
public string GetRedirectUrl(string redirectUrl)
|
|
||||||
{
|
|
||||||
return $"https://slack.com/oauth/v2/authorize?client_id={_clientId}&scope={_scopes}&redirect_uri={redirectUrl}";
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<string> ObtainTokenViaOAuth(string code, string redirectUrl)
|
|
||||||
{
|
|
||||||
var tokenResponse = await _httpClient.PostAsync("https://slack.com/api/oauth.v2.access",
|
|
||||||
new FormUrlEncodedContent(new[]
|
|
||||||
{
|
|
||||||
new KeyValuePair<string, string>("client_id", _clientId),
|
|
||||||
new KeyValuePair<string, string>("client_secret", _clientSecret),
|
|
||||||
new KeyValuePair<string, string>("code", code),
|
|
||||||
new KeyValuePair<string, string>("redirect_uri", redirectUrl)
|
|
||||||
}));
|
|
||||||
|
|
||||||
SlackOAuthResponse result;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
result = await tokenResponse.Content.ReadFromJsonAsync<SlackOAuthResponse>();
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
result = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result == null)
|
|
||||||
{
|
|
||||||
logger.LogError("Error obtaining token via OAuth: Unknown error");
|
|
||||||
return string.Empty;
|
|
||||||
}
|
|
||||||
if (!result.Ok)
|
|
||||||
{
|
|
||||||
logger.LogError("Error obtaining token via OAuth: " + result.Error);
|
|
||||||
return string.Empty;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result.AccessToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<List<string>> GetChannelIdsAsync(string token, List<string> channelNames)
|
public async Task<List<string>> GetChannelIdsAsync(string token, List<string> channelNames)
|
||||||
{
|
{
|
||||||
var matchingChannelIds = new List<string>();
|
var matchingChannelIds = new List<string>();
|
||||||
@ -111,6 +71,46 @@ public class SlackService(
|
|||||||
return await OpenDmChannel(token, userId);
|
return await OpenDmChannel(token, userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public string GetRedirectUrl(string redirectUrl)
|
||||||
|
{
|
||||||
|
return $"https://slack.com/oauth/v2/authorize?client_id={_clientId}&scope={_scopes}&redirect_uri={redirectUrl}";
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<string> ObtainTokenViaOAuth(string code, string redirectUrl)
|
||||||
|
{
|
||||||
|
var tokenResponse = await _httpClient.PostAsync("https://slack.com/api/oauth.v2.access",
|
||||||
|
new FormUrlEncodedContent(new[]
|
||||||
|
{
|
||||||
|
new KeyValuePair<string, string>("client_id", _clientId),
|
||||||
|
new KeyValuePair<string, string>("client_secret", _clientSecret),
|
||||||
|
new KeyValuePair<string, string>("code", code),
|
||||||
|
new KeyValuePair<string, string>("redirect_uri", redirectUrl)
|
||||||
|
}));
|
||||||
|
|
||||||
|
SlackOAuthResponse result;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
result = await tokenResponse.Content.ReadFromJsonAsync<SlackOAuthResponse>();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
result = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result == null)
|
||||||
|
{
|
||||||
|
logger.LogError("Error obtaining token via OAuth: Unknown error");
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
if (!result.Ok)
|
||||||
|
{
|
||||||
|
logger.LogError("Error obtaining token via OAuth: " + result.Error);
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.AccessToken;
|
||||||
|
}
|
||||||
|
|
||||||
public async Task SendSlackMessageByChannelIdAsync(string token, string message, string channelId)
|
public async Task SendSlackMessageByChannelIdAsync(string token, string message, string channelId)
|
||||||
{
|
{
|
||||||
var payload = JsonContent.Create(new { channel = channelId, text = message });
|
var payload = JsonContent.Create(new { channel = channelId, text = message });
|
||||||
|
@ -21,11 +21,8 @@ public class WebhookEventHandler(
|
|||||||
{
|
{
|
||||||
Guid organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
Guid organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
||||||
|
|
||||||
var configurations = await configurationRepository.GetConfigurationsAsync<WebhookConfiguration>(
|
var configurations = await configurationRepository.GetConfigurationsAsync<WebhookConfiguration>(organizationId,
|
||||||
IntegrationType.Webhook,
|
IntegrationType.Webhook, eventMessage.Type);
|
||||||
organizationId,
|
|
||||||
eventMessage.Type
|
|
||||||
);
|
|
||||||
|
|
||||||
foreach (var configuration in configurations)
|
foreach (var configuration in configurations)
|
||||||
{
|
{
|
||||||
|
@ -119,7 +119,7 @@ public class Startup
|
|||||||
globalSettings,
|
globalSettings,
|
||||||
globalSettings.EventLogging.RabbitMq.EventRepositoryQueueName));
|
globalSettings.EventLogging.RabbitMq.EventRepositoryQueueName));
|
||||||
|
|
||||||
services.AddSingleton<IOrganizationIntegrationConfigurationRepository, OrganizationIntegrationConfigurationRepository>();
|
services.AddSingleton<IOrganizationIntegrationConfigurationRepository, LocalOrganizationIntegrationConfigurationRepository>();
|
||||||
|
|
||||||
if (CoreHelpers.SettingHasValue(globalSettings.Slack.ClientId) &&
|
if (CoreHelpers.SettingHasValue(globalSettings.Slack.ClientId) &&
|
||||||
CoreHelpers.SettingHasValue(globalSettings.Slack.ClientSecret) &&
|
CoreHelpers.SettingHasValue(globalSettings.Slack.ClientSecret) &&
|
||||||
|
@ -1,4 +1,9 @@
|
|||||||
using Bit.Api.AdminConsole.Controllers;
|
using Bit.Api.AdminConsole.Controllers;
|
||||||
|
using Bit.Core.Context;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Test.Common.AutoFixture;
|
using Bit.Test.Common.AutoFixture;
|
||||||
using Bit.Test.Common.AutoFixture.Attributes;
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
@ -13,63 +18,112 @@ namespace Bit.Api.Test.AdminConsole.Controllers;
|
|||||||
public class SlackOAuthControllerTests
|
public class SlackOAuthControllerTests
|
||||||
{
|
{
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public async Task OAuthCallback_ThrowsBadResultWhenCodeIsEmpty(SutProvider<SlackOAuthController> sutProvider)
|
public async Task OAuthCallback_CompletesSuccessfully(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
{
|
|
||||||
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
|
||||||
|
|
||||||
var requestAction = await sutProvider.Sut.OAuthCallback(string.Empty);
|
|
||||||
|
|
||||||
Assert.IsType<BadRequestObjectResult>(requestAction);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task OAuthCallback_ThrowsBadResultWhenSlackServiceReturnsEmpty(SutProvider<SlackOAuthController> sutProvider)
|
|
||||||
{
|
{
|
||||||
|
var token = "xoxb-test-token";
|
||||||
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
sutProvider.GetDependency<ISlackService>()
|
sutProvider.GetDependency<ISlackService>()
|
||||||
.ObtainTokenViaOAuth(Arg.Any<string>(), Arg.Any<string>())
|
.ObtainTokenViaOAuth(Arg.Any<string>(), Arg.Any<string>())
|
||||||
.Returns(string.Empty);
|
.Returns(token);
|
||||||
|
|
||||||
var requestAction = await sutProvider.Sut.OAuthCallback("A_test_code");
|
var requestAction = await sutProvider.Sut.OAuthCallback(organizationId.ToString(), "A_test_code");
|
||||||
|
|
||||||
Assert.IsType<BadRequestObjectResult>(requestAction);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task OAuthCallback_CompletesSuccessfully(SutProvider<SlackOAuthController> sutProvider)
|
|
||||||
{
|
|
||||||
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
|
||||||
sutProvider.GetDependency<ISlackService>()
|
|
||||||
.ObtainTokenViaOAuth(Arg.Any<string>(), Arg.Any<string>())
|
|
||||||
.Returns("xoxb-test-token");
|
|
||||||
|
|
||||||
var requestAction = await sutProvider.Sut.OAuthCallback("A_test_code");
|
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IOrganizationIntegrationConfigurationRepository>().Received(1)
|
||||||
|
.CreateOrganizationIntegrationAsync(organizationId, IntegrationType.Slack, new SlackIntegration(token));
|
||||||
Assert.IsType<OkObjectResult>(requestAction);
|
Assert.IsType<OkObjectResult>(requestAction);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public void Redirect_ShouldRedirectToSlack(SutProvider<SlackOAuthController> sutProvider)
|
public async Task OAuthCallback_ThrowsBadResultWhenCodeIsEmpty(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
{
|
{
|
||||||
var expectedUrl = "https://localhost/";
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<BadRequestException>(async () => await sutProvider.Sut.OAuthCallback(organizationId.ToString(), string.Empty));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task OAuthCallback_ThrowsBadResultWhenSlackServiceReturnsEmpty(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<ISlackService>()
|
||||||
|
.ObtainTokenViaOAuth(Arg.Any<string>(), Arg.Any<string>())
|
||||||
|
.Returns(string.Empty);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<BadRequestException>(async () => await sutProvider.Sut.OAuthCallback(organizationId.ToString(), "A_test_code"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task OAuthCallback_ThrowsNotFoundIfUserIsNotOrganizationAdmin(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
|
{
|
||||||
|
var token = "xoxb-test-token";
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(false);
|
||||||
|
sutProvider.GetDependency<ISlackService>()
|
||||||
|
.ObtainTokenViaOAuth(Arg.Any<string>(), Arg.Any<string>())
|
||||||
|
.Returns(token);
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.OAuthCallback(organizationId.ToString(), "A_test_code"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task Redirect_ShouldRedirectToSlack(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
|
{
|
||||||
|
var expectedUrl = $"https://localhost/{organizationId.ToString()}";
|
||||||
|
|
||||||
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
sutProvider.GetDependency<ISlackService>().GetRedirectUrl(Arg.Any<string>()).Returns(expectedUrl);
|
sutProvider.GetDependency<ISlackService>().GetRedirectUrl(Arg.Any<string>()).Returns(expectedUrl);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.HttpContext.Request.Scheme
|
||||||
|
.Returns("https");
|
||||||
|
|
||||||
var requestAction = sutProvider.Sut.RedirectToSlack();
|
var requestAction = await sutProvider.Sut.RedirectToSlack(organizationId.ToString());
|
||||||
|
|
||||||
var redirectResult = Assert.IsType<RedirectResult>(requestAction);
|
var redirectResult = Assert.IsType<RedirectResult>(requestAction);
|
||||||
Assert.Equal(expectedUrl, redirectResult.Url);
|
Assert.Equal(expectedUrl, redirectResult.Url);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public void Redirect_ThrowsBadResultWhenSlackServiceReturnsEmpty(SutProvider<SlackOAuthController> sutProvider)
|
public async Task Redirect_ThrowsNotFoundWhenSlackServiceReturnsEmpty(SutProvider<SlackOAuthController> sutProvider, Guid organizationId)
|
||||||
{
|
{
|
||||||
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
sutProvider.GetDependency<ISlackService>().GetRedirectUrl(Arg.Any<string>()).Returns(string.Empty);
|
sutProvider.GetDependency<ISlackService>().GetRedirectUrl(Arg.Any<string>()).Returns(string.Empty);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(true);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.HttpContext.Request.Scheme
|
||||||
|
.Returns("https");
|
||||||
|
|
||||||
var requestAction = sutProvider.Sut.RedirectToSlack();
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.RedirectToSlack(organizationId.ToString()));
|
||||||
|
}
|
||||||
|
|
||||||
Assert.IsType<BadRequestObjectResult>(requestAction);
|
[Theory, BitAutoData]
|
||||||
|
public async Task Redirect_ThrowsNotFoundWhenUserIsNotOrganizationAdmin(SutProvider<SlackOAuthController> sutProvider,
|
||||||
|
Guid organizationId)
|
||||||
|
{
|
||||||
|
sutProvider.Sut.Url = Substitute.For<IUrlHelper>();
|
||||||
|
sutProvider.GetDependency<ISlackService>().GetRedirectUrl(Arg.Any<string>()).Returns(string.Empty);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.OrganizationOwner(organizationId)
|
||||||
|
.Returns(false);
|
||||||
|
sutProvider.GetDependency<ICurrentContext>()
|
||||||
|
.HttpContext.Request.Scheme
|
||||||
|
.Returns("https");
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<NotFoundException>(async () => await sutProvider.Sut.RedirectToSlack(organizationId.ToString()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -24,10 +24,8 @@ public class SlackEventHandlerTests
|
|||||||
private SutProvider<SlackEventHandler> GetSutProvider(
|
private SutProvider<SlackEventHandler> GetSutProvider(
|
||||||
List<IntegrationConfiguration<SlackConfiguration>> integrationConfigurations)
|
List<IntegrationConfiguration<SlackConfiguration>> integrationConfigurations)
|
||||||
{
|
{
|
||||||
_repository.GetConfigurationsAsync<SlackConfiguration>(
|
_repository.GetConfigurationsAsync<SlackConfiguration>(Arg.Any<Guid>(),
|
||||||
IntegrationType.Slack,
|
IntegrationType.Slack, Arg.Any<EventType>())
|
||||||
Arg.Any<Guid>(),
|
|
||||||
Arg.Any<EventType>())
|
|
||||||
.Returns(integrationConfigurations);
|
.Returns(integrationConfigurations);
|
||||||
|
|
||||||
return new SutProvider<SlackEventHandler>()
|
return new SutProvider<SlackEventHandler>()
|
||||||
|
@ -47,11 +47,8 @@ public class WebhookEventHandlerTests
|
|||||||
clientFactory.CreateClient(WebhookEventHandler.HttpClientName).Returns(_httpClient);
|
clientFactory.CreateClient(WebhookEventHandler.HttpClientName).Returns(_httpClient);
|
||||||
|
|
||||||
var repository = Substitute.For<IOrganizationIntegrationConfigurationRepository>();
|
var repository = Substitute.For<IOrganizationIntegrationConfigurationRepository>();
|
||||||
repository.GetConfigurationsAsync<WebhookConfiguration>(
|
repository.GetConfigurationsAsync<WebhookConfiguration>(Arg.Any<Guid>(),
|
||||||
IntegrationType.Webhook,
|
IntegrationType.Webhook, Arg.Any<EventType>()).Returns(configurations);
|
||||||
Arg.Any<Guid>(),
|
|
||||||
Arg.Any<EventType>()
|
|
||||||
).Returns(configurations);
|
|
||||||
|
|
||||||
return new SutProvider<WebhookEventHandler>()
|
return new SutProvider<WebhookEventHandler>()
|
||||||
.SetDependency(repository)
|
.SetDependency(repository)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user