mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 13:08:17 -05:00
168 lines
6.0 KiB
C#
168 lines
6.0 KiB
C#
using System.Net.Http.Headers;
|
|
using System.Net.Http.Json;
|
|
using System.Web;
|
|
using Bit.Core.Models.Slack;
|
|
using Bit.Core.Settings;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace Bit.Core.Services;
|
|
|
|
public class SlackService(
|
|
IHttpClientFactory httpClientFactory,
|
|
GlobalSettings globalSettings,
|
|
ILogger<SlackService> logger) : ISlackService
|
|
{
|
|
private readonly HttpClient _httpClient = httpClientFactory.CreateClient(HttpClientName);
|
|
private readonly string _clientId = globalSettings.Slack.ClientId;
|
|
private readonly string _clientSecret = globalSettings.Slack.ClientSecret;
|
|
private readonly string _scopes = globalSettings.Slack.Scopes;
|
|
|
|
public const string HttpClientName = "SlackServiceHttpClient";
|
|
|
|
public async Task<string> GetChannelIdAsync(string token, string channelName)
|
|
{
|
|
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)
|
|
{
|
|
var matchingChannelIds = new List<string>();
|
|
var baseUrl = "https://slack.com/api/conversations.list";
|
|
var nextCursor = string.Empty;
|
|
|
|
do
|
|
{
|
|
var uriBuilder = new UriBuilder(baseUrl);
|
|
var queryParameters = HttpUtility.ParseQueryString(uriBuilder.Query);
|
|
queryParameters["types"] = "public_channel,private_channel";
|
|
queryParameters["limit"] = "1000";
|
|
if (!string.IsNullOrEmpty(nextCursor))
|
|
{
|
|
queryParameters["cursor"] = nextCursor;
|
|
}
|
|
uriBuilder.Query = queryParameters.ToString();
|
|
|
|
var request = new HttpRequestMessage(HttpMethod.Get, uriBuilder.Uri);
|
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
|
|
|
var response = await _httpClient.SendAsync(request);
|
|
var result = await response.Content.ReadFromJsonAsync<SlackChannelListResponse>();
|
|
|
|
if (result != null && result.Ok)
|
|
{
|
|
matchingChannelIds.AddRange(result.Channels
|
|
.Where(channel => channelNames.Contains(channel.Name))
|
|
.Select(channel => channel.Id));
|
|
nextCursor = result.ResponseMetadata.NextCursor;
|
|
}
|
|
else
|
|
{
|
|
nextCursor = string.Empty;
|
|
}
|
|
|
|
} while (!string.IsNullOrEmpty(nextCursor));
|
|
|
|
return matchingChannelIds;
|
|
}
|
|
|
|
public async Task<string> GetDmChannelByEmailAsync(string token, string email)
|
|
{
|
|
var userId = await GetUserIdByEmailAsync(token, email);
|
|
return await OpenDmChannel(token, userId);
|
|
}
|
|
|
|
public async Task SendSlackMessageByChannelIdAsync(string token, string message, string channelId)
|
|
{
|
|
var payload = JsonContent.Create(new { channel = channelId, text = message });
|
|
var request = new HttpRequestMessage(HttpMethod.Post, "https://slack.com/api/chat.postMessage");
|
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
|
request.Content = payload;
|
|
|
|
await _httpClient.SendAsync(request);
|
|
}
|
|
|
|
private async Task<string> GetUserIdByEmailAsync(string token, string email)
|
|
{
|
|
var request = new HttpRequestMessage(HttpMethod.Get, $"https://slack.com/api/users.lookupByEmail?email={email}");
|
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
|
var response = await _httpClient.SendAsync(request);
|
|
var result = await response.Content.ReadFromJsonAsync<SlackUserResponse>();
|
|
|
|
if (result == null)
|
|
{
|
|
logger.LogError("Error retrieving Slack user ID: Unknown error");
|
|
return string.Empty;
|
|
}
|
|
if (!result.Ok)
|
|
{
|
|
logger.LogError("Error retrieving Slack user ID: " + result.Error);
|
|
return string.Empty;
|
|
}
|
|
|
|
return result.User.Id;
|
|
}
|
|
|
|
private async Task<string> OpenDmChannel(string token, string userId)
|
|
{
|
|
var payload = JsonContent.Create(new { users = userId });
|
|
var request = new HttpRequestMessage(HttpMethod.Post, "https://slack.com/api/conversations.open");
|
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
|
request.Content = payload;
|
|
var response = await _httpClient.SendAsync(request);
|
|
var result = await response.Content.ReadFromJsonAsync<SlackDmResponse>();
|
|
|
|
if (result == null)
|
|
{
|
|
logger.LogError("Error opening DM channel: Unknown error");
|
|
return string.Empty;
|
|
}
|
|
if (!result.Ok)
|
|
{
|
|
logger.LogError("Error opening DM channel: " + result.Error);
|
|
return string.Empty;
|
|
}
|
|
|
|
return result.Channel.Id;
|
|
}
|
|
}
|