/* * * (c) Copyright Ascensio System Limited 2010-2018 * * This program is freeware. You can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) version 3 as published by the Free Software Foundation (https://www.gnu.org/copyleft/gpl.html). * In accordance with Section 7(a) of the GNU GPL its Section 15 shall be amended to the effect that * Ascensio System SIA expressly excludes the warranty of non-infringement of any third-party rights. * * THIS PROGRAM IS DISTRIBUTED WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF MERCHANTABILITY OR * FITNESS FOR A PARTICULAR PURPOSE. For more details, see GNU GPL at https://www.gnu.org/copyleft/gpl.html * * You can contact Ascensio System SIA by email at sales@onlyoffice.com * * The interactive user interfaces in modified source and object code versions of ONLYOFFICE must display * Appropriate Legal Notices, as required under Section 5 of the GNU GPL version 3. * * Pursuant to Section 7 § 3(b) of the GNU GPL you must retain the original ONLYOFFICE logo which contains * relevant author attributions when distributing the software. If the display of the logo in its graphic * form is not reasonably feasible for technical reasons, you must include the words "Powered by ONLYOFFICE" * in every copy of the program you distribute. * Pursuant to Section 7 § 3(e) we decline to grant you any rights under trademark law for use of our trademarks. * */ namespace ASC.Core.Billing; [Singletone] public class TariffServiceStorage { public ICache Cache { get; } internal readonly ICacheNotify _notify; public TariffServiceStorage(ICacheNotify notify, ICache cache) { Cache = cache; _notify = notify; _notify.Subscribe((i) => { Cache.Remove(TariffService.GetTariffCacheKey(i.TenantId)); Cache.Remove(TariffService.GetBillingUrlCacheKey(i.TenantId)); Cache.Remove(TariffService.GetBillingPaymentCacheKey(i.TenantId)); // clear all payments }, CacheNotifyAction.Remove); //TODO: Change code of WCF -> not supported in .NET standard/.Net Core /*try { var section = (ClientSection)ConfigurationManager.GetSection("system.serviceModel/client"); if (section != null) { billingConfigured = section.Endpoints.Cast() .Any(e => e.Contract == typeof(IService).FullName); } } catch (Exception err) { log.Error(err); }*/ } } [Scope] class ConfigureTariffService : IConfigureNamedOptions { public ConfigureTariffService( IOptionsSnapshot quotaService, IOptionsSnapshot tenantService, CoreBaseSettings coreBaseSettings, CoreSettings coreSettings, IConfiguration configuration, DbContextManager coreDbContextManager, TariffServiceStorage tariffServiceStorage, IOptionsMonitor iLog) { this._quotaService = quotaService; this._tenantService = tenantService; _coreBaseSettings = coreBaseSettings; _coreSettings = coreSettings; _configuration = configuration; _coreDbContextManager = coreDbContextManager; _tariffServiceStorage = tariffServiceStorage; _logger = iLog; } private readonly IOptionsSnapshot _quotaService; private readonly IOptionsSnapshot _tenantService; private readonly CoreBaseSettings _coreBaseSettings; private readonly CoreSettings _coreSettings; private readonly IConfiguration _configuration; private readonly DbContextManager _coreDbContextManager; private readonly TariffServiceStorage _tariffServiceStorage; private readonly IOptionsMonitor _logger; public void Configure(string name, TariffService options) { Configure(options); options._quotaService = _quotaService.Get(name); options._tenantService = _tenantService.Get(name); options._lazyCoreDbContext = new Lazy(() => _coreDbContextManager.Get(name)); } public void Configure(TariffService options) { options._logger = _logger.CurrentValue; options._coreSettings = _coreSettings; options._configuration = _configuration; options._tariffServiceStorage = _tariffServiceStorage; options._options = _logger; options._coreBaseSettings = _coreBaseSettings; options._test = _configuration["core:payment:test"] == "true"; int.TryParse(_configuration["core:payment:delay"], out var paymentDelay); options._paymentDelay = paymentDelay; options._cache = _tariffServiceStorage.Cache; options._notify = _tariffServiceStorage._notify; options._quotaService = _quotaService.Value; options._tenantService = _tenantService.Value; options._lazyCoreDbContext = new Lazy(() => _coreDbContextManager.Value); } } public class TariffService : ITariffService { private const int _defaultTrialPeriod = 30; private static readonly TimeSpan _defaultCacheExpiration = TimeSpan.FromMinutes(5); private static readonly TimeSpan _standaloneCacheExpiration = TimeSpan.FromMinutes(15); internal ICache _cache; internal ICacheNotify _notify; internal ILog _logger; internal IQuotaService _quotaService; internal ITenantService _tenantService; internal bool _test; internal int _paymentDelay; public TimeSpan CacheExpiration { get; set; } internal CoreBaseSettings _coreBaseSettings; internal CoreSettings _coreSettings; internal IConfiguration _configuration; internal CoreDbContext CoreDbContext => _lazyCoreDbContext.Value; internal Lazy _lazyCoreDbContext; internal TariffServiceStorage _tariffServiceStorage; internal IOptionsMonitor _options; public BillingClient BillingClient { get; } public readonly int ActiveUsersMin; public readonly int ActiveUsersMax; public TariffService() { CacheExpiration = _defaultCacheExpiration; } public TariffService( IQuotaService quotaService, ITenantService tenantService, CoreBaseSettings coreBaseSettings, CoreSettings coreSettings, IConfiguration configuration, DbContextManager coreDbContextManager, TariffServiceStorage tariffServiceStorage, IOptionsMonitor options, Users.Constants constants, BillingClient billingClient) : this() { _logger = options.CurrentValue; _quotaService = quotaService; _tenantService = tenantService; _coreSettings = coreSettings; _configuration = configuration; _tariffServiceStorage = tariffServiceStorage; _options = options; BillingClient = billingClient; _coreBaseSettings = coreBaseSettings; _test = configuration["core:payment:test"] == "true"; int.TryParse(configuration["core:payment:delay"], out var paymentDelay); _paymentDelay = paymentDelay; _cache = _tariffServiceStorage.Cache; _notify = _tariffServiceStorage._notify; _lazyCoreDbContext = new Lazy(() => coreDbContextManager.Value); var range = (_configuration["core.payment-user-range"] ?? "").Split('-'); if (!int.TryParse(range[0], out ActiveUsersMin)) { ActiveUsersMin = 0; } if (range.Length < 2 || !int.TryParse(range[1], out ActiveUsersMax)) { ActiveUsersMax = constants.MaxEveryoneCount; } } public Tariff GetTariff(int tenantId, bool withRequestToPaymentSystem = true) { //single tariff for all portals if (_coreBaseSettings.Standalone) { tenantId = -1; } var key = GetTariffCacheKey(tenantId); var tariff = _cache.Get(key); if (tariff == null) { tariff = GetBillingInfo(tenantId); tariff = CalculateTariff(tenantId, tariff); _cache.Insert(key, tariff, DateTime.UtcNow.Add(GetCacheExpiration())); if (BillingClient.Configured && withRequestToPaymentSystem) { Task.Run(() => { try { var client = GetBillingClient(); var lastPayment = client.GetLastPayment(GetPortalId(tenantId)); var quota = _quotaService.GetTenantQuotas().SingleOrDefault(q => q.AvangateId == lastPayment.ProductId); if (quota == null) { throw new InvalidOperationException($"Quota with id {lastPayment.ProductId} not found for portal {GetPortalId(tenantId)}."); } var asynctariff = Tariff.CreateDefault(); asynctariff.QuotaId = quota.Tenant; asynctariff.Autorenewal = lastPayment.Autorenewal; asynctariff.DueDate = 9999 <= lastPayment.EndDate.Year ? DateTime.MaxValue : lastPayment.EndDate; if (quota.ActiveUsers == -1 && lastPayment.Quantity < ActiveUsersMin) { throw new BillingException(string.Format("The portal {0} is paid for {1} users", tenantId, lastPayment.Quantity)); } asynctariff.Quantity = lastPayment.Quantity; if (SaveBillingInfo(tenantId, asynctariff, false)) { asynctariff = CalculateTariff(tenantId, asynctariff); ClearCache(tenantId); _cache.Insert(key, asynctariff, DateTime.UtcNow.Add(GetCacheExpiration())); } } catch (BillingNotFoundException) { } catch (Exception error) { LogError(error, tenantId.ToString()); } }); } } return tariff; } public void SetTariff(int tenantId, Tariff tariff) { ArgumentNullException.ThrowIfNull(tariff); var q = _quotaService.GetTenantQuota(tariff.QuotaId); if (q == null) { return; } SaveBillingInfo(tenantId, tariff); if (q.Trial) { // reset trial date var tenant = _tenantService.GetTenant(tenantId); if (tenant != null) { tenant.VersionChanged = DateTime.UtcNow; _tenantService.SaveTenant(_coreSettings, tenant); } } ClearCache(tenantId); } internal static string GetTariffCacheKey(int tenantId) { return string.Format("{0}:{1}", tenantId, "tariff"); } internal static string GetBillingUrlCacheKey(int tenantId) { return string.Format("{0}:{1}", tenantId, "billing:urls"); } internal static string GetBillingPaymentCacheKey(int tenantId) { return string.Format("{0}:{1}", tenantId, "billing:payments"); } public void ClearCache(int tenantId) { _notify.Publish(new TariffCacheItem { TenantId = tenantId }, ASC.Common.Caching.CacheNotifyAction.Remove); } public IEnumerable GetPayments(int tenantId) { var key = GetBillingPaymentCacheKey(tenantId); var payments = _cache.Get>(key); if (payments == null) { payments = new List(); if (BillingClient.Configured) { try { var quotas = _quotaService.GetTenantQuotas(); var client = GetBillingClient(); foreach (var pi in client.GetPayments(GetPortalId(tenantId))) { var quota = quotas.SingleOrDefault(q => q.AvangateId == pi.ProductRef); if (quota != null) { pi.QuotaId = quota.Tenant; } payments.Add(pi); } } catch (Exception error) { LogError(error, tenantId.ToString()); } } _cache.Insert(key, payments, DateTime.UtcNow.Add(TimeSpan.FromMinutes(10))); } return payments; } public Uri GetShoppingUri(int? tenant, int quotaId, string affiliateId, string currency = null, string language = null, string customerId = null, string quantity = null) { var quota = _quotaService.GetTenantQuota(quotaId); if (quota == null) { return null; } var key = tenant.HasValue ? GetBillingUrlCacheKey(tenant.Value) : string.Format($"notenant{(!string.IsNullOrEmpty(affiliateId) ? "_" + affiliateId : "")}"); key += quota.Visible ? "" : "0"; if (!(_cache.Get>>(key) is IDictionary> urls)) { urls = new Dictionary>(); if (BillingClient.Configured) { try { var products = _quotaService.GetTenantQuotas() .Where(q => !string.IsNullOrEmpty(q.AvangateId) && q.Visible == quota.Visible) .Select(q => q.AvangateId) .ToArray(); var client = GetBillingClient(); urls = client.GetPaymentUrls( tenant.HasValue ? GetPortalId(tenant.Value) : null, products, tenant.HasValue ? GetAffiliateId(tenant.Value) : affiliateId, tenant.HasValue ? GetCampaign(tenant.Value) : null, !string.IsNullOrEmpty(currency) ? "__Currency__" : null, !string.IsNullOrEmpty(language) ? "__Language__" : null, !string.IsNullOrEmpty(customerId) ? "__CustomerID__" : null, !string.IsNullOrEmpty(quantity) ? "__Quantity__" : null ); } catch (Exception error) { _logger.Error(error); } } _cache.Insert(key, urls, DateTime.UtcNow.Add(TimeSpan.FromMinutes(10))); } ResetCacheExpiration(); if (!string.IsNullOrEmpty(quota.AvangateId) && urls.TryGetValue(quota.AvangateId, out var tuple)) { var result = tuple.Item2; if (result == null) { result = tuple.Item1; } else { var tariff = tenant.HasValue ? GetTariff(tenant.Value) : null; if (tariff == null || tariff.QuotaId == quotaId || tariff.State >= TariffState.Delay) { result = tuple.Item1; } } if (result == null) { return null; } result = new Uri(result.ToString() .Replace("__Currency__", HttpUtility.UrlEncode(currency ?? "")) .Replace("__Language__", HttpUtility.UrlEncode((language ?? "").ToLower())) .Replace("__CustomerID__", HttpUtility.UrlEncode(customerId ?? "")) .Replace("__Quantity__", HttpUtility.UrlEncode(quantity ?? ""))); return result; } return null; } public IDictionary> GetProductPriceInfo(params string[] productIds) { ArgumentNullException.ThrowIfNull(productIds); try { var key = "biling-prices" + string.Join(",", productIds); var result = _cache.Get>>(key); if (result == null) { var client = GetBillingClient(); result = client.GetProductPriceInfo(productIds); _cache.Insert(key, result, DateTime.Now.AddHours(1)); } return result; } catch (Exception error) { LogError(error); return productIds .Select(p => new { ProductId = p, Prices = new Dictionary() }) .ToDictionary(e => e.ProductId, e => e.Prices); } } public string GetButton(int tariffId, string partnerId) { return CoreDbContext.Buttons .Where(r => r.TariffId == tariffId && r.PartnerId == partnerId) .Select(r => r.ButtonUrl) .SingleOrDefault(); } public void SaveButton(int tariffId, string partnerId, string buttonUrl) { var efButton = new DbButton() { TariffId = tariffId, PartnerId = partnerId, ButtonUrl = buttonUrl }; CoreDbContext.AddOrUpdate(r => r.Buttons, efButton); CoreDbContext.SaveChanges(); } private Tariff GetBillingInfo(int tenant) { var r = CoreDbContext.Tariffs .Where(r => r.Tenant == tenant) .OrderByDescending(r => r.Id) .FirstOrDefault(); if (r == null) { return Tariff.CreateDefault(); } var tariff = Tariff.CreateDefault(); tariff.QuotaId = r.Tariff; tariff.DueDate = r.Stamp.Year < 9999 ? r.Stamp : DateTime.MaxValue; tariff.Quantity = r.Quantity; return tariff; } private bool SaveBillingInfo(int tenant, Tariff tariffInfo, bool renewal = true) { var inserted = false; var currentTariff = GetBillingInfo(tenant); if (!tariffInfo.EqualsByParams(currentTariff)) { using var tx = CoreDbContext.Database.BeginTransaction(); // last record is not the same var any = CoreDbContext.Tariffs .Any(r => r.Tenant == tenant && r.Tariff == tariffInfo.QuotaId && r.Stamp == tariffInfo.DueDate && r.Quantity == tariffInfo.Quantity); if (tariffInfo.DueDate == DateTime.MaxValue || renewal || any) { var efTariff = new DbTariff { Tenant = tenant, Tariff = tariffInfo.QuotaId, Stamp = tariffInfo.DueDate, Quantity = tariffInfo.Quantity, CreateOn = DateTime.UtcNow }; CoreDbContext.Tariffs.Add(efTariff); CoreDbContext.SaveChanges(); _cache.Remove(GetTariffCacheKey(tenant)); inserted = true; } tx.Commit(); } if (inserted) { var t = _tenantService.GetTenant(tenant); if (t != null) { // update tenant.LastModified to flush cache in documents _tenantService.SaveTenant(_coreSettings, t); } } return inserted; } public void DeleteDefaultBillingInfo() { const int tenant = Tenant.DefaultTenant; var tariffs = CoreDbContext.Tariffs.Where(r => r.Tenant == tenant).ToList(); foreach (var t in tariffs) { t.Tenant = -2; } CoreDbContext.SaveChanges(); ClearCache(tenant); } private Tariff CalculateTariff(int tenantId, Tariff tariff) { tariff.State = TariffState.Paid; var q = _quotaService.GetTenantQuota(tariff.QuotaId); if (q == null || q.GetFeature("old")) { tariff.QuotaId = Tenant.DefaultTenant; q = _quotaService.GetTenantQuota(tariff.QuotaId); } var delay = 0; if (q != null && q.Trial) { tariff.State = TariffState.Trial; if (tariff.DueDate == DateTime.MinValue || tariff.DueDate == DateTime.MaxValue) { var tenant = _tenantService.GetTenant(tenantId); if (tenant != null) { var fromDate = tenant.CreationDateTime < tenant.VersionChanged ? tenant.VersionChanged : tenant.CreationDateTime; var trialPeriod = GetPeriod("TrialPeriod", _defaultTrialPeriod); if (fromDate == DateTime.MinValue) { fromDate = DateTime.UtcNow.Date; } tariff.DueDate = trialPeriod != default ? fromDate.Date.AddDays(trialPeriod) : DateTime.MaxValue; } else { tariff.DueDate = DateTime.MaxValue; } } } else { delay = _paymentDelay; } if (tariff.DueDate != DateTime.MinValue && tariff.DueDate.Date < DateTime.Today && delay > 0) { tariff.State = TariffState.Delay; tariff.DelayDueDate = tariff.DueDate.Date.AddDays(delay); } if (tariff.DueDate == DateTime.MinValue || tariff.DueDate != DateTime.MaxValue && tariff.DueDate.Date.AddDays(delay) < DateTime.Today) { tariff.State = TariffState.NotPaid; if (_coreBaseSettings.Standalone) { if (q != null) { var defaultQuota = _quotaService.GetTenantQuota(Tenant.DefaultTenant); defaultQuota.Name = "overdue"; defaultQuota.Features = q.Features; defaultQuota.Support = false; _quotaService.SaveTenantQuota(defaultQuota); } var unlimTariff = Tariff.CreateDefault(); unlimTariff.LicenseDate = tariff.DueDate; tariff = unlimTariff; } } tariff.Prolongable = tariff.DueDate == DateTime.MinValue || tariff.DueDate == DateTime.MaxValue || tariff.State == TariffState.Trial || new DateTime(tariff.DueDate.Year, tariff.DueDate.Month, 1) <= new DateTime(DateTime.UtcNow.Year, DateTime.UtcNow.Month, 1).AddMonths(1); return tariff; } private int GetPeriod(string key, int defaultValue) { var settings = _tenantService.GetTenantSettings(Tenant.DefaultTenant, key); return settings != null ? Convert.ToInt32(Encoding.UTF8.GetString(settings)) : defaultValue; } private BillingClient GetBillingClient() { try { return new BillingClient(_test, _configuration); } catch (InvalidOperationException ioe) { throw new BillingNotConfiguredException(ioe.Message, ioe); } catch (ReflectionTypeLoadException rtle) { _logger.ErrorFormat("{0}{1}LoaderExceptions: {2}", rtle, Environment.NewLine, string.Join(Environment.NewLine, rtle.LoaderExceptions.Select(e => e.ToString()))); throw; } } private string GetPortalId(int tenant) { return _coreSettings.GetKey(tenant); } private string GetAffiliateId(int tenant) { return _coreSettings.GetAffiliateId(tenant); } private string GetCampaign(int tenant) { return _coreSettings.GetCampaign(tenant); } private TimeSpan GetCacheExpiration() { if (_coreBaseSettings.Standalone && CacheExpiration < _standaloneCacheExpiration) { CacheExpiration = CacheExpiration.Add(TimeSpan.FromSeconds(30)); } return CacheExpiration; } private void ResetCacheExpiration() { if (_coreBaseSettings.Standalone) { CacheExpiration = _defaultCacheExpiration; } } private void LogError(Exception error, string tenantId = null) { if (error is BillingNotFoundException) { _logger.DebugFormat("Payment tenant {0} not found: {1}", tenantId, error.Message); } else if (error is BillingNotConfiguredException) { _logger.DebugFormat("Billing tenant {0} not configured: {1}", tenantId, error.Message); } else { if (_logger.IsDebugEnabled) { _logger.Error("Billing tenant " + tenantId); _logger.Error(error); } else { _logger.ErrorFormat("Billing tenant {0}: {1}", tenantId, error.Message); } } } }