mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
179 lines
7.2 KiB
C#
179 lines
7.2 KiB
C#
using Bit.Core.Repositories;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.Extensions.Logging;
|
|
using Microsoft.Extensions.Options;
|
|
using Newtonsoft.Json;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Net.Http;
|
|
using System.Reflection;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using System.ComponentModel.DataAnnotations;
|
|
|
|
using Bit.Core.Settings;
|
|
using Bit.Core.Utilities;
|
|
|
|
namespace Bit.Billing.Controllers
|
|
{
|
|
[Route("freshdesk")]
|
|
public class FreshdeskController : Controller
|
|
{
|
|
private readonly BillingSettings _billingSettings;
|
|
private readonly IUserRepository _userRepository;
|
|
private readonly IOrganizationRepository _organizationRepository;
|
|
private readonly IOrganizationUserRepository _organizationUserRepository;
|
|
private readonly ILogger<AppleController> _logger;
|
|
private readonly GlobalSettings _globalSettings;
|
|
private readonly HttpClient _httpClient = new HttpClient();
|
|
private readonly string _freshdeskAuthkey;
|
|
|
|
public FreshdeskController(
|
|
IUserRepository userRepository,
|
|
IOrganizationRepository organizationRepository,
|
|
IOrganizationUserRepository organizationUserRepository,
|
|
IOptions<BillingSettings> billingSettings,
|
|
ILogger<AppleController> logger,
|
|
GlobalSettings globalSettings)
|
|
{
|
|
_billingSettings = billingSettings?.Value;
|
|
_userRepository = userRepository;
|
|
_organizationRepository = organizationRepository;
|
|
_organizationUserRepository = organizationUserRepository;
|
|
_logger = logger;
|
|
_globalSettings = globalSettings;
|
|
_freshdeskAuthkey = Convert.ToBase64String(
|
|
Encoding.UTF8.GetBytes($"{_billingSettings.FreshdeskApiKey}:X"));
|
|
}
|
|
|
|
[HttpPost("webhook")]
|
|
public async Task<IActionResult> PostWebhook()
|
|
{
|
|
if (HttpContext?.Request?.Query == null)
|
|
{
|
|
return new BadRequestResult();
|
|
}
|
|
|
|
var key = HttpContext.Request.Query.ContainsKey("key") ?
|
|
HttpContext.Request.Query["key"].ToString() : null;
|
|
if (!CoreHelpers.FixedTimeEquals(key, _billingSettings.FreshdeskWebhookKey))
|
|
{
|
|
return new BadRequestResult();
|
|
}
|
|
|
|
string body = null;
|
|
using (var reader = new StreamReader(HttpContext.Request.Body, Encoding.UTF8))
|
|
{
|
|
body = await reader.ReadToEndAsync();
|
|
}
|
|
|
|
if (string.IsNullOrWhiteSpace(body))
|
|
{
|
|
return new BadRequestResult();
|
|
}
|
|
|
|
try
|
|
{
|
|
dynamic data = JsonConvert.DeserializeObject(body);
|
|
string ticketId = data.ticket_id;
|
|
string ticketContactEmail = data.ticket_contact_email;
|
|
string ticketTags = data.ticket_tags;
|
|
if (string.IsNullOrWhiteSpace(ticketId) || string.IsNullOrWhiteSpace(ticketContactEmail))
|
|
{
|
|
return new BadRequestResult();
|
|
}
|
|
|
|
var updateBody = new Dictionary<string, object>();
|
|
var note = string.Empty;
|
|
var user = await _userRepository.GetByEmailAsync(ticketContactEmail);
|
|
if (user != null)
|
|
{
|
|
note += $"<li>User, {user.Email}: {_globalSettings.BaseServiceUri.Admin}/users/edit/{user.Id}</li>";
|
|
var tags = new HashSet<string>();
|
|
if (user.Premium)
|
|
{
|
|
tags.Add("Premium");
|
|
}
|
|
var orgs = await _organizationRepository.GetManyByUserIdAsync(user.Id);
|
|
foreach (var org in orgs)
|
|
{
|
|
note += $"<li>Org, {org.Name}: " +
|
|
$"{_globalSettings.BaseServiceUri.Admin}/organizations/edit/{org.Id}</li>";
|
|
var planName = GetAttribute<DisplayAttribute>(org.PlanType).Name.Split(" ").FirstOrDefault();
|
|
if (!string.IsNullOrWhiteSpace(planName))
|
|
{
|
|
tags.Add(string.Format("Org: {0}", planName));
|
|
}
|
|
}
|
|
if (tags.Any())
|
|
{
|
|
var tagsToUpdate = tags.ToList();
|
|
if (!string.IsNullOrWhiteSpace(ticketTags))
|
|
{
|
|
var splitTicketTags = ticketTags.Split(',');
|
|
for (var i = 0; i < splitTicketTags.Length; i++)
|
|
{
|
|
tagsToUpdate.Insert(i, splitTicketTags[i]);
|
|
}
|
|
}
|
|
updateBody.Add("tags", tagsToUpdate);
|
|
}
|
|
var updateRequest = new HttpRequestMessage(HttpMethod.Put,
|
|
string.Format("https://bitwarden.freshdesk.com/api/v2/tickets/{0}", ticketId));
|
|
updateRequest.Content = new StringContent(JsonConvert.SerializeObject(updateBody),
|
|
Encoding.UTF8, "application/json");
|
|
await CallFreshdeskApiAsync(updateRequest);
|
|
|
|
|
|
var noteBody = new Dictionary<string, object>
|
|
{
|
|
{ "body", $"<ul>{note}</ul>" },
|
|
{ "private", true }
|
|
};
|
|
var noteRequest = new HttpRequestMessage(HttpMethod.Post,
|
|
string.Format("https://bitwarden.freshdesk.com/api/v2/tickets/{0}/notes", ticketId));
|
|
noteRequest.Content = new StringContent(JsonConvert.SerializeObject(noteBody),
|
|
Encoding.UTF8, "application/json");
|
|
await CallFreshdeskApiAsync(noteRequest);
|
|
}
|
|
|
|
return new OkResult();
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
_logger.LogError(e, "Error processing freshdesk webhook.");
|
|
return new BadRequestResult();
|
|
}
|
|
}
|
|
|
|
private async Task<HttpResponseMessage> CallFreshdeskApiAsync(HttpRequestMessage request, int retriedCount = 0)
|
|
{
|
|
try
|
|
{
|
|
request.Headers.Add("Authorization", _freshdeskAuthkey);
|
|
var response = await _httpClient.SendAsync(request);
|
|
if (response.StatusCode != System.Net.HttpStatusCode.TooManyRequests || retriedCount > 3)
|
|
{
|
|
return response;
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
if (retriedCount > 3)
|
|
{
|
|
throw;
|
|
}
|
|
}
|
|
await Task.Delay(30000 * (retriedCount + 1));
|
|
return await CallFreshdeskApiAsync(request, retriedCount++);
|
|
}
|
|
|
|
private TAttribute GetAttribute<TAttribute>(Enum enumValue) where TAttribute : Attribute
|
|
{
|
|
return enumValue.GetType().GetMember(enumValue.ToString()).First().GetCustomAttribute<TAttribute>();
|
|
}
|
|
}
|
|
}
|