| 
									
										
										
										
											2023-07-05 16:03:16 +08:00
										 |  |  | package logic | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"fusenapi/constants" | 
					
						
							|  |  |  | 	"fusenapi/model/gmodel" | 
					
						
							|  |  |  | 	"fusenapi/utils/auth" | 
					
						
							|  |  |  | 	"fusenapi/utils/basic" | 
					
						
							|  |  |  | 	"fusenapi/utils/encryption_decryption" | 
					
						
							|  |  |  | 	"fusenapi/utils/id_generator" | 
					
						
							|  |  |  | 	"image" | 
					
						
							|  |  |  | 	"image/gif" | 
					
						
							|  |  |  | 	"image/jpeg" | 
					
						
							|  |  |  | 	"image/png" | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 10:54:33 +08:00
										 |  |  | 	"github.com/google/uuid" | 
					
						
							|  |  |  | 	"github.com/nfnt/resize" | 
					
						
							|  |  |  | 	"gorm.io/gorm" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-05 16:03:16 +08:00
										 |  |  | 	"context" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"fusenapi/server/product/internal/svc" | 
					
						
							|  |  |  | 	"fusenapi/server/product/internal/types" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/zeromicro/go-zero/core/logx" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type SaveDesignLogic struct { | 
					
						
							|  |  |  | 	logx.Logger | 
					
						
							|  |  |  | 	ctx    context.Context | 
					
						
							|  |  |  | 	svcCtx *svc.ServiceContext | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewSaveDesignLogic(ctx context.Context, svcCtx *svc.ServiceContext) *SaveDesignLogic { | 
					
						
							|  |  |  | 	return &SaveDesignLogic{ | 
					
						
							|  |  |  | 		Logger: logx.WithContext(ctx), | 
					
						
							|  |  |  | 		ctx:    ctx, | 
					
						
							|  |  |  | 		svcCtx: svcCtx, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *SaveDesignLogic) SaveDesign(req *types.SaveDesignReq, userinfo *auth.UserInfo) (resp *basic.Response) { | 
					
						
							|  |  |  | 	if userinfo.GetIdType() != auth.IDTYPE_User { | 
					
						
							|  |  |  | 		return resp.SetStatusWithMessage(basic.CodeUnAuth, "please login first") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//查询是否是加密的(不太合理) | 
					
						
							|  |  |  | 	encryptWebsetting, err := l.svcCtx.AllModels.FsWebSet.FindValueByKey(l.ctx, "is_encrypt") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if errors.Is(err, gorm.ErrRecordNotFound) { | 
					
						
							|  |  |  | 			return resp.SetStatusWithMessage(basic.CodeDbRecordNotFoundErr, "web setting is_encrypt is not exists") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		logx.Error(err) | 
					
						
							|  |  |  | 		return resp.SetStatusWithMessage(basic.CodeDbSqlErr, "failed to get web setting") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var postInfo types.SaveDesignReqRealStruct | 
					
						
							|  |  |  | 	//不加密 | 
					
						
							|  |  |  | 	if encryptWebsetting.Value == nil || *encryptWebsetting.Value == "0" { | 
					
						
							|  |  |  | 		if err = json.Unmarshal([]byte(req.Data), &postInfo); err != nil { | 
					
						
							|  |  |  | 			return resp.SetStatusWithMessage(basic.CodeJsonErr, "failed to parse json data,format may be invalid") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { //加密的 | 
					
						
							|  |  |  | 		//解密数据 | 
					
						
							|  |  |  | 		desData, err := encryption_decryption.CBCDecrypt(req.Data) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			logx.Error(err) | 
					
						
							|  |  |  | 			return resp.SetStatusWithMessage(basic.CodeAesCbcDecryptionErr, "failed to decryption data") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err = json.Unmarshal([]byte(desData), &postInfo); err != nil { | 
					
						
							|  |  |  | 			logx.Error(err) | 
					
						
							|  |  |  | 			return resp.SetStatusWithMessage(basic.CodeJsonErr, "failed to parse json data") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	infoBytes, _ := json.Marshal(postInfo.Data) | 
					
						
							|  |  |  | 	info := string(infoBytes) | 
					
						
							| 
									
										
										
										
											2023-08-22 10:54:33 +08:00
										 |  |  | 	now := time.Now().UTC() | 
					
						
							| 
									
										
										
										
											2023-07-05 16:03:16 +08:00
										 |  |  | 	logoColorBytes, _ := json.Marshal(postInfo.Data.Logo.Colors) | 
					
						
							|  |  |  | 	logoColor := string(logoColorBytes) | 
					
						
							|  |  |  | 	saveData := gmodel.FsProductDesign{ | 
					
						
							|  |  |  | 		UserId:     &userinfo.UserId, | 
					
						
							|  |  |  | 		ProductId:  &postInfo.ProductId, | 
					
						
							|  |  |  | 		TemplateId: &postInfo.TemplateId, | 
					
						
							|  |  |  | 		SizeId:     &postInfo.SizeId, | 
					
						
							|  |  |  | 		OptionalId: &postInfo.OptionalId, | 
					
						
							|  |  |  | 		Cover:      &postInfo.Cover, | 
					
						
							|  |  |  | 		Info:       &info, | 
					
						
							|  |  |  | 		Utime:      &now, | 
					
						
							|  |  |  | 		LogoColor:  &logoColor, | 
					
						
							|  |  |  | 		PageGuid:   &postInfo.PageGuid, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	switch postInfo.Sn { | 
					
						
							|  |  |  | 	case "": //新增 | 
					
						
							|  |  |  | 		status := int64(1) | 
					
						
							|  |  |  | 		postInfo.Sn, err = id_generator.GenSnowFlakeId() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			logx.Error(err) | 
					
						
							|  |  |  | 			return resp.SetStatusWithMessage(basic.CodeServiceErr, "failed to gen sn ") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		saveData.Status = &status | 
					
						
							|  |  |  | 		saveData.Sn = &postInfo.Sn | 
					
						
							|  |  |  | 		err = l.svcCtx.AllModels.FsProductDesign.Create(l.ctx, &saveData) | 
					
						
							|  |  |  | 	default: //更新 | 
					
						
							|  |  |  | 		err = l.svcCtx.AllModels.FsProductDesign.UpdateBySn(l.ctx, postInfo.Sn, &saveData) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		logx.Error(err) | 
					
						
							|  |  |  | 		return resp.SetStatusWithMessage(basic.CodeDbSqlErr, "failed to save design") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if postInfo.Cover == "" { | 
					
						
							|  |  |  | 		return resp.SetStatusWithMessage(basic.CodeOK, "success", types.SaveDesignRsp{Sn: postInfo.Sn}) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-07-05 16:21:26 +08:00
										 |  |  | 	// TODO 图片待优化处理 | 
					
						
							| 
									
										
										
										
											2023-07-05 16:08:25 +08:00
										 |  |  | 	/*if err = l.CreateStepThumbnailImage(l.ctx, postInfo.Cover); err != nil { | 
					
						
							| 
									
										
										
										
											2023-07-05 16:03:16 +08:00
										 |  |  | 		logx.Error(err) | 
					
						
							|  |  |  | 		return resp.SetStatusWithMessage(basic.CodeServiceErr, "failed to create step thumbnail image ") | 
					
						
							| 
									
										
										
										
											2023-07-05 16:08:25 +08:00
										 |  |  | 	}*/ | 
					
						
							| 
									
										
										
										
											2023-07-05 16:03:16 +08:00
										 |  |  | 	return resp.SetStatusWithMessage(basic.CodeOK, "success", types.SaveDesignRsp{Sn: postInfo.Sn}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 创建阶梯缩略图 | 
					
						
							|  |  |  | func (l *SaveDesignLogic) CreateStepThumbnailImage(ctx context.Context, coverImage string) error { | 
					
						
							|  |  |  | 	httpRsp, err := http.Get(coverImage) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer httpRsp.Body.Close() | 
					
						
							|  |  |  | 	coverImg, _, err := image.Decode(httpRsp.Body) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	coverImgOrgWith := coverImg.Bounds().Dx() | 
					
						
							|  |  |  | 	coverImgOrgHeight := coverImg.Bounds().Dy() | 
					
						
							|  |  |  | 	fileExt := path.Ext(coverImage) | 
					
						
							|  |  |  | 	for _, size := range constants.IMAGE_CROPPING_STEP_SIZE { | 
					
						
							|  |  |  | 		//尺寸大于原图 | 
					
						
							|  |  |  | 		if size > coverImgOrgWith { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		//缩放比例按照宽度来设定 | 
					
						
							|  |  |  | 		scale := size / coverImgOrgWith | 
					
						
							|  |  |  | 		height := scale * coverImgOrgHeight | 
					
						
							|  |  |  | 		tmpImage := resize.Resize(uint(size), uint(height), coverImg, resize.Lanczos3) | 
					
						
							|  |  |  | 		fileName := fmt.Sprintf("%s_%d_%s", uuid.New().String(), size, fileExt) | 
					
						
							|  |  |  | 		targetFile, err := os.Create(fileName) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		defer targetFile.Close() | 
					
						
							|  |  |  | 		switch fileExt { | 
					
						
							|  |  |  | 		case ".png": | 
					
						
							|  |  |  | 			err = png.Encode(targetFile, tmpImage) | 
					
						
							|  |  |  | 		case ".jpg", ".jpeg": | 
					
						
							|  |  |  | 			err = jpeg.Encode(targetFile, tmpImage, &jpeg.Options{Quality: 100}) | 
					
						
							|  |  |  | 		case ".gif": | 
					
						
							|  |  |  | 			err = gif.Encode(targetFile, tmpImage, nil) | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			err = errors.New("unSupport image format") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |