Merge branch 'main' of ssh://git.ma-al.com:8822/goc_daniel/b2b into orders
This commit is contained in:
@@ -21,6 +21,7 @@ import (
|
||||
"github.com/golang-jwt/jwt/v5"
|
||||
"golang.org/x/crypto/bcrypt"
|
||||
"gorm.io/gorm"
|
||||
"gorm.io/gorm/clause"
|
||||
)
|
||||
|
||||
// JWTClaims represents the JWT claims
|
||||
@@ -436,7 +437,7 @@ func (s *AuthService) RevokeAllRefreshTokens(userID uint) {
|
||||
// GetUserByID retrieves a user by ID
|
||||
func (s *AuthService) GetUserByID(userID uint) (*model.Customer, error) {
|
||||
var user model.Customer
|
||||
if err := s.db.Preload("Role.Permissions").First(&user, userID).Error; err != nil {
|
||||
if err := s.db.Preload("Role.Permissions").Preload(clause.Associations).First(&user, userID).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil, responseErrors.ErrUserNotFound
|
||||
}
|
||||
|
||||
@@ -6,7 +6,7 @@ import (
|
||||
|
||||
"git.ma-al.com/goc_daniel/b2b/app/config"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/model"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/repos/productDescriptionRepo"
|
||||
searchrepo "git.ma-al.com/goc_daniel/b2b/app/repos/searchRepo"
|
||||
constdata "git.ma-al.com/goc_daniel/b2b/app/utils/const_data"
|
||||
"github.com/meilisearch/meilisearch-go"
|
||||
)
|
||||
@@ -20,8 +20,8 @@ type MeiliIndexSettings struct {
|
||||
}
|
||||
|
||||
type MeiliService struct {
|
||||
productDescriptionRepo productDescriptionRepo.UIProductDescriptionRepo
|
||||
meiliClient meilisearch.ServiceManager
|
||||
searchRepo searchrepo.UISearchRepo
|
||||
meiliClient meilisearch.ServiceManager
|
||||
}
|
||||
|
||||
func New() *MeiliService {
|
||||
@@ -32,8 +32,8 @@ func New() *MeiliService {
|
||||
)
|
||||
|
||||
return &MeiliService{
|
||||
meiliClient: client,
|
||||
productDescriptionRepo: productDescriptionRepo.New(),
|
||||
meiliClient: client,
|
||||
searchRepo: searchrepo.New(),
|
||||
}
|
||||
}
|
||||
|
||||
@@ -50,7 +50,7 @@ func (s *MeiliService) CreateIndex(id_lang uint) error {
|
||||
|
||||
for {
|
||||
// Get batch of products from repo (includes scanning)
|
||||
products, err := s.productDescriptionRepo.GetMeiliProducts(id_lang, offset, batchSize)
|
||||
products, err := s.searchRepo.GetMeiliProducts(id_lang, offset, batchSize)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to get products batch at offset %d: %w", offset, err)
|
||||
}
|
||||
|
||||
@@ -3,31 +3,45 @@ package menuService
|
||||
import (
|
||||
"slices"
|
||||
"sort"
|
||||
"strconv"
|
||||
|
||||
"git.ma-al.com/goc_daniel/b2b/app/model"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/repos/categoriesRepo"
|
||||
categoryrepo "git.ma-al.com/goc_daniel/b2b/app/repos/categoryRepo"
|
||||
routesRepo "git.ma-al.com/goc_daniel/b2b/app/repos/routesRepo"
|
||||
constdata "git.ma-al.com/goc_daniel/b2b/app/utils/const_data"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/i18n"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/responseErrors"
|
||||
)
|
||||
|
||||
type MenuService struct {
|
||||
categoriesRepo categoriesRepo.UICategoriesRepo
|
||||
routesRepo routesRepo.UIRoutesRepo
|
||||
categoryRepo categoryrepo.UICategoryRepo
|
||||
routesRepo routesRepo.UIRoutesRepo
|
||||
}
|
||||
|
||||
func New() *MenuService {
|
||||
return &MenuService{
|
||||
categoriesRepo: categoriesRepo.New(),
|
||||
routesRepo: routesRepo.New(),
|
||||
categoryRepo: categoryrepo.New(),
|
||||
routesRepo: routesRepo.New(),
|
||||
}
|
||||
}
|
||||
|
||||
func (s *MenuService) GetCategoryTree(root_category_id uint, id_lang uint) (*model.Category, error) {
|
||||
all_categories, err := s.categoriesRepo.GetAllCategories(id_lang)
|
||||
all_categories, err := s.categoryRepo.RetrieveMenuCategories(id_lang)
|
||||
if err != nil {
|
||||
return &model.Category{}, err
|
||||
}
|
||||
|
||||
// remove blacklisted categories
|
||||
// to do so, we detach them from the main tree
|
||||
for i := 0; i < len(all_categories); i++ {
|
||||
if slices.Contains(constdata.CATEGORY_BLACKLIST, all_categories[i].CategoryID) {
|
||||
all_categories[i].ParentID = all_categories[i].CategoryID
|
||||
}
|
||||
}
|
||||
|
||||
iso_code := all_categories[0].IsoCode
|
||||
s.appendAdditional(&all_categories, id_lang, iso_code)
|
||||
|
||||
// find the root
|
||||
root_index := 0
|
||||
root_found := false
|
||||
@@ -98,7 +112,7 @@ func (s *MenuService) scannedToNormalCategory(scanned model.ScannedCategory) mod
|
||||
normal.CategoryID = scanned.CategoryID
|
||||
normal.Label = scanned.Name
|
||||
// normal.Active = scanned.Active == 1
|
||||
normal.Params = model.CategoryParams{CategoryID: normal.CategoryID, LinkRewrite: scanned.LinkRewrite, Locale: scanned.IsoCode}
|
||||
normal.Params = model.CategoryParams{CategoryID: normal.CategoryID, LinkRewrite: scanned.LinkRewrite, Locale: scanned.IsoCode, Filter: scanned.Filter}
|
||||
normal.Children = []model.Category{}
|
||||
return normal
|
||||
}
|
||||
@@ -114,11 +128,14 @@ func (a ByPosition) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||
func (a ByPosition) Less(i, j int) bool { return a[i].Position < a[j].Position }
|
||||
|
||||
func (s *MenuService) GetBreadcrumb(root_category_id uint, start_category_id uint, id_lang uint) ([]model.CategoryInBreadcrumb, error) {
|
||||
all_categories, err := s.categoriesRepo.GetAllCategories(id_lang)
|
||||
all_categories, err := s.categoryRepo.RetrieveMenuCategories(id_lang)
|
||||
if err != nil {
|
||||
return []model.CategoryInBreadcrumb{}, err
|
||||
}
|
||||
|
||||
iso_code := all_categories[0].IsoCode
|
||||
s.appendAdditional(&all_categories, id_lang, iso_code)
|
||||
|
||||
breadcrumb := []model.CategoryInBreadcrumb{}
|
||||
|
||||
start_index := 0
|
||||
@@ -211,3 +228,24 @@ func (s *MenuService) GetTopMenu(languageId uint, roleId uint) ([]*model.B2BTopM
|
||||
|
||||
return roots, nil
|
||||
}
|
||||
|
||||
func (s *MenuService) appendAdditional(all_categories *[]model.ScannedCategory, id_lang uint, iso_code string) {
|
||||
for i := 0; i < len(*all_categories); i++ {
|
||||
(*all_categories)[i].Filter = "category_id_in=" + strconv.Itoa(int((*all_categories)[i].CategoryID))
|
||||
}
|
||||
|
||||
var additional model.ScannedCategory
|
||||
additional.CategoryID = 10001
|
||||
additional.Name = "New Products"
|
||||
additional.Active = 1
|
||||
additional.Position = 10
|
||||
additional.ParentID = 2
|
||||
additional.IsRoot = 0
|
||||
additional.LinkRewrite = i18n.T___(id_lang, "category.new_products")
|
||||
additional.IsoCode = iso_code
|
||||
|
||||
additional.Visited = false
|
||||
additional.Filter = "is_new_in=true"
|
||||
|
||||
*all_categories = append(*all_categories, additional)
|
||||
}
|
||||
|
||||
@@ -2,6 +2,7 @@ package productService
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
|
||||
"git.ma-al.com/goc_daniel/b2b/app/model"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/repos/productsRepo"
|
||||
@@ -9,6 +10,7 @@ import (
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/query/filters"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/query/find"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/responseErrors"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/view"
|
||||
)
|
||||
|
||||
type ProductService struct {
|
||||
@@ -21,17 +23,108 @@ func New() *ProductService {
|
||||
}
|
||||
}
|
||||
|
||||
func (s *ProductService) GetJSON(p_id_product, p_id_lang, p_id_customer, b2b_id_country, p_quantity int) (*json.RawMessage, error) {
|
||||
products, err := s.productsRepo.GetJSON(p_id_product, constdata.SHOP_ID, p_id_lang, p_id_customer, b2b_id_country, p_quantity)
|
||||
func (s *ProductService) Get(
|
||||
p_id_product, p_id_lang, p_id_customer, b2b_id_country, p_quantity uint,
|
||||
) (*json.RawMessage, error) {
|
||||
|
||||
product, err := s.productsRepo.GetBase(p_id_product, constdata.SHOP_ID, p_id_lang)
|
||||
if err != nil {
|
||||
return products, err
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return products, nil
|
||||
price, err := s.productsRepo.GetPrice(p_id_product, nil, constdata.SHOP_ID, p_id_customer, b2b_id_country, p_quantity)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
variants, err := s.productsRepo.GetVariants(p_id_product, constdata.SHOP_ID, p_id_lang, p_id_customer, b2b_id_country, p_quantity)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
result := view.ProductFull{
|
||||
Product: product,
|
||||
Price: price,
|
||||
Variants: variants,
|
||||
}
|
||||
|
||||
if len(variants) > 0 {
|
||||
result.Variants = variants
|
||||
}
|
||||
|
||||
jsonBytes, err := json.Marshal(result)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
raw := json.RawMessage(jsonBytes)
|
||||
return &raw, nil
|
||||
}
|
||||
|
||||
func (s *ProductService) Find(id_lang, userID uint, p find.Paging, filters *filters.FiltersList) (find.Found[model.ProductInList], error) {
|
||||
return s.productsRepo.Find(id_lang, userID, p, filters)
|
||||
func (s *ProductService) Find(
|
||||
idLang uint,
|
||||
userID uint,
|
||||
p find.Paging,
|
||||
filters *filters.FiltersList,
|
||||
customer *model.Customer,
|
||||
quantity uint,
|
||||
shopID uint,
|
||||
) (*find.Found[model.ProductInList], error) {
|
||||
|
||||
if customer == nil || customer.Country == nil {
|
||||
return nil, errors.New("customer is nil or missing fields")
|
||||
}
|
||||
|
||||
found, err := s.productsRepo.Find(idLang, userID, p, filters)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// 1. collect simple products (no variants)
|
||||
simpleProductIndexes := make([]int, 0, len(found.Items))
|
||||
|
||||
for i := range found.Items {
|
||||
if found.Items[i].VariantsNumber <= 0 {
|
||||
simpleProductIndexes = append(simpleProductIndexes, i)
|
||||
}
|
||||
}
|
||||
|
||||
// 2. resolve prices ONLY for simple products
|
||||
for _, i := range simpleProductIndexes {
|
||||
price, err := s.productsRepo.GetPrice(
|
||||
found.Items[i].ProductID,
|
||||
nil,
|
||||
shopID,
|
||||
customer.ID,
|
||||
customer.CountryID,
|
||||
quantity,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
found.Items[i].PriceTaxExcl = price.FinalTaxExcl
|
||||
found.Items[i].PriceTaxIncl = price.FinalTaxIncl
|
||||
}
|
||||
|
||||
return found, nil
|
||||
}
|
||||
|
||||
func (s *ProductService) GetProductAttributes(
|
||||
langID uint,
|
||||
productID uint,
|
||||
shopID uint,
|
||||
customerID uint,
|
||||
countryID uint,
|
||||
quantity uint,
|
||||
) ([]view.ProductAttribute, error) {
|
||||
variants, err := s.productsRepo.GetVariants(productID, constdata.SHOP_ID, langID, customerID, countryID, quantity)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return variants, nil
|
||||
|
||||
}
|
||||
|
||||
func (s *ProductService) AddToFavorites(userID uint, productID uint) error {
|
||||
|
||||
@@ -4,6 +4,7 @@ import (
|
||||
"strings"
|
||||
"unicode"
|
||||
|
||||
"git.ma-al.com/goc_daniel/b2b/app/db"
|
||||
constdata "git.ma-al.com/goc_daniel/b2b/app/utils/const_data"
|
||||
"github.com/dlclark/regexp2"
|
||||
"golang.org/x/text/runes"
|
||||
@@ -22,7 +23,7 @@ func SanitizeSlug(s string) string {
|
||||
s = strings.TrimSpace(strings.ToLower(s))
|
||||
|
||||
// First apply explicit transliteration for language-specific letters.
|
||||
s = transliterateWithTable(s)
|
||||
s = transliterateSlug(s)
|
||||
|
||||
// Then normalize and strip any remaining combining marks.
|
||||
s = removeDiacritics(s)
|
||||
@@ -40,19 +41,17 @@ func SanitizeSlug(s string) string {
|
||||
return s
|
||||
}
|
||||
|
||||
func transliterateWithTable(s string) string {
|
||||
var b strings.Builder
|
||||
b.Grow(len(s))
|
||||
func transliterateSlug(s string) string {
|
||||
var cleared string
|
||||
|
||||
for _, r := range s {
|
||||
if repl, ok := constdata.TRANSLITERATION_TABLE[r]; ok {
|
||||
b.WriteString(repl)
|
||||
} else {
|
||||
b.WriteRune(r)
|
||||
}
|
||||
err := db.DB.Raw("SELECT slugify_eu(?)", s).Scan(&cleared).Error
|
||||
if err != nil {
|
||||
// log error
|
||||
_ = err
|
||||
return s
|
||||
}
|
||||
|
||||
return b.String()
|
||||
return cleared
|
||||
}
|
||||
|
||||
func removeDiacritics(s string) string {
|
||||
|
||||
124
app/service/specificPriceService/specificPriceService.go
Normal file
124
app/service/specificPriceService/specificPriceService.go
Normal file
@@ -0,0 +1,124 @@
|
||||
package specificPriceService
|
||||
|
||||
import (
|
||||
"context"
|
||||
|
||||
"git.ma-al.com/goc_daniel/b2b/app/model"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/repos/specificPriceRepo"
|
||||
"git.ma-al.com/goc_daniel/b2b/app/utils/responseErrors"
|
||||
)
|
||||
|
||||
type SpecificPriceService struct {
|
||||
specificPriceRepo specificPriceRepo.UISpecificPriceRepo
|
||||
}
|
||||
|
||||
func New() *SpecificPriceService {
|
||||
return &SpecificPriceService{
|
||||
specificPriceRepo: specificPriceRepo.New(),
|
||||
}
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) Create(ctx context.Context, pr *model.SpecificPrice) (*model.SpecificPrice, error) {
|
||||
if err := s.validateRequest(pr); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if err := s.specificPriceRepo.Create(ctx, pr); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return pr, nil
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) Update(ctx context.Context, id uint64, pr *model.SpecificPrice) (*model.SpecificPrice, error) {
|
||||
existing, err := s.specificPriceRepo.GetByID(ctx, id)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if existing == nil {
|
||||
return nil, responseErrors.ErrSpecificPriceNotFound
|
||||
}
|
||||
|
||||
if err := s.validateUpdateRequest(pr); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
pr.ID = id
|
||||
|
||||
if err := s.specificPriceRepo.Update(ctx, pr); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return pr, nil
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) GetByID(ctx context.Context, id uint64) (*model.SpecificPrice, error) {
|
||||
pr, err := s.specificPriceRepo.GetByID(ctx, id)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if pr == nil {
|
||||
return nil, responseErrors.ErrSpecificPriceNotFound
|
||||
}
|
||||
return pr, nil
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) List(ctx context.Context) ([]*model.SpecificPrice, error) {
|
||||
return s.specificPriceRepo.List(ctx)
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) SetActive(ctx context.Context, id uint64, active bool) error {
|
||||
pr, err := s.specificPriceRepo.GetByID(ctx, id)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if pr == nil {
|
||||
return responseErrors.ErrSpecificPriceNotFound
|
||||
}
|
||||
|
||||
return s.specificPriceRepo.SetActive(ctx, id, active)
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) Delete(ctx context.Context, id uint64) error {
|
||||
pr, err := s.specificPriceRepo.GetByID(ctx, id)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if pr == nil {
|
||||
return responseErrors.ErrSpecificPriceNotFound
|
||||
}
|
||||
|
||||
return s.specificPriceRepo.Delete(ctx, id)
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) validateRequest(pr *model.SpecificPrice) error {
|
||||
if pr.ReductionType != "amount" && pr.ReductionType != "percentage" {
|
||||
return responseErrors.ErrInvalidReductionType
|
||||
}
|
||||
|
||||
if pr.ReductionType == "percentage" && pr.PercentageReduction == nil {
|
||||
return responseErrors.ErrPercentageRequired
|
||||
}
|
||||
|
||||
if pr.ReductionType == "amount" && pr.Price == nil {
|
||||
return responseErrors.ErrPriceRequired
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *SpecificPriceService) validateUpdateRequest(pr *model.SpecificPrice) error {
|
||||
if pr.ReductionType != "" && pr.ReductionType != "amount" && pr.ReductionType != "percentage" {
|
||||
return responseErrors.ErrInvalidReductionType
|
||||
}
|
||||
|
||||
if pr.ReductionType == "percentage" && pr.PercentageReduction == nil {
|
||||
return responseErrors.ErrPercentageRequired
|
||||
}
|
||||
|
||||
if pr.ReductionType == "amount" && pr.Price == nil {
|
||||
return responseErrors.ErrPriceRequired
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
Reference in New Issue
Block a user