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/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e | ||||||
| 	github.com/spf13/viper v1.7.1 | 	github.com/spf13/viper v1.7.1 | ||||||
| 	github.com/u2takey/ffmpeg-go v0.5.0 | 	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 ( | require ( | ||||||
| @ -168,6 +168,7 @@ require ( | |||||||
| 	github.com/spf13/jwalterweatherman v1.0.0 // indirect | 	github.com/spf13/jwalterweatherman v1.0.0 // indirect | ||||||
| 	github.com/spf13/pflag v1.0.5 // indirect | 	github.com/spf13/pflag v1.0.5 // indirect | ||||||
| 	github.com/subosito/gotenv v1.2.0 // 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/go-sysconf v0.3.6 // indirect | ||||||
| 	github.com/tklauser/numcpus v0.2.2 // indirect | 	github.com/tklauser/numcpus v0.2.2 // indirect | ||||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // 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/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 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s= | ||||||
| github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= | 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/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 h1:ou+xSqlIw1xfGTg1uq1nif/htZ2S3EzRqLm2BP+tYU0= | ||||||
| github.com/tevid/gohamcrest v1.1.1/go.mod h1:3UvtWlqm8j5JbwYZh80D/PVBt0mJ1eJiYgZMibh0H/k= | 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-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-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-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-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-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-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | ||||||
| golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | 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", bundle.OrderRecordsList) | ||||||
| 			bundleOrderClientWebRoute.POST("bundle-order-list-V2", bundle.OrderRecordsListV2) | 			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", bundle.GetReconciliationList) | ||||||
|  | 			bundleOrderClientWebRoute.POST("reconciliation-list-download", bundle.GetReconciliationListDownload) | ||||||
| 		} | 		} | ||||||
| 		bundleOrderClientAppRoute := bundleOrderRoute.Group("common/app") | 		bundleOrderClientAppRoute := bundleOrderRoute.Group("common/app") | ||||||
| 		{ | 		{ | ||||||
|  | |||||||
| @ -12,12 +12,12 @@ import ( | |||||||
| 	"fonchain-fiee/pkg/service/bundle/common" | 	"fonchain-fiee/pkg/service/bundle/common" | ||||||
| 	"fonchain-fiee/pkg/service/bundle/logic" | 	"fonchain-fiee/pkg/service/bundle/logic" | ||||||
| 	bundleModel "fonchain-fiee/pkg/service/bundle/model" | 	bundleModel "fonchain-fiee/pkg/service/bundle/model" | ||||||
|  | 	"github.com/360EntSecGroup-Skylar/excelize" | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/gin-gonic/gin/binding" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 
 |  | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"github.com/gin-gonic/gin/binding" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func CreateBundleOrderAddSignature(c *gin.Context) { | func CreateBundleOrderAddSignature(c *gin.Context) { | ||||||
| @ -220,6 +220,10 @@ func CreateBundleOrderSignature(c *gin.Context) { | |||||||
| 	//有套餐并且套餐未过期
 | 	//有套餐并且套餐未过期
 | ||||||
| 	if orderRecordsList.OrderRecords != nil { | 	if orderRecordsList.OrderRecords != nil { | ||||||
| 		for _, orderInfo := range orderRecordsList.OrderRecords { | 		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") { | 			if orderInfo.CustomerID == strconv.FormatUint(userInfo.ID, 10) && orderInfo.ExpirationTime > time.Now().Format("2006-01-02") { | ||||||
| 				service.Error(c, errors.New(common.HadOrder)) | 				service.Error(c, errors.New(common.HadOrder)) | ||||||
| 				return | 				return | ||||||
| @ -481,6 +485,184 @@ func OrderRecordsListV2(c *gin.Context) { | |||||||
| 
 | 
 | ||||||
| 	service.Success(c, orderList) | 	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) { | func OrderRecordsList(c *gin.Context) { | ||||||
| 	var req bundle.OrderRecordsRequest | 	var req bundle.OrderRecordsRequest | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -17,7 +17,7 @@ const ( | |||||||
| 	InvalidOrderAmount = "订单金额错误" | 	InvalidOrderAmount = "订单金额错误" | ||||||
| 
 | 
 | ||||||
| 	HadPay                    = "订单已支付" | 	HadPay                    = "订单已支付" | ||||||
| 
 | 	ThereAreOutstandingOrders = "您还有未支付的订单,无法再次购买" | ||||||
| 	HadOrder                  = "您已购买过套餐,无法再次购买" | 	HadOrder                  = "您已购买过套餐,无法再次购买" | ||||||
| 	InvalidValueAddBundleNum  = "套餐数量无效" | 	InvalidValueAddBundleNum  = "套餐数量无效" | ||||||
| 	ThePackageHasExpired      = "当前套餐已过期" | 	ThePackageHasExpired      = "当前套餐已过期" | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"fonchain-fiee/api/bundle" | 	"fonchain-fiee/api/bundle" | ||||||
| 	"fonchain-fiee/pkg/service" | 	"fonchain-fiee/pkg/service" | ||||||
|  | 	"fonchain-fiee/pkg/utils" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| ) | ) | ||||||
| @ -23,3 +24,52 @@ func GetReconciliationList(c *gin.Context) { | |||||||
| 	return | 	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" | 	"crypto/sha256" | ||||||
| 	"encoding/hex" | 	"encoding/hex" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
| 	"fonchain-fiee/pkg/e" | 	"fonchain-fiee/pkg/e" | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/tealeg/xlsx" | ||||||
| 	"io" | 	"io" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @ -78,3 +81,44 @@ func CheckDirPath(path string, create bool) (exists bool, err error) { | |||||||
| 	exists = true | 	exists = true | ||||||
| 	return | 	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