合单下单-NATIVE
更新时间:2024.11.14使用合单支付接口,用户只输入一次密码,即可完成多个订单的支付。目前最多一次可支持50笔订单进行合单支付。
订单如果需要进行抽佣等,需要在合单中指定需要进行分账(profit_sharing为true);指定后,交易资金进入二级商户账户,处于冻结状态,可在后续使用分账接口进行分账,利用分账完结进行资金解冻,实现抽佣和对二级商户的账期。
接口说明
支持商户:【普通服务商】【平台商户】
请求方式:【POST】/v3/combine-transactions/native
请求域名:【主域名】https://api.mch.weixin.qq.com 使用该域名将访问就近的接入点
【备域名】https://api2.mch.weixin.qq.com 使用该域名将访问异地的接入点 ,指引点击查看
请求参数
Header HTTP头参数
Authorization 必填 string
请参考签名认证生成认证信息
Accept 必填 string
请设置为application/json
Content-Type 必填 string
请设置为application/json
body 包体参数
combine_appid 必填 string(32)
【合单商户Appid】 合单发起方的Appid
combine_out_trade_no 必填 string(32)
【合单商户订单号】 合单支付总订单号,最短2个字符,最长32个字符,只能是数字、大小写字母,以及_-|* ,且在同一个商户号下唯一
combine_mchid 必填 string(32)
【合单商户号】 合单发起方商户号
scene_info 选填 object
【场景信息】 场景信息
属性 | |
device_id 选填 string(16) 【商户端设备号】 终端设备号(门店号或收银设备ID) payer_client_ip 必填 string(45) 【用户终端IP】 用户端实际IP,支持IPv4和IPv6两种格式的IP地址。 |
sub_orders 必填 array[ReqSubOrder]
【子单信息】 子单列表,最多支持子单条数:50。
属性 | |||||||||
mchid 必填 string(32) 【子单商户号】 子单发起方商户号,与发起方Appid有绑定关系 attach 必填 string(128) 【附加数据】 附加数据,在查询API和支付通知中原样返回,可作为自定义参数使用。 amount 必填 object 【订单金额】 订单金额,指定面值和币种。
out_trade_no 必填 string(32) 【子单商户订单号】 商户系统内部订单号,最短2个字符,最长32个字符,只能是数字、大小写字母_-|* ,且在同一个商户号下唯一。 sub_mchid 选填 string(32) 【特约商户商户号】 特约商户商户号 description 必填 string(128) 【商品描述】 商品简单描述。需传入应用市场上的APP名字-实际商品名称,天天爱消除-游戏充值 settle_info 选填 object 【结算信息】 结算信息
sub_appid 选填 string 【子商户绑定的Appid】 服务商模式下,sub_mchid对应的sub_appid goods_tag 选填 string(32) 【订单优惠标记】 订单优惠标记,使用代金券或立减优惠功能时需要的参数,说明详见代金券或立减优惠 |
time_start 选填 string
【交易起始时间】 订单生效时间,按照rfc3339格式,格式为yyyy-MM-DDTHH:mm:ss+TIMEZONE,yyyy-MM-DD表示年月日,T出现在字符串中,表示time元素的开头,HH:mm:ss表示时分秒,TIMEZONE表示时区(+08:00表示东八区时间,领先UTC8小时,即北京时间)如 2018-06-08T10:34:56+08:00 代表,北京时间2018年06月08日10时34分56秒。
time_expire 选填 string
【交易结束时间】 订单失效时间,按照rfc3339格式,格式为yyyy-MM-DDTHH:mm:ss+TIMEZONE,yyyy-MM-DD表示年月日,T出现在字符串中,表示time元素的开头,HH:mm:ss表示时分秒,TIMEZONE表示时区(+08:00表示东八区时间,领先UTC8小时,即北京时间)如 2018-06-08T10:34:56+08:00 代表,北京时间2018年06月08日10时34分56秒。
notify_url 必填 string
【通知地址】 接收微信支付异步通知回调地址,通知url必须为直接可访问的url,不能携带参数。必须使用https协议。
请求示例
POST
合单支付-Native合单下单-请求示例
1curl -X POST \ 2 https://api.mch.weixin.qq.com/v3/combine-transactions/native \ 3 -H "Authorization: WECHATPAY2-SHA256-RSA2048 mchid=\"1900000001\",..." \ 4 -H "Accept: application/json" \ 5 -H "Content-Type: application/json" \ 6 -d '{ 7 "time_expire" : "2000-01-01T00:00:00+08:00", 8 "combine_appid" : "wxd678efh567hg6787", 9 "sub_orders" : [ 10 { 11 "mchid" : "1230000109", 12 "attach" : "深圳分店", 13 "amount" : { 14 "total_amount" : 10, 15 "currency" : "CNY" 16 }, 17 "out_trade_no" : "20150806125346", 18 "sub_mchid" : "1900000109", 19 "description" : "腾讯充值中心-QQ会员充值", 20 "settle_info" : { 21 "profit_sharing" : false 22 }, 23 "sub_appid" : "wxd678efh567hg6999", 24 "goods_tag" : "WXG" 25 } 26 ], 27 "combine_out_trade_no" : "20150806125346", 28 "notify_url" : "https://yourapp.com/notify", 29 "combine_mchid" : "1900000109", 30 "scene_info" : { 31 "device_id" : "POS1:1", 32 "payer_client_ip" : "14.17.22.32" 33 } 34 }' 35
需配合微信支付工具库 WXPayUtility 使用,请参考Java
1package com.java.demo; 2 3import com.java.utils.WXPayUtility; // 引用微信支付工具库,参考:https://pay.weixin.qq.com/doc/v3/partner/4014985777 4 5import com.google.gson.annotations.SerializedName; 6import com.google.gson.annotations.Expose; 7import okhttp3.MediaType; 8import okhttp3.OkHttpClient; 9import okhttp3.Request; 10import okhttp3.RequestBody; 11import okhttp3.Response; 12 13import java.io.IOException; 14import java.io.UncheckedIOException; 15import java.security.PrivateKey; 16import java.security.PublicKey; 17import java.util.ArrayList; 18import java.util.HashMap; 19import java.util.List; 20import java.util.Map; 21 22/** 23 * 合单下单-NATIVE 24 */ 25public class UnionNativePrepay { 26 private static String HOST = "https://api.mch.weixin.qq.com"; 27 private static String METHOD = "POST"; 28 private static String PATH = "/v3/combine-transactions/native"; 29 30 public static void main(String[] args) { 31 // TODO: 请准备商户开发必要参数,参考:https://pay.weixin.qq.com/doc/v3/partner/4013080340 32 UnionNativePrepay client = new UnionNativePrepay( 33 "19xxxxxxxx", // 商户号,是由微信支付系统生成并分配给每个商户的唯一标识符,商户号获取方式参考 https://pay.weixin.qq.com/doc/v3/partner/4013080340 34 "1DDE55AD98Exxxxxxxxxx", // 商户API证书序列号,如何获取请参考 https://pay.weixin.qq.com/doc/v3/partner/4013058924 35 "/path/to/apiclient_key.pem", // 商户API证书私钥文件路径,本地文件路径 36 "PUB_KEY_ID_xxxxxxxxxxxxx", // 微信支付公钥ID,如何获取请参考 https://pay.weixin.qq.com/doc/v3/partner/4013038589 37 "/path/to/wxp_pub.pem" // 微信支付公钥文件路径,本地文件路径 38 ); 39 40 UnionAPIv3NativePrepayRequest request = new UnionAPIv3NativePrepayRequest(); 41 request.combineAppid = "wxd678efh567hg6787"; 42 request.combineOutTradeNo = "20150806125346"; 43 request.combineMchid = "1900000109"; 44 request.sceneInfo = new UnionSceneInfo(); 45 request.sceneInfo.deviceId = "POS1:1"; 46 request.sceneInfo.payerClientIp = "14.17.22.32"; 47 request.subOrders = new ArrayList<>(); 48 { 49 UnionCommonSubOrder subOrdersItem0 = new UnionCommonSubOrder(); 50 subOrdersItem0.mchid = "1230000109"; 51 subOrdersItem0.subMchid = "1900000109"; 52 subOrdersItem0.subAppid = "wxd678efh567hg6999"; 53 subOrdersItem0.outTradeNo = "20150806125346"; 54 subOrdersItem0.amount = new UnionAmountInfo(); 55 subOrdersItem0.amount.totalAmount = 10L; 56 subOrdersItem0.amount.currency = "CNY"; 57 subOrdersItem0.attach = "深圳分店"; 58 subOrdersItem0.description = "腾讯充值中心-QQ会员充值"; 59 subOrdersItem0.goodsTag = "WXG"; 60 subOrdersItem0.settleInfo = new UnionSettleInfo(); 61 subOrdersItem0.settleInfo.profitSharing = false; 62 request.subOrders.add(subOrdersItem0); 63 }; 64 request.timeExpire = "2000-01-01T00:00:00+08:00"; 65 request.notifyUrl = "https://yourapp.com/notify"; 66 try { 67 UnionAPIv3NativePrepayResponse response = client.run(request); 68 // TODO: 请求成功,继续业务逻辑 69 System.out.println(response); 70 } catch (WXPayUtility.ApiException e) { 71 // TODO: 请求失败,根据状态码执行不同的逻辑 72 e.printStackTrace(); 73 } 74 } 75 76 public UnionAPIv3NativePrepayResponse run(UnionAPIv3NativePrepayRequest request) { 77 String uri = PATH; 78 String reqBody = WXPayUtility.toJson(request); 79 80 Request.Builder reqBuilder = new Request.Builder().url(HOST + uri); 81 reqBuilder.addHeader("Accept", "application/json"); 82 reqBuilder.addHeader("Wechatpay-Serial", wechatPayPublicKeyId); 83 reqBuilder.addHeader("Authorization", WXPayUtility.buildAuthorization(mchid, certificateSerialNo,privateKey, METHOD, uri, reqBody)); 84 reqBuilder.addHeader("Content-Type", "application/json"); 85 RequestBody requestBody = RequestBody.create(MediaType.parse("application/json; charset=utf-8"), reqBody); 86 reqBuilder.method(METHOD, requestBody); 87 Request httpRequest = reqBuilder.build(); 88 89 // 发送HTTP请求 90 OkHttpClient client = new OkHttpClient.Builder().build(); 91 try (Response httpResponse = client.newCall(httpRequest).execute()) { 92 String respBody = WXPayUtility.extractBody(httpResponse); 93 if (httpResponse.code() >= 200 && httpResponse.code() < 300) { 94 // 2XX 成功,验证应答签名 95 WXPayUtility.validateResponse(this.wechatPayPublicKeyId, this.wechatPayPublicKey, 96 httpResponse.headers(), respBody); 97 98 // 从HTTP应答报文构建返回数据 99 return WXPayUtility.fromJson(respBody, UnionAPIv3NativePrepayResponse.class); 100 } else { 101 throw new WXPayUtility.ApiException(httpResponse.code(), respBody, httpResponse.headers()); 102 } 103 } catch (IOException e) { 104 throw new UncheckedIOException("Sending request to " + uri + " failed.", e); 105 } 106 } 107 108 private final String mchid; 109 private final String certificateSerialNo; 110 private final PrivateKey privateKey; 111 private final String wechatPayPublicKeyId; 112 private final PublicKey wechatPayPublicKey; 113 114 public UnionNativePrepay(String mchid, String certificateSerialNo, String privateKeyFilePath, String wechatPayPublicKeyId, String wechatPayPublicKeyFilePath) { 115 this.mchid = mchid; 116 this.certificateSerialNo = certificateSerialNo; 117 this.privateKey = WXPayUtility.loadPrivateKeyFromPath(privateKeyFilePath); 118 this.wechatPayPublicKeyId = wechatPayPublicKeyId; 119 this.wechatPayPublicKey = WXPayUtility.loadPublicKeyFromPath(wechatPayPublicKeyFilePath); 120 } 121 122 public static class UnionAPIv3NativePrepayRequest { 123 @SerializedName("combine_appid") 124 public String combineAppid; 125 126 @SerializedName("combine_out_trade_no") 127 public String combineOutTradeNo; 128 129 @SerializedName("combine_mchid") 130 public String combineMchid; 131 132 @SerializedName("scene_info") 133 public UnionSceneInfo sceneInfo; 134 135 @SerializedName("sub_orders") 136 public List<UnionCommonSubOrder> subOrders = new ArrayList<UnionCommonSubOrder>(); 137 138 @SerializedName("time_expire") 139 public String timeExpire; 140 141 @SerializedName("notify_url") 142 public String notifyUrl; 143 } 144 145 public static class UnionAPIv3NativePrepayResponse { 146 @SerializedName("code_url") 147 public String codeUrl; 148 } 149 150 public static class UnionSceneInfo { 151 @SerializedName("device_id") 152 public String deviceId; 153 154 @SerializedName("payer_client_ip") 155 public String payerClientIp; 156 } 157 158 public static class UnionCommonSubOrder { 159 @SerializedName("mchid") 160 public String mchid; 161 162 @SerializedName("sub_mchid") 163 public String subMchid; 164 165 @SerializedName("sub_appid") 166 public String subAppid; 167 168 @SerializedName("out_trade_no") 169 public String outTradeNo; 170 171 @SerializedName("amount") 172 public UnionAmountInfo amount; 173 174 @SerializedName("attach") 175 public String attach; 176 177 @SerializedName("description") 178 public String description; 179 180 @SerializedName("goods_tag") 181 public String goodsTag; 182 183 @SerializedName("settle_info") 184 public UnionSettleInfo settleInfo; 185 } 186 187 public static class UnionAmountInfo { 188 @SerializedName("total_amount") 189 public Long totalAmount; 190 191 @SerializedName("currency") 192 public String currency; 193 } 194 195 public static class UnionSettleInfo { 196 @SerializedName("profit_sharing") 197 public Boolean profitSharing; 198 } 199 200} 201
需配合微信支付工具库 wxpay_utility 使用,请参考Go
1package main 2 3import ( 4 "bytes" 5 "demo/wxpay_utility" // 引用微信支付工具库,参考 https://pay.weixin.qq.com/doc/v3/partner/4015119446 6 "encoding/json" 7 "fmt" 8 "net/http" 9 "net/url" 10 "time" 11) 12 13func main() { 14 // TODO: 请准备商户开发必要参数,参考:https://pay.weixin.qq.com/doc/v3/partner/4013080340 15 config, err := wxpay_utility.CreateMchConfig( 16 "19xxxxxxxx", // 商户号,是由微信支付系统生成并分配给每个商户的唯一标识符,商户号获取方式参考 https://pay.weixin.qq.com/doc/v3/partner/4013080340 17 "1DDE55AD98Exxxxxxxxxx", // 商户API证书序列号,如何获取请参考 https://pay.weixin.qq.com/doc/v3/partner/4013058924 18 "/path/to/apiclient_key.pem", // 商户API证书私钥文件路径,本地文件路径 19 "PUB_KEY_ID_xxxxxxxxxxxxx", // 微信支付公钥ID,如何获取请参考 https://pay.weixin.qq.com/doc/v3/partner/4013038589 20 "/path/to/wxp_pub.pem", // 微信支付公钥文件路径,本地文件路径 21 ) 22 if err != nil { 23 fmt.Println(err) 24 return 25 } 26 27 request := &UnionApiv3NativePrepayRequest{ 28 CombineAppid: wxpay_utility.String("wxd678efh567hg6787"), 29 CombineOutTradeNo: wxpay_utility.String("20150806125346"), 30 CombineMchid: wxpay_utility.String("1900000109"), 31 SceneInfo: &UnionSceneInfo{ 32 DeviceId: wxpay_utility.String("POS1:1"), 33 PayerClientIp: wxpay_utility.String("14.17.22.32"), 34 }, 35 SubOrders: []UnionCommonSubOrder{ 36 UnionCommonSubOrder{ 37 Mchid: wxpay_utility.String("1230000109"), 38 SubMchid: wxpay_utility.String("1900000109"), 39 SubAppid: wxpay_utility.String("wxd678efh567hg6999"), 40 OutTradeNo: wxpay_utility.String("20150806125346"), 41 Amount: &UnionAmountInfo{ 42 TotalAmount: wxpay_utility.Int64(10), 43 Currency: wxpay_utility.String("CNY"), 44 }, 45 Attach: wxpay_utility.String("深圳分店"), 46 Description: wxpay_utility.String("腾讯充值中心-QQ会员充值"), 47 GoodsTag: wxpay_utility.String("WXG"), 48 SettleInfo: &UnionSettleInfo{ 49 ProfitSharing: wxpay_utility.Bool(false), 50 }, 51 }, 52 }, 53 TimeExpire: wxpay_utility.Time(time.Now()), 54 NotifyUrl: wxpay_utility.String("https://yourapp.com/notify"), 55 } 56 57 response, err := UnionNativePrepay(config, request) 58 if err != nil { 59 fmt.Printf("请求失败: %+v\n", err) 60 // TODO: 请求失败,根据状态码执行不同的处理 61 return 62 } 63 64 // TODO: 请求成功,继续业务逻辑 65 fmt.Printf("请求成功: %+v\n", response) 66} 67 68func UnionNativePrepay(config *wxpay_utility.MchConfig, request *UnionApiv3NativePrepayRequest) (response *UnionApiv3NativePrepayResponse, err error) { 69 const ( 70 host = "https://api.mch.weixin.qq.com" 71 method = "POST" 72 path = "/v3/combine-transactions/native" 73 ) 74 75 reqUrl, err := url.Parse(fmt.Sprintf("%s%s", host, path)) 76 if err != nil { 77 return nil, err 78 } 79 reqBody, err := json.Marshal(request) 80 if err != nil { 81 return nil, err 82 } 83 httpRequest, err := http.NewRequest(method, reqUrl.String(), bytes.NewReader(reqBody)) 84 if err != nil { 85 return nil, err 86 } 87 httpRequest.Header.Set("Accept", "application/json") 88 httpRequest.Header.Set("Wechatpay-Serial", config.WechatPayPublicKeyId()) 89 httpRequest.Header.Set("Content-Type", "application/json") 90 authorization, err := wxpay_utility.BuildAuthorization(config.MchId(), config.CertificateSerialNo(), config.PrivateKey(), method, reqUrl.RequestURI(), reqBody) 91 if err != nil { 92 return nil, err 93 } 94 httpRequest.Header.Set("Authorization", authorization) 95 96 client := &http.Client{} 97 httpResponse, err := client.Do(httpRequest) 98 if err != nil { 99 return nil, err 100 } 101 respBody, err := wxpay_utility.ExtractResponseBody(httpResponse) 102 if err != nil { 103 return nil, err 104 } 105 if httpResponse.StatusCode >= 200 && httpResponse.StatusCode < 300 { 106 // 2XX 成功,验证应答签名 107 err = wxpay_utility.ValidateResponse( 108 config.WechatPayPublicKeyId(), 109 config.WechatPayPublicKey(), 110 &httpResponse.Header, 111 respBody, 112 ) 113 if err != nil { 114 return nil, err 115 } 116 response := &UnionApiv3NativePrepayResponse{} 117 if err := json.Unmarshal(respBody, response); err != nil { 118 return nil, err 119 } 120 121 return response, nil 122 } else { 123 return nil, wxpay_utility.NewApiException( 124 httpResponse.StatusCode, 125 httpResponse.Header, 126 respBody, 127 ) 128 } 129} 130 131type UnionApiv3NativePrepayRequest struct { 132 CombineAppid *string `json:"combine_appid,omitempty"` 133 CombineOutTradeNo *string `json:"combine_out_trade_no,omitempty"` 134 CombineMchid *string `json:"combine_mchid,omitempty"` 135 SceneInfo *UnionSceneInfo `json:"scene_info,omitempty"` 136 SubOrders []UnionCommonSubOrder `json:"sub_orders,omitempty"` 137 TimeExpire *time.Time `json:"time_expire,omitempty"` 138 NotifyUrl *string `json:"notify_url,omitempty"` 139} 140 141type UnionApiv3NativePrepayResponse struct { 142 CodeUrl *string `json:"code_url,omitempty"` 143} 144 145type UnionSceneInfo struct { 146 DeviceId *string `json:"device_id,omitempty"` 147 PayerClientIp *string `json:"payer_client_ip,omitempty"` 148} 149 150type UnionCommonSubOrder struct { 151 Mchid *string `json:"mchid,omitempty"` 152 SubMchid *string `json:"sub_mchid,omitempty"` 153 SubAppid *string `json:"sub_appid,omitempty"` 154 OutTradeNo *string `json:"out_trade_no,omitempty"` 155 Amount *UnionAmountInfo `json:"amount,omitempty"` 156 Attach *string `json:"attach,omitempty"` 157 Description *string `json:"description,omitempty"` 158 GoodsTag *string `json:"goods_tag,omitempty"` 159 SettleInfo *UnionSettleInfo `json:"settle_info,omitempty"` 160} 161 162type UnionAmountInfo struct { 163 TotalAmount *int64 `json:"total_amount,omitempty"` 164 Currency *string `json:"currency,omitempty"` 165} 166 167type UnionSettleInfo struct { 168 ProfitSharing *bool `json:"profit_sharing,omitempty"` 169} 170
应答参数
200 OK
code_url 必填 string(512)
【二维码链接】 二维码链接
应答示例
200 OK
1{ 2 "code_url" : "weixin://wxpay/bizpayurl/up?pr=NwY5Mz9&groupid=00" 3} 4
错误码
公共错误码
状态码 | 错误码 | 描述 | 解决方案 |
---|---|---|---|
400 | PARAM_ERROR | 参数错误 | 请根据错误提示正确传入参数 |
400 | INVALID_REQUEST | HTTP 请求不符合微信支付 APIv3 接口规则 | 请参阅 接口规则 |
401 | SIGN_ERROR | 验证不通过 | 请参阅 签名常见问题 |
500 | SYSTEM_ERROR | 系统异常,请稍后重试 | 请稍后重试 |
业务错误码
状态码 | 错误码 | 描述 | 解决方案 |
---|---|---|---|
202 | USERPAYING | 用户支付中,需要输入密码 | 等待5秒,然后调用被扫订单结果查询API,查询当前订单的不同状态,决定下一步的操作 |
400 | APPID_MCHID_NOT_MATCH | AppID和mch_id不匹配 | 请确认AppID和mch_id是否匹配 |
400 | INVALID_REQUEST | 无效请求 | 请根据接口返回的详细信息检查 |
400 | MCH_NOT_EXISTS | 商户号不存在 | 请检查商户号是否正确 |
400 | ORDER_CLOSED | 订单已关闭 | 当前订单已关闭,请重新下单 |
401 | SIGN_ERROR | 签名错误 | 请检查签名参数和方法是否都符合签名算法要求 |
403 | ACCOUNTERROR | 账号异常 | 用户账号异常,无需更多操作 |
403 | NOAUTH | 商户无权限 | 请商户前往申请此接口相关权限 |
403 | OUT_TRADE_NO_USED | 商户订单号重复 | 请核实商户订单号是否重复提交 |
403 | RULELIMIT | 业务规则限制 | 因业务规则限制请求频率,请查看接口返回的详细信息 |
403 | TRADE_ERROR | 交易错误 | 因业务原因交易失败,请查看接口返回的详细信息 |
404 | ORDERNOTEXIST | 订单不存在 | 请检查订单是否发起过交易 |
429 | FREQUENCY_LIMITED | 频率超限 | 请降低请求接口频率 |
500 | BANKERROR | 银行系统异常 | 银行系统异常,请用相同参数重新调用 |
500 | INVALID_TRANSACTIONID | 订单号非法 | 请检查微信支付订单号是否正确 |
500 | OPENID_MISMATCH | OpenID和AppID不匹配 | 请确认OpenID和AppID是否匹配 |
500 | SYSTEMERROR | 系统错误 | 系统异常,请用相同参数重新调用 |