mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
Refactored Slack and Webhook integrations to pull configurations dynamically from a new Repository
This commit is contained in:
parent
dc342713a6
commit
862f1821c8
7
src/Core/AdminConsole/Enums/IntegrationType.cs
Normal file
7
src/Core/AdminConsole/Enums/IntegrationType.cs
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
namespace Bit.Core.Enums;
|
||||||
|
|
||||||
|
public enum IntegrationType : int
|
||||||
|
{
|
||||||
|
Slack = 1,
|
||||||
|
Webhook = 2,
|
||||||
|
}
|
@ -0,0 +1,7 @@
|
|||||||
|
namespace Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
|
public class IntegrationConfiguration<T>
|
||||||
|
{
|
||||||
|
public T Configuration { get; set; }
|
||||||
|
public string Template { get; set; } = string.Empty;
|
||||||
|
}
|
@ -0,0 +1,8 @@
|
|||||||
|
namespace Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
|
public class SlackConfiguration
|
||||||
|
{
|
||||||
|
public string Token { get; set; } = string.Empty;
|
||||||
|
public List<string> Channels { get; set; } = new();
|
||||||
|
public List<string> UserEmails { get; set; } = new();
|
||||||
|
}
|
@ -0,0 +1,7 @@
|
|||||||
|
namespace Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
|
public class WebhookConfiguration
|
||||||
|
{
|
||||||
|
public string Url { get; set; } = string.Empty;
|
||||||
|
public string ApiKey { get; set; } = string.Empty;
|
||||||
|
}
|
@ -0,0 +1,15 @@
|
|||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
|
||||||
|
#nullable enable
|
||||||
|
|
||||||
|
namespace Bit.Core.Repositories;
|
||||||
|
|
||||||
|
public interface IOrganizationIntegrationConfigurationRepository
|
||||||
|
{
|
||||||
|
Task<IntegrationConfiguration<T>?> GetConfigurationAsync<T>(Guid organizationId, IntegrationType integrationType, EventType eventType);
|
||||||
|
Task<IEnumerable<IntegrationConfiguration<T>>> GetAllConfigurationsAsync<T>(Guid organizationId);
|
||||||
|
Task AddConfigurationAsync<T>(Guid organizationId, IntegrationType integrationType, EventType eventType, IntegrationConfiguration<T> configuration);
|
||||||
|
Task UpdateConfigurationAsync<T>(IntegrationConfiguration<T> configuration);
|
||||||
|
Task DeleteConfigurationAsync(Guid id);
|
||||||
|
}
|
@ -0,0 +1,65 @@
|
|||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
using Bit.Core.Settings;
|
||||||
|
|
||||||
|
namespace Bit.Core.Repositories;
|
||||||
|
|
||||||
|
#nullable enable
|
||||||
|
|
||||||
|
public class OrganizationIntegrationConfigurationRepository(GlobalSettings globalSettings)
|
||||||
|
: IOrganizationIntegrationConfigurationRepository
|
||||||
|
{
|
||||||
|
private readonly string _slackToken = globalSettings.EventLogging.SlackToken;
|
||||||
|
private readonly string _slackUserEmail = globalSettings.EventLogging.SlackUserEmail;
|
||||||
|
private readonly string _webhookUrl = globalSettings.EventLogging.WebhookUrl;
|
||||||
|
|
||||||
|
public async Task<IntegrationConfiguration<T>?> GetConfigurationAsync<T>(
|
||||||
|
Guid organizationId,
|
||||||
|
IntegrationType integrationType,
|
||||||
|
EventType eventType)
|
||||||
|
{
|
||||||
|
switch (integrationType)
|
||||||
|
{
|
||||||
|
case IntegrationType.Slack:
|
||||||
|
if (string.IsNullOrWhiteSpace(_slackToken) || string.IsNullOrWhiteSpace(_slackUserEmail))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return new IntegrationConfiguration<SlackConfiguration>()
|
||||||
|
{
|
||||||
|
Configuration = new SlackConfiguration
|
||||||
|
{
|
||||||
|
Token = _slackToken,
|
||||||
|
Channels = new List<string> { },
|
||||||
|
UserEmails = new List<string> { _slackUserEmail }
|
||||||
|
},
|
||||||
|
Template = "This is a test of the new Slack integration"
|
||||||
|
} as IntegrationConfiguration<T>;
|
||||||
|
case IntegrationType.Webhook:
|
||||||
|
if (string.IsNullOrWhiteSpace(_webhookUrl))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return new IntegrationConfiguration<WebhookConfiguration>()
|
||||||
|
{
|
||||||
|
Configuration = new WebhookConfiguration()
|
||||||
|
{
|
||||||
|
Url = _webhookUrl,
|
||||||
|
},
|
||||||
|
Template = "{ \"newObject\": true }"
|
||||||
|
} as IntegrationConfiguration<T>;
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<IEnumerable<IntegrationConfiguration<T>>> GetAllConfigurationsAsync<T>(Guid organizationId) => throw new NotImplementedException();
|
||||||
|
|
||||||
|
public async Task AddConfigurationAsync<T>(Guid organizationId, IntegrationType integrationType, EventType eventType,
|
||||||
|
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();
|
||||||
|
}
|
@ -1,32 +1,38 @@
|
|||||||
using System.Text.Json;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Models.Data;
|
using Bit.Core.Models.Data;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
|
||||||
namespace Bit.Core.Services;
|
namespace Bit.Core.Services;
|
||||||
|
|
||||||
public class SlackEventHandler(
|
public class SlackEventHandler(
|
||||||
GlobalSettings globalSettings,
|
IOrganizationIntegrationConfigurationRepository configurationRepository,
|
||||||
SlackMessageSender slackMessageSender)
|
SlackMessageSender slackMessageSender
|
||||||
: IEventMessageHandler
|
) : IEventMessageHandler
|
||||||
{
|
{
|
||||||
private readonly string _token = globalSettings.EventLogging.SlackToken;
|
|
||||||
private readonly string _email = globalSettings.EventLogging.SlackUserEmail;
|
|
||||||
|
|
||||||
public async Task HandleEventAsync(EventMessage eventMessage)
|
public async Task HandleEventAsync(EventMessage eventMessage)
|
||||||
{
|
{
|
||||||
await slackMessageSender.SendDirectMessageByEmailAsync(
|
Guid organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
||||||
_token,
|
|
||||||
JsonSerializer.Serialize(eventMessage),
|
var configuration = await configurationRepository.GetConfigurationAsync<SlackConfiguration>(
|
||||||
_email
|
organizationId,
|
||||||
);
|
IntegrationType.Slack,
|
||||||
|
eventMessage.Type);
|
||||||
|
if (configuration is not null)
|
||||||
|
{
|
||||||
|
await slackMessageSender.SendDirectMessageByEmailAsync(
|
||||||
|
configuration.Configuration.Token,
|
||||||
|
configuration.Template,
|
||||||
|
configuration.Configuration.UserEmails.First()
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task HandleManyEventsAsync(IEnumerable<EventMessage> eventMessages)
|
public async Task HandleManyEventsAsync(IEnumerable<EventMessage> eventMessages)
|
||||||
{
|
{
|
||||||
await slackMessageSender.SendDirectMessageByEmailAsync(
|
foreach (var eventMessage in eventMessages)
|
||||||
_token,
|
{
|
||||||
JsonSerializer.Serialize(eventMessages),
|
await HandleEventAsync(eventMessage);
|
||||||
_email
|
}
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,30 +1,45 @@
|
|||||||
using System.Net.Http.Json;
|
using System.Net.Http.Json;
|
||||||
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Models.Data;
|
using Bit.Core.Models.Data;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Models.Data.Integrations;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
|
||||||
|
#nullable enable
|
||||||
|
|
||||||
namespace Bit.Core.Services;
|
namespace Bit.Core.Services;
|
||||||
|
|
||||||
public class WebhookEventHandler(
|
public class WebhookEventHandler(
|
||||||
IHttpClientFactory httpClientFactory,
|
IHttpClientFactory httpClientFactory,
|
||||||
GlobalSettings globalSettings)
|
IOrganizationIntegrationConfigurationRepository configurationRepository)
|
||||||
: IEventMessageHandler
|
: IEventMessageHandler
|
||||||
{
|
{
|
||||||
private readonly HttpClient _httpClient = httpClientFactory.CreateClient(HttpClientName);
|
private readonly HttpClient _httpClient = httpClientFactory.CreateClient(HttpClientName);
|
||||||
private readonly string _webhookUrl = globalSettings.EventLogging.WebhookUrl;
|
|
||||||
|
|
||||||
public const string HttpClientName = "WebhookEventHandlerHttpClient";
|
public const string HttpClientName = "WebhookEventHandlerHttpClient";
|
||||||
|
|
||||||
public async Task HandleEventAsync(EventMessage eventMessage)
|
public async Task HandleEventAsync(EventMessage eventMessage)
|
||||||
{
|
{
|
||||||
var content = JsonContent.Create(eventMessage);
|
Guid organizationId = eventMessage.OrganizationId ?? Guid.NewGuid();
|
||||||
var response = await _httpClient.PostAsync(_webhookUrl, content);
|
|
||||||
response.EnsureSuccessStatusCode();
|
var configuration = await configurationRepository.GetConfigurationAsync<WebhookConfiguration>(
|
||||||
|
organizationId,
|
||||||
|
IntegrationType.Webhook,
|
||||||
|
eventMessage.Type);
|
||||||
|
if (configuration is not null)
|
||||||
|
{
|
||||||
|
var content = JsonContent.Create(configuration.Template);
|
||||||
|
var response = await _httpClient.PostAsync(
|
||||||
|
configuration.Configuration.Url,
|
||||||
|
content);
|
||||||
|
response.EnsureSuccessStatusCode();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task HandleManyEventsAsync(IEnumerable<EventMessage> eventMessages)
|
public async Task HandleManyEventsAsync(IEnumerable<EventMessage> eventMessages)
|
||||||
{
|
{
|
||||||
var content = JsonContent.Create(eventMessages);
|
foreach (var eventMessage in eventMessages)
|
||||||
var response = await _httpClient.PostAsync(_webhookUrl, content);
|
{
|
||||||
response.EnsureSuccessStatusCode();
|
await HandleEventAsync(eventMessage);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -74,6 +74,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<Folder Include="AdminConsole\Utilities\" />
|
||||||
<Folder Include="Resources\" />
|
<Folder Include="Resources\" />
|
||||||
<Folder Include="Properties\" />
|
<Folder Include="Properties\" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
using Bit.Core.AdminConsole.Services.Implementations;
|
using Bit.Core.AdminConsole.Services.Implementations;
|
||||||
using Bit.Core.Context;
|
using Bit.Core.Context;
|
||||||
using Bit.Core.IdentityServer;
|
using Bit.Core.IdentityServer;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
using Bit.Core.Utilities;
|
using Bit.Core.Utilities;
|
||||||
@ -117,33 +118,26 @@ public class Startup
|
|||||||
globalSettings,
|
globalSettings,
|
||||||
globalSettings.EventLogging.RabbitMq.EventRepositoryQueueName));
|
globalSettings.EventLogging.RabbitMq.EventRepositoryQueueName));
|
||||||
|
|
||||||
if (CoreHelpers.SettingHasValue(globalSettings.EventLogging.SlackToken) &&
|
services.AddSingleton<IOrganizationIntegrationConfigurationRepository, OrganizationIntegrationConfigurationRepository>();
|
||||||
CoreHelpers.SettingHasValue(globalSettings.EventLogging.SlackUserEmail))
|
|
||||||
{
|
|
||||||
services.AddHttpClient(SlackMessageSender.HttpClientName);
|
|
||||||
services.AddSingleton<SlackMessageSender>();
|
|
||||||
services.AddSingleton<SlackEventHandler>();
|
|
||||||
|
|
||||||
services.AddSingleton<IHostedService>(provider =>
|
services.AddHttpClient(SlackMessageSender.HttpClientName);
|
||||||
new RabbitMqEventListenerService(
|
services.AddSingleton<SlackMessageSender>();
|
||||||
provider.GetRequiredService<SlackEventHandler>(),
|
services.AddSingleton<SlackEventHandler>();
|
||||||
provider.GetRequiredService<ILogger<RabbitMqEventListenerService>>(),
|
|
||||||
globalSettings,
|
|
||||||
globalSettings.EventLogging.RabbitMq.SlackQueueName));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (CoreHelpers.SettingHasValue(globalSettings.EventLogging.WebhookUrl))
|
services.AddSingleton<IHostedService>(provider =>
|
||||||
{
|
new RabbitMqEventListenerService(
|
||||||
services.AddSingleton<WebhookEventHandler>();
|
provider.GetRequiredService<SlackEventHandler>(),
|
||||||
services.AddHttpClient(WebhookEventHandler.HttpClientName);
|
provider.GetRequiredService<ILogger<RabbitMqEventListenerService>>(),
|
||||||
|
globalSettings,
|
||||||
|
globalSettings.EventLogging.RabbitMq.SlackQueueName));
|
||||||
|
|
||||||
services.AddSingleton<IHostedService>(provider =>
|
services.AddSingleton<WebhookEventHandler>();
|
||||||
new RabbitMqEventListenerService(
|
services.AddSingleton<IHostedService>(provider =>
|
||||||
provider.GetRequiredService<WebhookEventHandler>(),
|
new RabbitMqEventListenerService(
|
||||||
provider.GetRequiredService<ILogger<RabbitMqEventListenerService>>(),
|
provider.GetRequiredService<WebhookEventHandler>(),
|
||||||
globalSettings,
|
provider.GetRequiredService<ILogger<RabbitMqEventListenerService>>(),
|
||||||
globalSettings.EventLogging.RabbitMq.WebhookQueueName));
|
globalSettings,
|
||||||
}
|
globalSettings.EventLogging.RabbitMq.WebhookQueueName));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user