Compare commits
	
		
			4 Commits
		
	
	
		
			057565b0aa
			...
			e370308b10
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| e370308b10 | |||
| cf18598b96 | |||
| d332b0e1d4 | |||
| 8000f56c06 | 
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										3
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								go.mod
									
									
									
									
									
								
							| @ -109,7 +109,7 @@ require ( | ||||
| 	github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e | ||||
| 	github.com/spf13/viper v1.7.1 | ||||
| 	github.com/u2takey/ffmpeg-go v0.5.0 | ||||
| 	golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 | ||||
| 	golang.org/x/image v0.0.0-20220902085622-e7cb96979f69 | ||||
| ) | ||||
| 
 | ||||
| require ( | ||||
| @ -168,6 +168,7 @@ require ( | ||||
| 	github.com/spf13/jwalterweatherman v1.0.0 // indirect | ||||
| 	github.com/spf13/pflag v1.0.5 // indirect | ||||
| 	github.com/subosito/gotenv v1.2.0 // indirect | ||||
| 	github.com/tealeg/xlsx v1.0.5 // indirect | ||||
| 	github.com/tklauser/go-sysconf v0.3.6 // indirect | ||||
| 	github.com/tklauser/numcpus v0.2.2 // indirect | ||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | ||||
|  | ||||
							
								
								
									
										5
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								go.sum
									
									
									
									
									
								
							| @ -833,6 +833,8 @@ github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gt | ||||
| github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= | ||||
| github.com/subosito/gotenv v1.2.0 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s= | ||||
| github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= | ||||
| github.com/tealeg/xlsx v1.0.5 h1:+f8oFmvY8Gw1iUXzPk+kz+4GpbDZPK1FhPiQRd+ypgE= | ||||
| github.com/tealeg/xlsx v1.0.5/go.mod h1:btRS8dz54TDnvKNosuAqxrM1QgN1udgk9O34bDCnORM= | ||||
| github.com/tebeka/strftime v0.1.3/go.mod h1:7wJm3dZlpr4l/oVK0t1HYIc4rMzQ2XJlOMIUJUJH6XQ= | ||||
| github.com/tevid/gohamcrest v1.1.1 h1:ou+xSqlIw1xfGTg1uq1nif/htZ2S3EzRqLm2BP+tYU0= | ||||
| github.com/tevid/gohamcrest v1.1.1/go.mod h1:3UvtWlqm8j5JbwYZh80D/PVBt0mJ1eJiYgZMibh0H/k= | ||||
| @ -973,8 +975,9 @@ golang.org/x/exp v0.0.0-20200331195152-e8c3332aa8e5/go.mod h1:4M0jN8W1tt0AVLNr8H | ||||
| golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= | ||||
| golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= | ||||
| golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= | ||||
| golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| golang.org/x/image v0.0.0-20220902085622-e7cb96979f69 h1:Lj6HJGCSn5AjxRAH2+r35Mir4icalbqku+CLUtjnvXY= | ||||
| golang.org/x/image v0.0.0-20220902085622-e7cb96979f69/go.mod h1:doUCurBvlfPMKfmIpRIywoHmhN3VyhnoFDbvIEWF4hY= | ||||
| golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | ||||
| golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
|  | ||||
| @ -24,7 +24,9 @@ func BundleOrderRouter(r *gin.RouterGroup) { | ||||
| 		{ | ||||
| 			bundleOrderClientWebRoute.POST("bundle-order-list", bundle.OrderRecordsList) | ||||
| 			bundleOrderClientWebRoute.POST("bundle-order-list-V2", bundle.OrderRecordsListV2) | ||||
| 			bundleOrderClientWebRoute.POST("bundle-order-list-download", bundle.OrderRecordsListDownload) | ||||
| 			bundleOrderClientWebRoute.POST("reconciliation-list", bundle.GetReconciliationList) | ||||
| 			bundleOrderClientWebRoute.POST("reconciliation-list-download", bundle.GetReconciliationListDownload) | ||||
| 		} | ||||
| 		bundleOrderClientAppRoute := bundleOrderRoute.Group("common/app") | ||||
| 		{ | ||||
|  | ||||
| @ -12,12 +12,12 @@ import ( | ||||
| 	"fonchain-fiee/pkg/service/bundle/common" | ||||
| 	"fonchain-fiee/pkg/service/bundle/logic" | ||||
| 	bundleModel "fonchain-fiee/pkg/service/bundle/model" | ||||
| 	"github.com/360EntSecGroup-Skylar/excelize" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/gin-gonic/gin/binding" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/gin-gonic/gin/binding" | ||||
| ) | ||||
| 
 | ||||
| func CreateBundleOrderAddSignature(c *gin.Context) { | ||||
| @ -220,6 +220,10 @@ func CreateBundleOrderSignature(c *gin.Context) { | ||||
| 	//有套餐并且套餐未过期
 | ||||
| 	if orderRecordsList.OrderRecords != nil { | ||||
| 		for _, orderInfo := range orderRecordsList.OrderRecords { | ||||
| 			if orderInfo.CustomerID == strconv.FormatUint(userInfo.ID, 10) && orderInfo.Status == 1 { | ||||
| 				service.Error(c, errors.New(common.ThereAreOutstandingOrders)) | ||||
| 				return | ||||
| 			} | ||||
| 			if orderInfo.CustomerID == strconv.FormatUint(userInfo.ID, 10) && orderInfo.ExpirationTime > time.Now().Format("2006-01-02") { | ||||
| 				service.Error(c, errors.New(common.HadOrder)) | ||||
| 				return | ||||
| @ -481,6 +485,184 @@ func OrderRecordsListV2(c *gin.Context) { | ||||
| 
 | ||||
| 	service.Success(c, orderList) | ||||
| } | ||||
| func OrderRecordsListDownload(c *gin.Context) { | ||||
| 	var req bundle.OrderRecordsRequestV2 | ||||
| 	if err := c.ShouldBindBodyWith(&req, binding.JSON); err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	res := &bundle.OrderRecordsResponseV2{} | ||||
| 
 | ||||
| 	// Step 1: 如果有姓名/电话筛选,先查用户列表
 | ||||
| 	if req.CustomerName != "" { | ||||
| 		userListResp, err := service.AccountFieeProvider.UserList(context.Background(), &accountFiee.UserListRequest{ | ||||
| 			BlurNameTel: req.CustomerName, | ||||
| 			Domain:      "app", | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			service.Error(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 
 | ||||
| 		if len(userListResp.UserList) == 0 { | ||||
| 			// 没查到用户,直接返回空结果
 | ||||
| 			res.Page = req.Page | ||||
| 			res.PageSize = req.PageSize | ||||
| 			res.Total = 0 | ||||
| 			service.Success(c, res) | ||||
| 			return | ||||
| 		} | ||||
| 
 | ||||
| 		// 提取用户ID列表
 | ||||
| 		for _, u := range userListResp.UserList { | ||||
| 			req.UserIds = append(req.UserIds, int64(u.Id)) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Step 2: 查询订单列表
 | ||||
| 	orderList, err := service.BundleProvider.OrderRecordsListV2(context.Background(), &req) | ||||
| 	if err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// Step 3: 如果订单不为空,查一次用户信息填充(只查一次)
 | ||||
| 	if len(orderList.BundleInfo) > 0 { | ||||
| 		// 收集订单里的所有用户ID
 | ||||
| 		userIdSet := make(map[int64]struct{}) | ||||
| 		for _, i := range orderList.BundleInfo { | ||||
| 			userIdSet[i.CustomerId] = struct{}{} | ||||
| 		} | ||||
| 
 | ||||
| 		var userIds []int64 | ||||
| 		for id := range userIdSet { | ||||
| 			userIds = append(userIds, id) | ||||
| 		} | ||||
| 
 | ||||
| 		userListResp, err := service.AccountFieeProvider.UserList(context.Background(), &accountFiee.UserListRequest{ | ||||
| 			Ids:    userIds, | ||||
| 			Domain: "app", | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			service.Error(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 
 | ||||
| 		// 建立用户ID -> 用户信息映射
 | ||||
| 		userMap := make(map[int64]*accountFiee.UserListInfo, len(userListResp.UserList)) | ||||
| 		for _, u := range userListResp.UserList { | ||||
| 			userMap[int64(u.Id)] = u | ||||
| 		} | ||||
| 
 | ||||
| 		// 填充订单中的用户信息
 | ||||
| 		for _, item := range orderList.BundleInfo { | ||||
| 			if u, ok := userMap[item.CustomerId]; ok { | ||||
| 				item.CustomerName = u.Name | ||||
| 				item.TelNum = u.TelNum | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	excelFile, err := exportExcel(orderList.BundleInfo) | ||||
| 	if err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// 返回 Excel 文件流给前端
 | ||||
| 	c.Header("Content-Type", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet") | ||||
| 	c.Header("Content-Disposition", "attachment; filename=order_list.xlsx") | ||||
| 	c.Header("File-Name", "order_list.xlsx") | ||||
| 	c.Header("Access-Control-Expose-Headers", "File-Name") | ||||
| 	_ = excelFile.Write(c.Writer) | ||||
| 
 | ||||
| } | ||||
| func exportExcel(orderList []*bundle.OrderBundleRecordInfo) (*excelize.File, error) { | ||||
| 	f := excelize.NewFile() | ||||
| 	sheetName := "Sheet1" | ||||
| 	f.SetSheetName("Sheet1", sheetName) | ||||
| 
 | ||||
| 	headers := []string{ | ||||
| 		"套餐订单号", "套餐", "套餐付款状态", "艺人手机号", "艺人", "套餐订单创建时间", "套餐金额", | ||||
| 		"增值服务订单号", "增值税服务金额", "结算金额", "币种", "手续费", "汇率", "增值订单创建时间", "增值付款状态", | ||||
| 	} | ||||
| 	for i, h := range headers { | ||||
| 		cell := fmt.Sprintf("%s%d", string(rune('A'+i)), 1) | ||||
| 		f.SetCellValue(sheetName, cell, h) | ||||
| 	} | ||||
| 
 | ||||
| 	rowIndex := 2 | ||||
| 	for _, bundleInfo := range orderList { | ||||
| 		addCount := len(bundleInfo.AddBundleInfo) | ||||
| 		mergeRows := 1 | ||||
| 		if addCount > 1 { | ||||
| 			mergeRows = addCount | ||||
| 		} | ||||
| 
 | ||||
| 		// 写入主订单信息并合并单元格
 | ||||
| 		for i := 0; i < 7; i++ { | ||||
| 			col := string(rune('A' + i)) | ||||
| 			startCell := fmt.Sprintf("%s%d", col, rowIndex) | ||||
| 			endCell := fmt.Sprintf("%s%d", col, rowIndex+mergeRows-1) | ||||
| 			if mergeRows > 1 { | ||||
| 				f.MergeCell(sheetName, startCell, endCell) | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("A%d", rowIndex), bundleInfo.OrderNo) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("B%d", rowIndex), bundleInfo.BundleName) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("C%d", rowIndex), GetPayStatusText(bundleInfo.PayStatus)) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("D%d", rowIndex), bundleInfo.TelNum) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("E%d", rowIndex), bundleInfo.CustomerName) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("F%d", rowIndex), bundleInfo.BundleCreateAt) | ||||
| 		f.SetCellValue(sheetName, fmt.Sprintf("G%d", rowIndex), bundleInfo.Amount) | ||||
| 
 | ||||
| 		if addCount > 0 { | ||||
| 			for i, add := range bundleInfo.AddBundleInfo { | ||||
| 				r := rowIndex + i | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("H%d", r), add.OrderAddNo) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("I%d", r), add.Amount) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("J%d", r), add.SettlementAmount) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("K%d", r), GetCurrencyTypeText(add.CurrencyType)) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("L%d", r), add.HandlingFee) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("M%d", r), add.ExchangeRate) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("N%d", r), add.OrderAddCreateAt) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("O%d", r), GetPayStatusText(add.AddPayStatus)) | ||||
| 			} | ||||
| 		} else { | ||||
| 			for i := 8; i <= 15; i++ { | ||||
| 				col := string(rune('A' + i)) | ||||
| 				f.SetCellValue(sheetName, fmt.Sprintf("%s%d", col, rowIndex), "") | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		rowIndex += mergeRows | ||||
| 	} | ||||
| 
 | ||||
| 	return f, nil | ||||
| } | ||||
| func GetPayStatusText(status int32) string { | ||||
| 	switch status { | ||||
| 	case 1: | ||||
| 		return "未支付" | ||||
| 	case 2: | ||||
| 		return "已支付" | ||||
| 	default: | ||||
| 		return strconv.Itoa(int(status)) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // 辅助函数:获取货币类型文本
 | ||||
| func GetCurrencyTypeText(currencyType int32) string { | ||||
| 	switch currencyType { | ||||
| 	case 1: | ||||
| 		return "人民币" | ||||
| 	case 2: | ||||
| 		return "美元" | ||||
| 	default: | ||||
| 		return strconv.Itoa(int(currencyType)) | ||||
| 	} | ||||
| } | ||||
| func OrderRecordsList(c *gin.Context) { | ||||
| 	var req bundle.OrderRecordsRequest | ||||
| 
 | ||||
|  | ||||
| @ -17,7 +17,7 @@ const ( | ||||
| 	InvalidOrderAmount = "订单金额错误" | ||||
| 
 | ||||
| 	HadPay                    = "订单已支付" | ||||
| 
 | ||||
| 	ThereAreOutstandingOrders = "您还有未支付的订单,无法再次购买" | ||||
| 	HadOrder                  = "您已购买过套餐,无法再次购买" | ||||
| 	InvalidValueAddBundleNum  = "套餐数量无效" | ||||
| 	ThePackageHasExpired      = "当前套餐已过期" | ||||
|  | ||||
| @ -4,6 +4,7 @@ import ( | ||||
| 	"context" | ||||
| 	"fonchain-fiee/api/bundle" | ||||
| 	"fonchain-fiee/pkg/service" | ||||
| 	"fonchain-fiee/pkg/utils" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/gin-gonic/gin/binding" | ||||
| ) | ||||
| @ -23,3 +24,52 @@ func GetReconciliationList(c *gin.Context) { | ||||
| 	return | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| func GetReconciliationListDownload(c *gin.Context) { | ||||
| 	var req bundle.GetReconciliationListReq | ||||
| 	if err := c.ShouldBindBodyWith(&req, binding.JSON); err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	detail, detailErr := service.BundleProvider.GetReconciliationList(context.Background(), &req) | ||||
| 	if detailErr != nil { | ||||
| 		service.Error(c, detailErr) | ||||
| 		return | ||||
| 	} | ||||
| 	titleList := []string{ | ||||
| 		"关联套餐订单号", "关联增值服务订单号", "对账单创建时间", "艺人", "艺人手机号", "套餐", "支付金额", "币种", "支付渠道", "支付时间", "支付状态", "流水号", | ||||
| 	} | ||||
| 	var dataList []interface{} | ||||
| 
 | ||||
| 	for _, i := range detail.List { | ||||
| 		payStatus := GetPayStatusText(i.PayStatus) | ||||
| 		currencyType := GetCurrencyTypeText(i.CurrencyType) | ||||
| 		payChannel := "未知" | ||||
| 		if i.PayChannel == 1 { | ||||
| 			payChannel = "支付宝" | ||||
| 		} | ||||
| 		data := []any{ | ||||
| 			i.BundleOrderOn, | ||||
| 			i.BundleAddOrderOn, | ||||
| 			i.CreationTime, | ||||
| 			i.UserName, | ||||
| 			i.UserTel, | ||||
| 			i.BundleName, | ||||
| 			i.PayAmount, | ||||
| 			currencyType, | ||||
| 			payChannel, | ||||
| 			i.PayTime, | ||||
| 			payStatus, | ||||
| 			i.SerialNumber, | ||||
| 		} | ||||
| 		dataList = append(dataList, &data) | ||||
| 	} | ||||
| 	content, err := utils.ToExcelByType(titleList, dataList, "slice", "") | ||||
| 	if err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	utils.ResponseXls(c, content, "对账单") | ||||
| 	return | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -5,7 +5,10 @@ import ( | ||||
| 	"crypto/sha256" | ||||
| 	"encoding/hex" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"fonchain-fiee/pkg/e" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/tealeg/xlsx" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"net/http" | ||||
| @ -78,3 +81,44 @@ func CheckDirPath(path string, create bool) (exists bool, err error) { | ||||
| 	exists = true | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| // ToExcelByType 转成Excel类型
 | ||||
| func ToExcelByType(titleList []string, dataList []interface{}, dataType string, filePath string) (content io.ReadSeeker, err error) { | ||||
| 	// 生成一个新的文件
 | ||||
| 	file := xlsx.NewFile() | ||||
| 	// 添加sheet页
 | ||||
| 	sheet, _ := file.AddSheet("Sheet1") | ||||
| 	// 插入表头
 | ||||
| 	titleRow := sheet.AddRow() | ||||
| 	for _, v := range titleList { | ||||
| 		cell := titleRow.AddCell() | ||||
| 		cell.Value = v | ||||
| 	} | ||||
| 	// 插入内容
 | ||||
| 	for _, v := range dataList { | ||||
| 		row := sheet.AddRow() | ||||
| 		if dataType == "struct" { | ||||
| 			row.WriteStruct(v, -1) | ||||
| 		} else if dataType == "slice" { | ||||
| 			row.WriteSlice(v, -1) | ||||
| 		} | ||||
| 	} | ||||
| 	var buffer bytes.Buffer | ||||
| 	_ = file.Write(&buffer) | ||||
| 	if filePath != "" { | ||||
| 		if err = file.Save(filePath); err != nil { | ||||
| 			return | ||||
| 		} | ||||
| 	} else { | ||||
| 		content = bytes.NewReader(buffer.Bytes()) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| // ResponseXls  content 为上面生成的io.ReadSeeker, fileTag 为返回前端的文件名
 | ||||
| func ResponseXls(c *gin.Context, content io.ReadSeeker, fileTag string) { | ||||
| 	fileName := fmt.Sprintf("%s.%s", fileTag, ExcelPrefix) | ||||
| 	c.Writer.Header().Add("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, fileName)) | ||||
| 	c.Writer.Header().Add("Content-Type", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet") | ||||
| 	http.ServeContent(c.Writer, c.Request, fileName, time.Now(), content) | ||||
| } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user