shipping.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. package order
  2. import (
  3. "fmt"
  4. "time"
  5. "github.com/silenceper/wechat/v2/miniprogram/context"
  6. "github.com/silenceper/wechat/v2/util"
  7. )
  8. const (
  9. // 发货信息录入
  10. uploadShippingInfoURL = "https://api.weixin.qq.com/wxa/sec/order/upload_shipping_info?access_token=%s"
  11. // 查询订单发货状态
  12. getShippingOrderURL = "https://api.weixin.qq.com/wxa/sec/order/get_order?access_token=%s"
  13. // 查询订单列表
  14. getShippingOrderListURL = "https://api.weixin.qq.com/wxa/sec/order/get_order_list?access_token=%s"
  15. // 确认收货提醒接口
  16. notifyConfirmReceiveURL = "https://api.weixin.qq.com/wxa/sec/order/notify_confirm_receive?access_token=%s"
  17. )
  18. // Shipping 发货信息管理
  19. type Shipping struct {
  20. *context.Context
  21. }
  22. // NewShipping init
  23. func NewShipping(ctx *context.Context) *Shipping {
  24. return &Shipping{ctx}
  25. }
  26. // UploadShippingInfo 发货信息录入
  27. // see https://developers.weixin.qq.com/miniprogram/dev/platform-capabilities/business-capabilities/order-shipping/order-shipping.html
  28. func (shipping *Shipping) UploadShippingInfo(in *UploadShippingInfoRequest) (err error) {
  29. accessToken, err := shipping.GetAccessToken()
  30. if err != nil {
  31. return
  32. }
  33. uri := fmt.Sprintf(uploadShippingInfoURL, accessToken)
  34. response, err := util.PostJSON(uri, in)
  35. if err != nil {
  36. return
  37. }
  38. // 使用通用方法返回错误
  39. return util.DecodeWithCommonError(response, "UploadShippingInfo")
  40. }
  41. // GetShippingOrder 查询订单发货状态
  42. func (shipping *Shipping) GetShippingOrder(in *GetShippingOrderRequest) (res ShippingOrderResponse, err error) {
  43. accessToken, err := shipping.GetAccessToken()
  44. if err != nil {
  45. return
  46. }
  47. uri := fmt.Sprintf(getShippingOrderURL, accessToken)
  48. response, err := util.PostJSON(uri, in)
  49. if err != nil {
  50. return
  51. }
  52. err = util.DecodeWithError(response, &res, "GetShippingOrder")
  53. return
  54. }
  55. // GetShippingOrderList 查询订单列表
  56. func (shipping *Shipping) GetShippingOrderList(in *GetShippingOrderListRequest) (res GetShippingOrderListResponse, err error) {
  57. accessToken, err := shipping.GetAccessToken()
  58. if err != nil {
  59. return
  60. }
  61. uri := fmt.Sprintf(getShippingOrderListURL, accessToken)
  62. response, err := util.PostJSON(uri, in)
  63. if err != nil {
  64. return
  65. }
  66. err = util.DecodeWithError(response, &res, "GetShippingOrderList")
  67. return
  68. }
  69. // NotifyConfirmReceive 确认收货提醒接口
  70. func (shipping *Shipping) NotifyConfirmReceive(in *NotifyConfirmReceiveRequest) (err error) {
  71. accessToken, err := shipping.GetAccessToken()
  72. if err != nil {
  73. return
  74. }
  75. uri := fmt.Sprintf(notifyConfirmReceiveURL, accessToken)
  76. response, err := util.PostJSON(uri, in)
  77. if err != nil {
  78. return
  79. }
  80. // 使用通用方法返回错误
  81. return util.DecodeWithCommonError(response, "NotifyConfirmReceive")
  82. }
  83. // UploadShippingInfoRequest 发货信息录入请求参数
  84. type UploadShippingInfoRequest struct {
  85. OrderKey *ShippingOrderKey `json:"order_key"` // 订单,需要上传物流信息的订单
  86. LogisticsType LogisticsType `json:"logistics_type"` // 物流模式
  87. DeliveryMode DeliveryMode `json:"delivery_mode"` // 发货模式
  88. IsAllDelivered bool `json:"is_all_delivered"` // 分拆发货模式时必填,用于标识分拆发货模式下是否已全部发货完成
  89. ShippingList []*ShippingInfo `json:"shipping_list"` // 物流信息列表,发货物流单列表,支持统一发货(单个物流单)和分拆发货(多个物流单)两种模式
  90. UploadTime *time.Time `json:"upload_time"` // 上传时间,用于标识请求的先后顺序
  91. Payer *ShippingPayer `json:"payer"` // 支付人信息
  92. }
  93. // ShippingOrderKey 订单
  94. type ShippingOrderKey struct {
  95. OrderNumberType NumberType `json:"order_number_type"` // 订单单号类型,用于确认需要上传详情的订单。枚举值1,使用下单商户号和商户侧单号;枚举值2,使用微信支付单号。
  96. TransactionID string `json:"transaction_id"` // 原支付交易对应的微信订单号
  97. Mchid string `json:"mchid"` // 支付下单商户的商户号,由微信支付生成并下发
  98. OutTradeNo string `json:"out_trade_no"` // 商户系统内部订单号,只能是数字、大小写字母`_-*`且在同一个商户号下唯一
  99. }
  100. // ShippingPayer 支付者信息
  101. type ShippingPayer struct {
  102. Openid string `json:"openid"` // 用户标识,用户在小程序appid下的唯一标识
  103. }
  104. // ShippingInfo 物流信息
  105. type ShippingInfo struct {
  106. TrackingNo string `json:"tracking_no"` // 物流单号,物流快递发货时必填
  107. ExpressCompany string `json:"express_company"` // 物流公司编码,快递公司ID,物流快递发货时必填;参见「查询物流公司编码列表」
  108. ItemDesc string `json:"item_desc"` // 商品信息,例如:微信红包抱枕*1个,限120个字以内
  109. Contact ShippingContact `json:"contact"` // 联系方式,当发货的物流公司为顺丰时,联系方式为必填,收件人或寄件人联系方式二选一
  110. }
  111. // ShippingContact 联系方式
  112. type ShippingContact struct {
  113. ConsignorContact string `json:"consignor_contact"` // 寄件人联系方式,寄件人联系方式,采用掩码传输,最后4位数字不能打掩码
  114. ReceiverContact string `json:"receiver_contact"` // 收件人联系方式,收件人联系方式,采用掩码传输,最后4位数字不能打掩码
  115. }
  116. // DeliveryMode 发货模式
  117. type DeliveryMode uint8
  118. const (
  119. // DeliveryModeUnifiedDelivery 统一发货
  120. DeliveryModeUnifiedDelivery DeliveryMode = 1
  121. // DeliveryModeSplitDelivery 分拆发货
  122. DeliveryModeSplitDelivery DeliveryMode = 2
  123. )
  124. // LogisticsType 物流模式
  125. type LogisticsType uint8
  126. const (
  127. // LogisticsTypeExpress 实体物流配送采用快递公司进行实体物流配送形式
  128. LogisticsTypeExpress LogisticsType = 1
  129. // LogisticsTypeSameCity 同城配送
  130. LogisticsTypeSameCity LogisticsType = 2
  131. // LogisticsTypeVirtual 虚拟商品,虚拟商品,例如话费充值,点卡等,无实体配送形式
  132. LogisticsTypeVirtual LogisticsType = 3
  133. // LogisticsTypeSelfPickup 用户自提
  134. LogisticsTypeSelfPickup LogisticsType = 4
  135. )
  136. // NumberType 订单单号类型
  137. type NumberType uint8
  138. const (
  139. // NumberTypeOutTradeNo 使用下单商户号和商户侧单号
  140. NumberTypeOutTradeNo NumberType = 1
  141. // NumberTypeTransactionID 使用微信支付单号
  142. NumberTypeTransactionID NumberType = 2
  143. )
  144. // GetShippingOrderRequest 查询订单发货状态参数
  145. type GetShippingOrderRequest struct {
  146. TransactionID string `json:"transaction_id"` // 原支付交易对应的微信订单号
  147. MerchantID string `json:"merchant_id"` // 支付下单商户的商户号,由微信支付生成并下发
  148. SubMerchantID string `json:"sub_merchant_id"` //二级商户号
  149. MerchantTradeNo string `json:"merchant_trade_no"` //商户系统内部订单号,只能是数字、大小写字母`_-*`且在同一个商户号下唯一。
  150. }
  151. // ShippingItem 物流信息
  152. type ShippingItem struct {
  153. TrackingNo string `json:"tracking_no"` // 物流单号,示例值: "323244567777
  154. ExpressCompany string `json:"express_company"` // 物流公司编码,快递公司ID,物流快递发货时必填;参见「查询物流公司编码列表」
  155. UploadTime int64 `json:"upload_time"` // 上传物流信息时间,时间戳形式
  156. }
  157. // ShippingDetail 发货信息
  158. type ShippingDetail struct {
  159. DeliveryMode DeliveryMode `json:"delivery_mode"` // 发货模式
  160. LogisticsType LogisticsType `json:"logistics_type"` // 物流模式
  161. FinishShipping bool `json:"finish_shipping"` // 是否已全部发货
  162. FinishShippingCount int `json:"finish_shipping_count"` // 已完成全部发货的次数
  163. GoodsDesc string `json:"goods_desc"` // 在小程序后台发货信息录入页录入的商品描述
  164. ShippingList []*ShippingItem `json:"shipping_list"` // 物流信息列表
  165. }
  166. // ShippingOrder 订单发货状态
  167. type ShippingOrder struct {
  168. TransactionID string `json:"transaction_id"` // 原支付交易对应的微信订单号
  169. MerchantTradeNo string `json:"merchant_trade_no"` // 商户系统内部订单号,只能是数字、大小写字母`_-*`且在同一个商户号下唯一
  170. MerchantID string `json:"merchant_id"` // 支付下单商户的商户号,由微信支付生成并下发
  171. SubMerchantID string `json:"sub_merchant_id"` // 二级商户号
  172. Description string `json:"description"` // 以分号连接的该支付单的所有商品描述,当超过120字时自动截断并以 “...” 结尾
  173. PaidAmount int64 `json:"paid_amount"` // 支付单实际支付金额,整型,单位:分钱
  174. Openid string `json:"openid"` // 支付者openid
  175. TradeCreateTime int64 `json:"trade_create_time"` // 交易创建时间,时间戳形式
  176. PayTime int64 `json:"pay_time"` // 支付时间,时间戳形式
  177. InComplaint bool `json:"in_complaint"` // 是否处在交易纠纷中
  178. OrderState State `json:"order_state"` // 订单状态枚举:(1) 待发货;(2) 已发货;(3) 确认收货;(4) 交易完成;(5) 已退款
  179. Shipping *ShippingDetail `json:"shipping"` // 订单发货信息
  180. }
  181. // ShippingOrderResponse 查询订单发货状态返回参数
  182. type ShippingOrderResponse struct {
  183. util.CommonError
  184. Order ShippingOrder `json:"order"` // 订单发货信息
  185. }
  186. // State 订单状态
  187. type State uint8
  188. const (
  189. // StateWaitShipment 待发货
  190. StateWaitShipment State = 1
  191. // StateShipped 已发货
  192. StateShipped State = 2
  193. // StateConfirm 确认收货
  194. StateConfirm State = 3
  195. // StateComplete 交易完成
  196. StateComplete State = 4
  197. // StateRefund 已退款
  198. StateRefund State = 5
  199. )
  200. // GetShippingOrderListRequest 查询订单列表请求参数
  201. type GetShippingOrderListRequest struct {
  202. PayTimeRange *TimeRange `json:"pay_time_range"` // 支付时间范围
  203. OrderState State `json:"order_state,omitempty"` // 订单状态
  204. Openid string `json:"openid,omitempty"` // 支付者openid
  205. LastIndex string `json:"last_index,omitempty"` // 翻页时使用,获取第一页时不用传入,如果查询结果中 has_more 字段为 true,则传入该次查询结果中返回的 last_index 字段可获取下一页
  206. PageSize int64 `json:"page_size"` // 每页数量,最多50条
  207. }
  208. // TimeRange 时间范围
  209. type TimeRange struct {
  210. BeginTime int64 `json:"begin_time,omitempty"` // 查询开始时间,时间戳形式
  211. EndTime int64 `json:"end_time,omitempty"` // 查询结束时间,时间戳形式
  212. }
  213. // GetShippingOrderListResponse 查询订单列表返回参数
  214. type GetShippingOrderListResponse struct {
  215. util.CommonError
  216. OrderList []*ShippingOrder `json:"order_list"`
  217. LastIndex string `json:"last_index"`
  218. HasMore bool `json:"has_more"`
  219. }
  220. // NotifyConfirmReceiveRequest 确认收货提醒接口请求参数
  221. type NotifyConfirmReceiveRequest struct {
  222. TransactionID string `json:"transaction_id"` // 原支付交易对应的微信订单号
  223. MerchantID string `json:"merchant_id"` // 支付下单商户的商户号,由微信支付生成并下发
  224. SubMerchantID string `json:"sub_merchant_id"` // 二级商户号
  225. MerchantTradeNo string `json:"merchant_trade_no"` // 商户系统内部订单号,只能是数字、大小写字母`_-*`且在同一个商户号下唯一
  226. ReceivedTime int64 `json:"received_time"` // 收货时间,时间戳形式
  227. }