using DomainResults.Common;
using DataProviders.Collections;
using ExtensionMethods;
using Core.Enumerations;
using WeatherForecast.Models;
using WeatherForecast.Models.Requests;
using WeatherForecast.Services.Abstractions;
namespace WeatherForecast.Services {
  /// 
  /// 
  /// 
  public interface IShopItemService {
    
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    (Guid?, IDomainResult) Post(Guid siteId, string sku, ShopItemRequestModel requestModel);
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    (ShopItemResponseModel?, IDomainResult) Get(Guid siteId, string sku);
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    (ShopItemResponseModel?, IDomainResult) GetSlug(Guid siteId, string slug);
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    (Guid?, IDomainResult) Update(Guid siteId, string sku, ShopItemRequestModel requestData);
    
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    IDomainResult Delete(Guid siteId, string sku);
  }
  /// 
  /// 
  /// 
  public class ShopItemService : PostItemServiceBase, IShopItemService {
    private readonly IShopCatalogDataProvider _shopCatalogDataProvider;
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public ShopItemService(
      ILogger logger,
      IShopCatalogDataProvider shopCatalogDataProvider,
      ICategoryDataProvider categoryDataProvider
    ) : base(logger, categoryDataProvider) {
      _shopCatalogDataProvider = shopCatalogDataProvider;
      
    }
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public (Guid?, IDomainResult) Post(Guid siteId, string sku, ShopItemRequestModel requestData) {
      try {
        var item = requestData.ToDomainObject();
        item.SiteId = siteId;
        item.Sku = sku;
        var (_, getResult) = _shopCatalogDataProvider.Get(siteId, sku);
        if (getResult.IsSuccess)
          return IDomainResult.Failed();
        // TODO: should be recovered from users by jwt
        item.Author = "fdc5aa50-ee68-4bae-a8e6-b8ae2c258f60".ToGuid();
        var (categories, addCategoriesResult) = AddCategoryIfNullOrEmpty(siteId, requestData.Categories);
        if (!addCategoriesResult.IsSuccess || categories == null)
          return (null, addCategoriesResult);
        item.Categories = categories;
        var (id, insertResult) = _shopCatalogDataProvider.Insert(item);
        if (!insertResult.IsSuccess)
          return IDomainResult.Failed();
        return IDomainResult.Success(id);
      }
      catch (Exception ex) {
        _logger.LogError(ex, "Unhandled exception");
        return IDomainResult.Failed(ex.Message);
      }
    }
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public (ShopItemResponseModel?, IDomainResult) Get(Guid siteId, string sku) {
      try {
        var (item, result) = _shopCatalogDataProvider.Get(siteId, sku);
        if (!result.IsSuccess || item == null)
          return (null, result);
        var (categories, getCategoryResult) = _categoryDataProvider.GetMany(siteId, item.Categories);
        if (!getCategoryResult.IsSuccess || categories == null)
          return IDomainResult.Failed();
        return IDomainResult.Success(new ShopItemResponseModel(item, categories));
      }
      catch (Exception ex) {
        _logger.LogError(ex, "Unhandled exception");
        return IDomainResult.Failed(ex.Message);
      }
    }
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public (ShopItemResponseModel?, IDomainResult) GetSlug(Guid siteId, string slug) {
      try {
        var (item, result) = _shopCatalogDataProvider.GetBySlug(siteId, slug);
        if (!result.IsSuccess || item == null)
          return (null, result);
        var locale = item.L10n.SingleOrDefault(x => x.Slug == slug)?.Locale ?? Locales.Us;
        var (categories, getCategoryResult) = _categoryDataProvider.GetMany(siteId, item.Categories);
        if (!getCategoryResult.IsSuccess || categories == null)
          return IDomainResult.Failed();
        return IDomainResult.Success(new ShopItemResponseModel(item, categories, locale));
      }
      catch (Exception ex) {
        _logger.LogError(ex, "Unhandled exception");
        return IDomainResult.Failed(ex.Message);
      }
    }
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public (Guid?, IDomainResult) Update(Guid siteId, string sku, ShopItemRequestModel requestData) {
      try {
        var (item, getResult) = _shopCatalogDataProvider.Get(siteId, sku);
        if (!getResult.IsSuccess || item == null)
          return (null, getResult);
        // construct domain object from model
        var newItem = requestData.ToDomainObject();
        newItem.Id = item.Id;
        newItem.SiteId = siteId;
        newItem.Sku = sku;
        newItem.Created = item.Created;
        newItem.Author = item.Author;
        var (categories, addCategoriesResult) = AddCategoryIfNullOrEmpty(siteId, requestData.Categories);
        if (!addCategoriesResult.IsSuccess || categories == null)
          return (null, addCategoriesResult);
        item.Categories = categories;
        if (!item.Equals(newItem)) {
          var (id, updateResult) = _shopCatalogDataProvider.Update(newItem);
          if (!updateResult.IsSuccess || id == null)
            return (null, updateResult);
        }
        return IDomainResult.Success(item.Id);
      }
      catch (Exception ex) {
        _logger.LogError(ex, "Unhandled exception");
        return IDomainResult.Failed(ex.Message);
      }
    }
    /// 
    /// 
    /// 
    /// 
    /// 
    /// 
    public IDomainResult Delete(Guid siteId, string sku) {
      try {
        var (item, getResult) = _shopCatalogDataProvider.Get(siteId, sku);
        if (!getResult.IsSuccess || item == null)
          return getResult;
        var result = _shopCatalogDataProvider.Delete(item.Id);
        if (!result.IsSuccess)
          return result;
        return IDomainResult.Success();
      }
      catch (Exception ex) {
        _logger.LogError(ex, "Unhandled exception");
        return IDomainResult.Failed(ex.Message);
      }
    }
  }
}