1、概要
公众号是以微信用户的一个联系人形式存在的,支付是微信服务号的核心一环。
本篇主要介绍微信支付这一功能,避免大家再跳微信支付的坑。
1.1 关于Magicodes.WeChat.SDK
MAGICODES.WECHAT.SDK为心莱团队封装的轻量级微信SDK,现已全部开源,开源库地址为:https://github.com/xin-lai/Magicodes.WeChat.SDK
更多介绍,请关注后续博客。
2、微信公众号支付
用户已有商城网址,用户通过微信消息、微信扫描二维码、微信自定义菜单等操作在微信内打开网页时,可以调用微信支付完成下单购买流程。
2.1 支付流程
2.1.1 微信网页支付流程
备注:实际流程其实非常简单
① 用户支付之前,程序生成订单并调用统一下单API()
② 微信系统返回预付单信息
③ 根据信息生成JSAPI页面调用的支付参数并签名,jssdk调用
④ 用户支付,jsapi向微信系统发送请求
⑤ 微信系统返回支付结果给用户,同时异步发送结果给程序后台(程序没有收到通知,可以调用查询接口)
⑥ 支付完成,用户界面根据结果做相关页面跳转或提示处理,程序后台根据通知做订单状态变更等逻辑处理。
2.1.2 刷卡支付
后续更新
2.1.3 扫码支付
后续更新
2.1.4 app支付
后续更新
2.2 注意事项
2.3 开发实践
2.3.1 开发配置
1、设置测试目录
在微信公众平台设置。支付测试状态下,设置测试目录,测试人的微信号添加到白名单,发起支付的页面目录必须与设置的精确匹配。并将支付链接发到对应的公众号会话窗口中才能正常发起支付测试。注意正式目录一定不能与测试目录设置成一样,否则支付会出错。
友情提示:如果是使用测试目录的地址,一定要记得把个人测试微信号添加到白名单。
2、设置正式支付目录
根据图中栏目顺序进入修改栏目,勾选JSAPI网页支付开通该权限,并配置好支付授权目录,该目录必须是发起支付的页面的精确目录,子目录下无法正常调用支付。具体界面如图:
友情提示:注意红色框框里面的说明,一不小心会很容易进坑的。
2.3.2 开发程序
直接看代码吧
微信支付业务类
1 /// <summary> 2 /// 微信支付接口,官方API:https://mp.weixin.qq.com/paymch/readtemplate?t=mp/business/course2_tmpl&lang=zh_CN&token=25857919#4 3 4 /// </summary> 5 public class TenPayV3 : PayBase 6 7 { 8 public UnifiedorderResult Unifiedorder(UnifiedorderRequest model) 9 10 { 11 12 var url = "https://api.mch.weixin.qq.com/pay/unifiedorder"; 13 14 15 16 17 18 UnifiedorderResult result = null; 19 20 21 22 model.AppId = WeiChatConfig.AppId; 23 24 model.MchId = PayConfig.MchId; 25 26 if (model.NotifyUrl == null) 27 28 model.NotifyUrl = PayConfig.Notify; 29 30 Dictionary<string, string> dictionary = PayUtil.GetAuthors<UnifiedorderRequest>(model); 31 32 model.Sign = PayUtil.CreateMd5Sign(dictionary, PayConfig.TenPayKey);//生成Sign 33 34 Dictionary<string, string> dict = PayUtil.GetAuthors<UnifiedorderRequest>(model); 35 36 result = PostXML<UnifiedorderResult>(url, model); 37 38 return result; 39 40 } 41 42 43 44 /// <summary> 45 46 /// 订单查询接口 47 48 /// </summary> 49 50 /// <param name="data"></param> 51 52 /// <returns></returns> 53 54 public static string OrderQuery(string data) 55 56 { 57 58 var urlFormat = "https://api.mch.weixin.qq.com/pay/orderquery"; 59 60 61 62 var formDataBytes = data == null ? new byte[0] : Encoding.UTF8.GetBytes(data); 63 64 using (MemoryStream ms = new MemoryStream()) 65 66 { 67 68 ms.Write(formDataBytes, 0, formDataBytes.Length); 69 70 ms.Seek(0, SeekOrigin.Begin);//设置指针读取位置 71 72 return RequestUtility.HttpPost(urlFormat, null, ms); 73 74 } 75 76 } 77 78 79 80 /// <summary> 81 82 /// 关闭订单接口 83 84 /// </summary> 85 86 /// <param name="data">关闭订单需要post的xml数据</param> 87 88 /// <returns></returns> 89 90 public static string CloseOrder(string data) 91 92 { 93 94 var urlFormat = "https://api.mch.weixin.qq.com/pay/closeorder"; 95 96 97 98 var formDataBytes = data == null ? new byte[0] : Encoding.UTF8.GetBytes(data); 99 100 using (MemoryStream ms = new MemoryStream()) 101 102 { 103 104 ms.Write(formDataBytes, 0, formDataBytes.Length); 105 106 ms.Seek(0, SeekOrigin.Begin);//设置指针读取位置 107 108 return RequestUtility.HttpPost(urlFormat, null, ms); 109 110 } 111 112 } 113 114 115 116 117 118 119 120 /// <summary> 121 122 /// 退款查询接口 123 124 /// </summary> 125 126 /// <param name="data"></param> 127 128 /// <returns></returns> 129 130 public static string RefundQuery(string data) 131 132 { 133 134 var urlFormat = "https://api.mch.weixin.qq.com/pay/refundquery"; 135 136 137 138 var formDataBytes = data == null ? new byte[0] : Encoding.UTF8.GetBytes(data); 139 140 using (MemoryStream ms = new MemoryStream()) 141 142 { 143 144 ms.Write(formDataBytes, 0, formDataBytes.Length); 145 146 ms.Seek(0, SeekOrigin.Begin);//设置指针读取位置 147 148 return RequestUtility.HttpPost(urlFormat, null, ms); 149 150 } 151 152 } 153 154 155 156 /// <summary> 157 158 /// 对账单接口 159 160 /// </summary> 161 162 /// <param name="data"></param> 163 164 /// <returns></returns> 165 166 public static string DownloadBill(string data) 167 168 { 169 170 var urlFormat = "https://api.mch.weixin.qq.com/pay/downloadbill"; 171 172 173 174 var formDataBytes = data == null ? new byte[0] : Encoding.UTF8.GetBytes(data); 175 176 using (MemoryStream ms = new MemoryStream()) 177 178 { 179 180 ms.Write(formDataBytes, 0, formDataBytes.Length); 181 182 ms.Seek(0, SeekOrigin.Begin);//设置指针读取位置 183 184 return RequestUtility.HttpPost(urlFormat, null, ms); 185 186 } 187 188 } 189 190 191 192 /// <summary> 193 194 /// 短链接转换接口 195 196 /// </summary> 197 198 /// <param name="data"></param> 199 200 /// <returns></returns> 201 202 public static string ShortUrl(string data) 203 204 { 205 206 var urlFormat = "https://api.mch.weixin.qq.com/tools/shorturl"; 207 208 209 210 var formDataBytes = data == null ? new byte[0] : Encoding.UTF8.GetBytes(data); 211 212 using (MemoryStream ms = new MemoryStream()) 213 214 { 215 216 ms.Write(formDataBytes, 0, formDataBytes.Length); 217 218 ms.Seek(0, SeekOrigin.Begin);//设置指针读取位置 219 220 return RequestUtility.HttpPost(urlFormat, null, ms); 221 222 } 223 224 } 225 226 /// <summary> 227 228 /// 229 230 /// </summary> 231 232 /// <param name="page"></param> 233 234 /// <returns></returns> 235 236 public NotifyResult Notify(Stream inputStream) 237 238 { 239 240 NotifyResult result = null; 241 242 string data = PayUtil.PostInput(inputStream); 243 244 result = XmlHelper.DeserializeObject<NotifyResult>(data); 245 246 return result; 247 248 } 249 250 /// <summary> 251 252 /// 通知并返回处理XML 253 254 /// </summary> 255 256 /// <param name="inputStream">输入流</param> 257 258 /// <param name="successAction">成功处理逻辑回调函数</param> 259 260 /// <param name="failAction">失败处理逻辑回调函数</param> 261 262 /// <param name="successMsg">成功返回消息</param> 263 264 /// <param name="errorMsg">失败返回消息</param> 265 266 /// <param name="isSync">是否异步执行相关处理逻辑</param> 267 268 /// <returns></returns> 269 270 public string NotifyAndReurnResult(Stream inputStream, Action<NotifyResult> successAction, Action<NotifyResult> failAction, string successMsg = "OK", string errorMsg = "FAIL", bool isSync = true) 271 272 { 273 274 var result = Notify(inputStream); 275 276 var request = new NotifyRequest(); 277 278 request.ReturnCode = "FAIL"; 279 280 if (result.IsSuccess()) 281 282 { 283 284 if (isSync) 285 286 Task.Run(() => successAction(result)); 287 288 else 289 290 successAction.Invoke(result); 291 292 //交易成功 293 294 request.ReturnCode = "SUCCESS"; 295 296 request.ReturnMsg = successMsg; 297 298 return XmlHelper.SerializeObject(request); 299 300 } 301 302 else 303 304 { 305 306 if (isSync) 307 308 Task.Run(() => failAction(result)); 309 310 else 311 312 failAction.Invoke(result); 313 314 request.ReturnMsg = errorMsg; 315 316 return XmlHelper.SerializeObject(request); 317 318 } 319 320 } 321 322 } 323 324 }
把返回参数和请求参数,序列化成对象,方便我们在编写我们本身逻辑的时候调用
1 [XmlRoot("xml")] 2 [Serializable()] 3 public class Result : PayResult 4 { 5 /// <summary> 6 /// 微信分配的公众账号ID 7 /// </summary> 8 [XmlElement("appid")] 9 public string AppId { get; set; } 10 /// <summary> 11 /// 微信支付分配的商户号 12 /// </summary> 13 [XmlElement("mch_id")] 14 public string Mch_Id { get; set; } 15 /// <summary> 16 /// 微信支付分配的终端设备号 17 /// </summary> 18 [XmlElement("device_info")] 19 public string Device_Info { get; set; } 20 /// <summary> 21 /// 随机字符串,不长于32 位 22 /// </summary> 23 [XmlElement("nonce_str")] 24 public string NonceStr { get; set; } 25 /// <summary> 26 /// 签名 27 /// </summary> 28 [XmlElement("sign")] 29 public string Sign { get; set; } 30 /// <summary> 31 /// SUCCESS/FAIL 32 /// </summary> 33 [XmlElement("result_code")] 34 public string ResultCode { get; set; } 35 [XmlElement("err_code")] 36 public string ErrCode { get; set; } 37 [XmlElement("err_code_des")] 38 public string ErrCodeDes { get; set; } 39 } 40 41 [XmlRoot("xml")] 42 [Serializable()] 43 public class UnifiedorderResult : Result 44 { 45 /// <summary> 46 /// 交易类型:JSAPI、NATIVE、APP 47 /// </summary> 48 [XmlElement("trade_type")] 49 public string TradeType { get; set; } 50 /// <summary> 51 /// 微信生成的预支付ID,用于后续接口调用中使用 52 /// </summary> 53 [XmlElement("prepay_id")] 54 public string PrepayId { get; set; } 55 /// <summary> 56 /// trade_type为NATIVE时有返回,此参数可直接生成二维码展示出来进行扫码支付 57 /// </summary> 58 [XmlElement("code_url")] 59 public string CodeUrl { get; set; } 60 } 61 62 63 [XmlRoot("xml")] 64 [Serializable()] 65 public class UnifiedorderRequest 66 { 67 /// <summary> 68 /// OpenId 69 /// </summary> 70 [XmlElement("openid")] 71 public string OpenId { get; set; } 72 /// <summary> 73 /// 【不用填写】微信开放平台审核通过的应用APPID 74 /// </summary> 75 [XmlElement("appid")] 76 public string AppId { get; set; } 77 /// <summary> 78 /// 【不用填写】微信支付分配的商户号 79 /// </summary> 80 [XmlElement("mch_id")] 81 public string MchId { get; set; } 82 83 /// <summary> 84 /// 终端设备号(门店号或收银设备ID),默认请传"WEB" 85 /// </summary> 86 [XmlElement("device_info")] 87 public string DeviceInfo { get; set; } 88 /// <summary> 89 /// 随机字符串,不长于32位 90 /// </summary> 91 [XmlElement("nonce_str")] 92 public string NonceStr { get; set; } 93 /// <summary> 94 /// 【不用填写】签名 95 /// </summary> 96 [XmlElement("sign")] 97 public string Sign { get; set; } 98 /// <summary> 99 /// 商品描述交易字段格式根据不同的应用场景按照以下格式: APP——需传入应用市场上的APP名字-实际商品名称,天天爱消除-游戏充值。 100 /// </summary> 101 [XmlElement("body")] 102 public string Body { get; set; } 103 /// <summary> 104 /// 商品名称明细列表 105 /// </summary> 106 [XmlElement("detail")] 107 public string Detail { get; set; } 108 /// <summary> 109 /// 附加数据,在查询API和支付通知中原样返回,该字段主要用于商户携带订单的自定义数据 110 /// </summary> 111 [XmlElement("attach")] 112 public string Attach { get; set; } 113 /// <summary> 114 /// 商户系统内部的订单号,32个字符内、可包含字母, 其他说明见商户订单号 115 /// </summary> 116 [XmlElement("out_trade_no")] 117 public string OutTradeNo { get; set; } 118 /// <summary> 119 /// 符合ISO 4217标准的三位字母代码,默认人民币:CNY 120 /// </summary> 121 [XmlElement("fee_type")] 122 public string FeeType { get; set; } 123 /// <summary> 124 /// 订单总金额,单位为分 125 /// </summary> 126 [XmlElement("total_fee")] 127 public string TotalFee { get; set; } 128 /// <summary> 129 /// 用户端实际ip 130 /// </summary> 131 [XmlElement("spbill_create_ip")] 132 public string SpbillCreateIp { get; set; } 133 /// <summary> 134 /// 订单生成时间,格式为yyyyMMddHHmmss,如2009年12月25日9点10分10秒表示为20091225091010。 135 /// </summary> 136 [XmlElement("time_start")] 137 public string TimeStart { get; set; } 138 /// <summary> 139 /// 订单失效时间,格式为yyyyMMddHHmmss,如2009年12月27日9点10分10秒表示为20091227091010 140 /// </summary> 141 [XmlElement("time_expire")] 142 public string TimeExpire { get; set; } 143 /// <summary> 144 /// 商品标记,代金券或立减优惠功能的参数,说明详见代金券或立减优惠 145 /// </summary> 146 [XmlElement("goods_tag")] 147 public string GoodsTag { get; set; } 148 /// <summary> 149 /// 【不用填写】接收微信支付异步通知回调地址,通知url必须为直接可访问的url,不能携带参数 150 /// </summary> 151 [XmlElement("notify_url")] 152 public string NotifyUrl { get; set; } 153 /// <summary> 154 /// 支付类型(JSAPI,NATIVE,APP)公众号内支付填JSAPI 155 /// </summary> 156 [XmlElement("trade_type")] 157 public string TradeType { get; set; } 158 /// <summary> 159 /// no_credit--指定不能使用信用卡支付 160 /// </summary> 161 [XmlElement("limit_pay")] 162 public string LimitPay { get; set; } 163 } 164 [XmlRoot("xml")] 165 [Serializable()] 166 public class NotifyResult : PayResult 167 { 168 /// <summary> 169 /// 微信分配的公众账号ID(企业号corpid即为此appId) 170 /// </summary> 171 [XmlElement("appid")] 172 public string AppId { get; set; } 173 /// <summary> 174 /// 微信支付分配的商户号 175 /// </summary> 176 [XmlElement("mch_id")] 177 public string MchId { get; set; } 178 /// <summary> 179 /// 微信支付分配的终端设备号 180 /// </summary> 181 [XmlElement("device_info")] 182 public string DeviceInfo { get; set; } 183 /// <summary> 184 /// 随机字符串,不长于32位 185 /// </summary> 186 [XmlElement("nonce_str")] 187 public string NonceStr { get; set; } 188 /// <summary> 189 /// 签名 190 /// </summary> 191 [XmlElement("sign")] 192 public string Sign { get; set; } 193 /// <summary> 194 /// 业务结果,SUCCESS/FAIL 195 /// </summary> 196 [XmlElement("result_code")] 197 public string ResultCode { get; set; } 198 /// <summary> 199 /// 错误返回的信息描述 200 /// </summary> 201 [XmlElement("err_code")] 202 public string ErrCode { get; set; } 203 /// <summary> 204 /// 错误返回的信息描述 205 /// </summary> 206 [XmlElement("err_code_des")] 207 public string ErrCodeDes { get; set; } 208 /// <summary> 209 /// 用户在商户appid下的唯一标识 210 /// </summary> 211 [XmlElement("openid")] 212 public string OpenId { get; set; } 213 /// <summary> 214 /// 用户是否关注公众账号,Y-关注,N-未关注,仅在公众账号类型支付有效 215 /// </summary> 216 [XmlElement("is_subscribe")] 217 public string IsSubscribe { get; set; } 218 /// <summary> 219 /// 交易类型,JSAPI、NATIVE、APP 220 /// </summary> 221 [XmlElement("trade_type")] 222 public string TradeType { get; set; } 223 /// <summary> 224 /// 银行类型,采用字符串类型的银行标识,银行类型见银行列表 225 /// </summary> 226 [XmlElement("bank_type")] 227 public string BankType { get; set; } 228 /// <summary> 229 /// 订单总金额,单位为分 230 /// </summary> 231 [XmlElement("total_fee")] 232 public string TotalFee { get; set; } 233 /// <summary> 234 /// 应结订单金额=订单金额-非充值代金券金额,应结订单金额<=订单金额 235 /// </summary> 236 [XmlElement("settlement_total_fee")] 237 public string SettlementTotalFee { get; set; } 238 /// <summary> 239 /// 货币类型,符合ISO4217标准的三位字母代码,默认人民币:CNY,其他值列表详见货币类型 240 /// </summary> 241 [XmlElement("fee_type")] 242 public string FeeType { get; set; } 243 /// <summary> 244 /// 货币类型现金支付金额订单现金支付金额,详见支付金额 245 /// </summary> 246 [XmlElement("cash_fee")] 247 public string CashFee { get; set; } 248 /// <summary> 249 /// 货币类型,符合ISO4217标准的三位字母代码,默认人民币:CNY,其他值列表详见货币类型 250 /// </summary> 251 [XmlElement("cash_fee_type")] 252 public string CashFeeType { get; set; } 253 /// <summary> 254 /// 代金券金额<=订单金额,订单金额-代金券金额=现金支付金额,详见支付金额] 255 /// </summary> 256 [XmlElement("coupon_fee")] 257 public string CouponFee { get; set; } 258 /// <summary> 259 /// 代金券使用数量 260 /// </summary> 261 [XmlElement("coupon_count")] 262 public string CouponCount { get; set; } 263 /// <summary> 264 /// CASH--充值代金券 NO_CASH---非充值代金券 订单使用代金券时有返回(取值:CASH、NO_CASH)。$n为下标,从0开始编号,举例:coupon_type_$0 265 /// </summary> 266 [XmlElement("coupon_type_$n")] 267 public string CouponTypeN { get; set; } 268 /// <summary> 269 /// 代金券ID,$n为下标,从0开始编号 270 /// </summary> 271 [XmlElement("coupon_id_$n")] 272 public string CouponIdN { get; set; } 273 /// <summary> 274 /// 单个代金券支付金额,$n为下标,从0开始编号 275 /// </summary> 276 [XmlElement("coupon_fee_$n")] 277 public string CouponFeeN { get; set; } 278 /// <summary> 279 /// 微信支付订单号 280 /// </summary> 281 [XmlElement("transaction_id")] 282 public string TransactionId { get; set; } 283 /// <summary> 284 /// 商户系统的订单号,与请求一致 285 /// </summary> 286 [XmlElement("out_trade_no")] 287 public string OutTradeNo { get; set; } 288 /// <summary> 289 /// 商家数据包,原样返回 290 /// </summary> 291 [XmlElement("attach")] 292 public string Attach { get; set; } 293 /// <summary> 294 /// 支付完成时间,格式为yyyyMMddHHmmss,如2009年12月25日9点10分10秒表示为20091225091010。其他详见时间规则 295 /// </summary> 296 [XmlElement("time_end")] 297 public string TimeEnd { get; set; } 298 } 299 [XmlRoot("xml")] 300 [Serializable()] 301 public class NotifyRequest 302 { 303 /// <summary> 304 /// SUCCESS/FAIL SUCCESS表示商户接收通知成功并校验成功 305 /// </summary> 306 [XmlElement("return_code")] 307 public string ReturnCode { get; set; } 308 /// <summary> 309 /// 返回信息,如非空,为错误原因:签名失败 参数格式校验错误 310 /// </summary> 311 [XmlElement("return_msg")] 312 public string ReturnMsg { get; set; } 313 }
这些地方是获取微信支付的配置信息,appid,machid,密钥等信息。由于我们项目本身涉及到了多租户功能,所以Key是租户Id,这里通过Key获取不能租户的配置信息,如果不要实现多租户,默认不填写就OK,然后我们也可以通过把配置信息封装成一个对象写死在这里。具体源码可以去http:githup.com/xin-lai下载。全部功能已开源。
1 /// <summary> 2 3 /// POST提交请求,返回ApiResult对象 4 5 /// </summary> 6 7 /// <typeparam name="T">ApiResult对象</typeparam> 8 9 /// <param name="url">请求地址</param> 10 11 /// <param name="obj">提交的数据对象</param> 12 13 /// <returns>ApiResult对象</returns> 14 15 protected T PostXML<T>(string url, object obj, Func<string, string> serializeStrFunc = null) where T : PayResult 16 17 { 18 19 var wr = new WeChatApiWebRequestHelper(); 20 21 string resultStr = null; 22 23 var result = wr.HttpPost<T>(url, obj, out resultStr, serializeStrFunc, inputDataType: WebRequestDataTypes.XML, outDataType: WebRequestDataTypes.XML); 24 25 if (result != null) 26 27 { 28 29 result.DetailResult = resultStr; 30 31 } 32 33 return result; 34 35 } 36 37 /// <summary> 38 39 /// POST提交请求,带证书,返回ApiResult对象 40 41 /// </summary> 42 43 /// <typeparam name="T">ApiResult对象</typeparam> 44 45 /// <param name="url">请求地址</param> 46 47 /// <param name="obj">提交的数据对象</param> 48 49 /// <returns>ApiResult对象</returns> 50 51 protected T PostXML<T>(string url, object obj, X509Certificate2 cer, Func<string, string> serializeStrFunc = null) where T : PayResult 52 53 { 54 55 var wr = new WeChatApiWebRequestHelper(); 56 57 string resultStr = null; 58 59 var result = wr.HttpPost<T>(url, obj,cer, out resultStr, serializeStrFunc, inputDataType: WebRequestDataTypes.XML, outDataType: WebRequestDataTypes.XML); 60 61 if (result != null) 62 63 { 64 65 result.DetailResult = resultStr; 66 67 } 68 69 return result; 70 71 } 72 73 }
PayUtiy类,封装了一些公共方法
1 public static class PayUtil 2 3 { 4 5 /// <summary> 6 7 /// 随机生成Noncestr 8 9 /// </summary> 10 11 /// <returns></returns> 12 13 public static string GetNoncestr() 14 15 { 16 17 Random random = new Random(); 18 19 return MD5UtilHelper.GetMD5(random.Next(1000).ToString(), "GBK"); 20 21 } 22 23 /// <summary> 24 25 /// 根据当前系统时间加随机序列来生成订单号 26 27 /// </summary> 28 29 /// <returns>订单号</returns> 30 31 public static string GenerateOutTradeNo() 32 33 { 34 35 var ran = new Random(); 36 37 return string.Format("{0}{1}", UnixStamp(), ran.Next(999)); 38 39 } 40 41 /// <summary> 42 43 /// 获取时间戳 44 45 /// </summary> 46 47 /// <returns></returns> 48 49 public static string GetTimestamp() 50 51 { 52 53 TimeSpan ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0); 54 55 return Convert.ToInt64(ts.TotalSeconds).ToString(); 56 57 } 58 59 60 61 /// <summary> 62 63 /// 对字符串进行URL编码 64 65 /// </summary> 66 67 /// <param name="instr"></param> 68 69 /// <param name="charset"></param> 70 71 /// <returns></returns> 72 73 public static string UrlEncode(string instr, string charset) 74 75 { 76 77 //return instr; 78 79 if (instr == null || instr.Trim() == "") 80 81 return ""; 82 83 else 84 85 { 86 87 string res; 88 89 90 91 try 92 93 { 94 95 res = System.Web.HttpUtility.UrlEncode(instr, Encoding.GetEncoding(charset)); 96 97 98 99 } 100 101 catch (Exception ex) 102 103 { 104 105 res = System.Web.HttpUtility.UrlEncode(instr, Encoding.GetEncoding("GB2312")); 106 107 } 108 109 110 111 112 113 return res; 114 115 } 116 117 } 118 119 120 121 /// <summary> 122 123 /// 对字符串进行URL解码 124 125 /// </summary> 126 127 /// <param name="instr"></param> 128 129 /// <param name="charset"></param> 130 131 /// <returns></returns> 132 133 public static string UrlDecode(string instr, string charset) 134 135 { 136 137 if (instr == null || instr.Trim() == "") 138 139 return ""; 140 141 else 142 143 { 144 145 string res; 146 147 148 149 try 150 151 { 152 153 res = System.Web.HttpUtility.UrlDecode(instr, Encoding.GetEncoding(charset)); 154 155 156 157 } 158 159 catch (Exception ex) 160 161 { 162 163 res = System.Web.HttpUtility.UrlDecode(instr, Encoding.GetEncoding("GB2312")); 164 165 } 166 167 168 169 170 171 return res; 172 173 174 175 } 176 177 } 178 179 180 181 182 183 /// <summary> 184 185 /// 取时间戳生成随即数,替换交易单号中的后10位流水号 186 187 /// </summary> 188 189 /// <returns></returns> 190 191 public static UInt32 UnixStamp() 192 193 { 194 195 TimeSpan ts = DateTime.Now - TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)); 196 197 return Convert.ToUInt32(ts.TotalSeconds); 198 199 } 200 201 /// <summary> 202 203 /// 取随机数 204 205 /// </summary> 206 207 /// <param name="length"></param> 208 209 /// <returns></returns> 210 211 public static string BuildRandomStr(int length) 212 213 { 214 215 Random rand = new Random(); 216 217 218 219 int num = rand.Next(); 220 221 222 223 string str = num.ToString(); 224 225 226 227 if (str.Length > length) 228 229 { 230 231 str = str.Substring(0, length); 232 233 } 234 235 else if (str.Length < length) 236 237 { 238 239 int n = length - str.Length; 240 241 while (n > 0) 242 243 { 244 245 str.Insert(0, "0"); 246 247 n--; 248 249 } 250 251 } 252 253 254 255 return str; 256 257 } 258 259 /// <summary> 260 261 /// 循环获取一个实体类每个字段的XmlAttribute属性的值 262 263 /// </summary> 264 265 /// <typeparam name="T"></typeparam> 266 267 /// <returns></returns> 268 269 public static Dictionary<string, string> GetAuthors<T>(T model) 270 271 { 272 273 Dictionary<string, string> _dict = new Dictionary<string, string>(); 274 275 276 277 Type type = model.GetType(); //获取类型 278 279 280 281 PropertyInfo[] props = typeof(T).GetProperties(); 282 283 foreach (PropertyInfo prop in props) 284 285 { 286 287 object[] attrs = prop.GetCustomAttributes(true); 288 289 foreach (object attr in attrs) 290 291 { 292 293 XmlElementAttribute authAttr = attr as XmlElementAttribute; 294 295 if (authAttr != null) 296 297 { 298 299 string auth = authAttr.ElementName; 300 301 302 303 PropertyInfo property = type.GetProperty(prop.Name); 304 305 string value = (string)property.GetValue(model, null); //获取属性值 306 307 308 309 _dict.Add(auth, value); 310 311 } 312 313 } 314 315 } 316 317 return _dict; 318 319 } 320 321 322 323 /// <summary> 324 325 /// 创建md5摘要,规则是:按参数名称a-z排序,遇到空值的参数不参加签名 326 327 /// </summary> 328 329 /// <param name="key">参数名</param> 330 331 /// <param name="value">参数值</param> 332 333 /// key和value通常用于填充最后一组参数 334 335 /// <returns></returns> 336 337 public static string CreateMd5Sign(Dictionary<string, string> dict, string value) 338 339 { 340 341 ArrayList akeys = new ArrayList(); 342 343 foreach (var x in dict) 344 345 { 346 347 if ("sign".CompareTo(x.Key) == 0) 348 349 continue; 350 351 akeys.Add(x.Key); 352 353 } 354 355 StringBuilder sb = new StringBuilder(); 356 357 akeys.Sort(); 358 359 360 361 foreach (string k in akeys) 362 363 { 364 365 string v = (string)dict[k]; 366 367 if (null != v && "".CompareTo(v) != 0 368 369 && "sign".CompareTo(k) != 0 && "key".CompareTo(k) != 0) 370 371 { 372 373 sb.Append(k + "=" + v + "&"); 374 375 } 376 377 } 378 379 sb.Append("key=" + value); 380 381 var md5 = MD5.Create(); 382 383 var bs = md5.ComputeHash(Encoding.UTF8.GetBytes(sb.ToString())); 384 385 var sbuilder = new StringBuilder(); 386 387 foreach (byte b in bs) 388 389 { 390 391 sbuilder.Append(b.ToString("x2")); 392 393 } 394 395 //所有字符转为大写 396 397 return sbuilder.ToString().ToUpper(); 398 399 } 400 401 /// <summary> 402 403 /// 接收post数据 404 405 /// </summary> 406 407 /// <param name="context"></param> 408 409 /// <returns></returns> 410 411 public static string PostInput(Stream stream) 412 413 { 414 415 int count = 0; 416 417 byte[] buffer = new byte[1024]; 418 419 StringBuilder builder = new StringBuilder(); 420 421 while ((count = stream.Read(buffer, 0, 1024)) > 0) 422 423 { 424 425 builder.Append(Encoding.UTF8.GetString(buffer, 0, count)); 426 427 } 428 429 return builder.ToString(); 430 431 } 432 433 } 434 435 PayResult类,请求参数基类 436 437 [XmlRoot("xml")] 438 439 [Serializable()] 440 441 public class PayResult 442 443 { 444 445 public virtual bool IsSuccess() 446 447 { 448 449 return this.ReturnCode == "SUCCESS"; 450 451 } 452 453 /// <summary> 454 455 /// 返回状态码 456 457 /// SUCCESS/FAIL,此字段是通信标识,非交易标识,交易是否成功需要查看result_code来判断 458 459 /// </summary> 460 461 [XmlElement("return_code")] 462 463 public string ReturnCode { get; set; } 464 465 466 467 /// <summary> 468 469 /// 返回信息,返回信息,如非空,为错误原因,签名失败,参数格式校验错误 470 471 /// </summary> 472 473 [XmlElement("return_msg")] 474 475 public string Message { get; set; } 476 477 478 479 /// <summary> 480 481 /// 详细内容 482 483 /// </summary> 484 485 [XmlIgnore] 486 487 public string DetailResult { get; set; } 488 489 }
支付完成之后,异步回掉的处理
1 /// <summary> 2 3 /// 微信支付回调地址 4 5 /// </summary> 6 7 /// <param name="tenantId"></param> 8 9 /// <returns></returns> 10 11 [Route("PayNotify/{tenantId}")] 12 13 [AllowAnonymous] 14 15 public ActionResult PayNotify(int tenantId) 16 17 { 18 19 Action<NotifyResult> successAction = (result) => 20 21 { 22 23 using (var context = new AppDbContext()) 24 25 { 26 27 var order = context.Order_Infos.FirstOrDefault(o => o.Code == result.OutTradeNo); 28 29 if (null != order) 30 31 { 32 33 //修改订单状态 34 35 order.State = EnumOrderStatus.Overhang; 36 37 order.ThirdPayType = EnumThirdPayType.WX; 38 39 order.PaymentOn = DateTime.Now; 40 41 order.UpdateTime = DateTime.Now; 42 43 context.SaveChanges(); 44 45 46 47 48 49 } 50 51 else 52 53 { 54 55 logger.Log(LoggerLevels.Debug, "Order information does not exist!"); 56 57 } 58 59 } 60 61 //此处编写成功处理逻辑 62 63 logger.Log(LoggerLevels.Debug, "Success: JSON:" + JsonConvert.SerializeObject(result)); 64 65 }; 66 67 Action<NotifyResult> failAction = (result) => 68 69 { 70 71 //此处编写失败处理逻辑 72 73 logger.Log(LoggerLevels.Debug, "Fail: JSON:" + JsonConvert.SerializeObject(result)); 74 75 }; 76 77 return Content(WeiChatApisContext.Current.TenPayV3Api.NotifyAndReurnResult(Request.InputStream, successAction, failAction)); 78 79 }
当前是mvc项目,所以我建了一个webapi控制器,路由地址是http://xx.com/api/ PayNotify/{tenantId},tenantId是租户id,非必填,但是如果没有,要记得把路由里面的参数也去掉。这个地址就是本文前面配置的回调地址,一定要配置正确
然后是调用微信的统一下单的方法
1 /// <summary> 2 3 /// 微信支付(统一下单) 4 5 /// </summary> 6 7 /// <param name="id"></param> 8 9 /// <returns></returns> 10 11 [HttpGet] 12 13 [Route("Pay/{id}")] 14 15 public IHttpActionResult WechatPay(Guid id) 16 17 { 18 19 try 20 21 { 22 23 //查询订单 24 25 var order = db.Order_Infos.SingleOrDefault(o => o.Id == id && o.OpenId == WeiChatApplicationContext.Current.WeiChatUser.OpenId); 26 27 if (null == order) 28 29 return BadRequest("订单信息不存在"); 30 31 #region 统一下单 32 33 var model = new UnifiedorderRequest(); 34 35 model.OpenId = WeiChatApplicationContext.Current.WeiChatUser.OpenId; 36 37 model.SpbillCreateIp = "8.8.8.8"; 38 39 model.OutTradeNo = order.Code; 40 41 model.TotalFee = Convert.ToInt32((order.TotalPrice + order.Shipping) * 100).ToString(); 42 43 model.NonceStr = PayUtil.GetNoncestr(); 44 45 model.TradeType = "JSAPI"; 46 47 model.Body = "购买商品"; 48 49 model.DeviceInfo = "WEB"; 50 51 var result = WeiChatApisContext.Current.TenPayV3Api.Unifiedorder(model); 52 53 54 55 Dictionary<string, string> _dict = new Dictionary<string, string>(); 56 57 _dict.Add("appId", result.AppId); 58 59 _dict.Add("timeStamp", PayUtil.GetTimestamp()); 60 61 _dict.Add("nonceStr", result.NonceStr); 62 63 _dict.Add("package", "prepay_id=" + result.PrepayId); 64 65 _dict.Add("signType", "MD5"); 66 67 _dict.Add("paySign", PayUtil.CreateMd5Sign(_dict, WeiChatConfigManager.Current.GetPayConfig().TenPayKey)); 68 69 #endregion 70 71 return Ok(_dict); 72 73 } 74 75 catch (Exception ex) 76 77 { 78 79 log.Log(LoggerLevels.Error, "WechatPay:" + ex.Message); 80 81 } 82 83 return BadRequest("操作失败,请联系管理员!"); 84 85 }
这也是一个webapi或mvc控制器,给前台页面调用,作用是通过微信的统一下单方法获取微信的预付单,然后再生成一个供给jssdk调用的对象。
页面上的jssdk调用方法
1 function onBridgeReady(data) { 2 3 WeixinJSBridge.invoke('getBrandWCPayRequest', data, function (res) { 4 5 is_suc = true; 6 7 if (res.err_msg == "get_brand_wcpay_request:ok") { //支付成功后 8 9 location.href = '@Url.TenantAction("PaySuccess", "Personal")'; 10 11 } else { 12 13 14 15 } 16 17 }); 18 19 } 20 21 function CallPay(data) { 22 23 if (typeof WeixinJSBridge == "undefined") { 24 25 if (document.addEventListener) { 26 27 document.addEventListener('WeixinJSBridgeReady', onBridgeReady(data), false); 28 29 } else if (document.attachEvent) { 30 31 document.attachEvent('WeixinJSBridgeReady', onBridgeReady(data)); 32 33 document.attachEvent('onWeixinJSBridgeReady', onBridgeReady(data)); 34 35 } 36 37 } else { 38 39 onBridgeReady(data); 40 41 } 42 43 } 44 45 46 47 this.toPay = function () { 48 49 wc.restApi.get({ 50 51 isBlockUI: false, 52 53 url: '/api/MyOrder/Pay/' + _orderid, 54 55 success: function (result) { 56 57 CallPay(ko.toJS(result)); 58 59 } 60 61 }); 62 63 }
注意这里,这里先是通过ajax请求到我们前面所写的微信统一下单那个action,然后获取到返回的那个对象
关于ko.toJS(),这是knockout里面的方法,作用是把result处理成json对象。
好了,到此微信支付已经处理完了,详细代码请移步http://github.com/xin-lai下载最新源码。