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 _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, ILogger 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 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(); var note = string.Empty; var user = await _userRepository.GetByEmailAsync(ticketContactEmail); if (user != null) { note += $"
  • User, {user.Email}: {_globalSettings.BaseServiceUri.Admin}/users/edit/{user.Id}
  • "; var tags = new HashSet(); if (user.Premium) { tags.Add("Premium"); } var orgs = await _organizationRepository.GetManyByUserIdAsync(user.Id); foreach (var org in orgs) { note += $"
  • Org, {org.Name}: " + $"{_globalSettings.BaseServiceUri.Admin}/organizations/edit/{org.Id}
  • "; var planName = GetAttribute(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 { { "body", $"
      {note}
    " }, { "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 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(Enum enumValue) where TAttribute : Attribute { return enumValue.GetType().GetMember(enumValue.ToString()).First().GetCustomAttribute(); } } }