来源: [转载]微信企业付款开发 C# – LLorJJ999 – 博客园
一、第一步:需要准备的东东
<add key=”sAppID” value=”wxf2303b260*******”/> // 开发者应用ID ,如图一
<add key=”sAppSecret” value=”5b83018b0b69c4231c14a2a25*******”/> //开发者应用密钥,如图一
<add key=”sPartnerKey” value=”E07D38C4EC0C9B42BF808633*******”/> //商户秘钥 如图二
<add key=”sPartner” value=”**********”/> // 商户号 申请微信支付通过后,收到的邮件里面有, 10位的商户号
(图一),登录地址:https://mp.weixin.qq.com/
(图二,你自己设置的32位秘钥),登录地址:https://pay.weixin.qq.com/index.php/home/login?return_url=%2F
二、第二步:安装证书,windows下面选择.p12格式的证书;取证书时一定得用 绝对路径(如:D:\wwwroot\BusinessCard\weixin\businesscard\cert\apiclient_cert.p12)
证书pkcs12格式 (apiclient_cert.p12)
三、第三步:帐户里面必须有钱,如下图
四、第四步:源码如下:
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Net.Security; using System.Security.Cryptography; using System.Security.Cryptography.X509Certificates; using System.Text; using System.Web; namespace BusinessCard.Web.Code { public class Class1 { /// <summary> /// 企业付款给个人,直接入帐到微信钱包中 /// </summary> public static string TENPAY = "1"; public static string APPID = Vincent._WebConfig.GetAppSettingsString("sAppID"); //开发者应用ID public static string PARTNER = Vincent._WebConfig.GetAppSettingsString("sPartner"); //商户号 public static string APPSECRET = Vincent._WebConfig.GetAppSettingsString("sAppSecret"); //开发者应用密钥 public static string PARTNER_KEY = Vincent._WebConfig.GetAppSettingsString("sPartnerKey"); //商户秘钥 public const string URL = "https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers"; //服务器异步通知页面路径(流量卡) public static string WebUrl = Vincent._WebConfig.GetAppSettingsString("WebUrl"); public static readonly string NOTIFY_URL_Card_Store = "http://" + WebUrl + "/weixinpay/WXPayNotify_URL.aspx";// ConfigurationManager.AppSettings["WXPayNotify_URL_CardStore"].ToString(); public static readonly string NOTIFY_URL_Card_User = "http://" + WebUrl + "/weixinpay/WXPayNotify_URL.aspx"; //ConfigurationManager.AppSettings["WXPayNotify_URL_CardUser"].ToString(); public static readonly string NOTIFY_URL_HB_Store = "http://" + WebUrl + "/weixinpay/WXPayNotify_URL.aspx";// ConfigurationManager.AppSettings["WXPayNotify_URL_CardStore"].ToString(); //=======【代理服务器设置】=================================== /* 默认IP和端口号分别为0.0.0.0和0,此时不开启代理(如有需要才设置) */ public const string PROXY_URL = "http://10.152.18.220:8080"; //=======【证书路径设置】===================================== /* 证书路径,注意应该填写绝对路径(仅退款、撤销订单时需要) */ public const string SSLCERT_PATH = "weixin\\businesscard\\cert\\apiclient_cert.p12"; public static string SSLCERT_PASSWORD = PARTNER; /// <summary> /// 企业付款给个人 /// </summary> /// <returns></returns> public static string EnterprisePay(string Bill_No, string toOpenid, decimal Charge_Amt, string userName, string title) { //公众账号appid mch_appid 是 wx8888888888888888 String 微信分配的公众账号ID(企业号corpid即为此appId) //商户号 mchid 是 1900000109 String(32) 微信支付分配的商户号 //设备号 device_info 否 013467007045764 String(32) 微信支付分配的终端设备号 //随机字符串 nonce_str 是 5K8264ILTKCH16CQ2502SI8ZNMTM67VS String(32) 随机字符串,不长于32位 //签名 sign 是 C380BEC2BFD727A4B6845133519F3AD6 String(32) 签名,详见签名算法 //商户订单号 partner_trade_no 是 10000098201411111234567890 String 商户订单号,需保持唯一性 //用户openid openid 是 oxTWIuGaIt6gTKsQRLau2M0yL16E String 商户appid下,某用户的openid //校验用户姓名选项 check_name 是 OPTION_CHECK String NO_CHECK:不校验真实姓名 //FORCE_CHECK:强校验真实姓名(未实名认证的用户会校验失败,无法转账) //OPTION_CHECK:针对已实名认证的用户才校验真实姓名(未实名认证用户不校验,可以转账成功) //收款用户姓名 re_user_name 可选 马花花 String 收款用户真实姓名。 // 如果check_name设置为FORCE_CHECK或OPTION_CHECK,则必填用户真实姓名 //金额 amount 是 10099 int 企业付款金额,单位为分 //企业付款描述信息 desc 是 理赔 String 企业付款操作说明信息。必填。 //Ip地址 spbill_create_ip 是 192.168.0.1 String(32) 调用接口的机器Ip地址 Bill_No = PARTNER + getTimestamp() + Bill_No; //订单号组成 商户号 + 随机时间串 + 记录ID //设置package订单参数 SortedDictionary<string, string> dic = new SortedDictionary<string, string>(); string total_fee = (Charge_Amt * 100).ToString("f0"); string wx_nonceStr = Guid.NewGuid().ToString().Replace("-", ""); //Interface_WxPay.getNoncestr(); dic.Add("mch_appid", APPID); dic.Add("mchid", PARTNER);//财付通帐号商家 //dic.Add("device_info", "013467007045711");//可为空 dic.Add("nonce_str", wx_nonceStr); dic.Add("partner_trade_no", Bill_No); dic.Add("openid", toOpenid); dic.Add("check_name", "NO_CHECK"); dic.Add("amount", total_fee); dic.Add("desc", title);//商品描述 dic.Add("spbill_create_ip", "211.149.234.224"); //用户的公网ip,不是商户服务器IP //生成签名 string get_sign = BuildRequest(dic, PARTNER_KEY); Vincent._Log.SaveMessage("第一步 get_sign:" + get_sign); string _req_data = "<xml>"; _req_data += "<mch_appid>" + APPID + "</mch_appid>"; _req_data += "<mchid>" + PARTNER + "</mchid>"; _req_data += "<nonce_str>" + wx_nonceStr + "</nonce_str>"; _req_data += "<partner_trade_no>" + Bill_No + "</partner_trade_no>"; _req_data += "<openid>" + toOpenid + "</openid>"; _req_data += "<check_name>NO_CHECK</check_name>"; _req_data += "<amount>" + total_fee + "</amount>"; _req_data += "<desc>" + title + "</desc>"; _req_data += "<spbill_create_ip>211.149.234.224</spbill_create_ip>"; _req_data += "<sign>" + get_sign + "</sign>"; _req_data += "</xml>"; Vincent._Log.SaveMessage("企业付款生成的xml:" + _req_data.Trim()); var result = HttpPost(URL, _req_data.Trim(), true, 300); //var result = HttpPost(URL, _req_data, Encoding.UTF8); Vincent._Log.SaveMessage("返回结果:" + result); return result; //ReturnValue retValue = StreamReaderUtils.StreamReader(URL, Encoding.UTF8.GetBytes(_req_data), System.Text.Encoding.UTF8, true); //Vincent._Log.SaveMessage("返回结果:" + retValue.ErrorCode); //return retValue.ErrorCode; } public static string getTimestamp() { TimeSpan ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0); return Convert.ToInt64(ts.TotalSeconds).ToString(); } public static string BuildRequest(SortedDictionary<string, string> sParaTemp, string key) { //获取过滤后的数组 Dictionary<string, string> dicPara = new Dictionary<string, string>(); dicPara = FilterPara(sParaTemp); //组合参数数组 string prestr = CreateLinkString(dicPara); //拼接支付密钥 string stringSignTemp = prestr + "&key=" + key; Vincent._Log.SaveMessage("生成签名的参数:" + stringSignTemp); //获得加密结果 string myMd5Str = GetMD5(stringSignTemp.Trim()); //返回转换为大写的加密串 return myMd5Str.ToUpper(); } /// <summary> /// 除去数组中的空值和签名参数并以字母a到z的顺序排序 /// </summary> /// <param name="dicArrayPre">过滤前的参数组</param> /// <returns>过滤后的参数组</returns> public static Dictionary<string, string> FilterPara(SortedDictionary<string, string> dicArrayPre) { Dictionary<string, string> dicArray = new Dictionary<string, string>(); foreach (KeyValuePair<string, string> temp in dicArrayPre) { if (temp.Key != "sign" && !string.IsNullOrEmpty(temp.Value)) { dicArray.Add(temp.Key, temp.Value); } } return dicArray; } //组合参数数组 public static string CreateLinkString(Dictionary<string, string> dicArray) { StringBuilder prestr = new StringBuilder(); foreach (KeyValuePair<string, string> temp in dicArray) { prestr.Append(temp.Key + "=" + temp.Value + "&"); } int nLen = prestr.Length; prestr.Remove(nLen - 1, 1); return prestr.ToString(); } //加密 public static string GetMD5(string pwd) { MD5 md5Hasher = MD5.Create(); byte[] data = md5Hasher.ComputeHash(Encoding.UTF8.GetBytes(pwd)); StringBuilder sBuilder = new StringBuilder(); for (int i = 0; i < data.Length; i++) { sBuilder.Append(data[i].ToString("x2")); } return sBuilder.ToString(); } public static string HttpPost(string postUrl, string paramData, Encoding dataEncode) { string ret = string.Empty; try { byte[] byteArray = dataEncode.GetBytes(paramData); //转化 HttpWebRequest webReq = (HttpWebRequest)WebRequest.Create(new Uri(postUrl)); webReq.Method = "POST"; webReq.ContentType = "application/x-www-form-urlencoded"; webReq.ContentLength = byteArray.Length; Stream newStream = webReq.GetRequestStream(); newStream.Write(byteArray, 0, byteArray.Length);//写入参数 newStream.Close(); HttpWebResponse response = (HttpWebResponse)webReq.GetResponse(); StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.Default); ret = sr.ReadToEnd(); sr.Close(); response.Close(); newStream.Close(); } catch (Exception ex) { Vincent._Log.SaveMessage("Post提交异常:" + ex.Message); } return ret; } public static bool CheckValidationResult(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors) { //直接确认,否则打不开 return true; } /// <summary> /// post提交支付 /// </summary> /// <param name="xml"></param> /// <param name="url"></param> /// <param name="isUseCert">是否使用证书</param> /// <param name="timeout"></param> /// <returns></returns> public static string HttpPost(string url, string xml, bool isUseCert, int timeout) { System.GC.Collect();//垃圾回收,回收没有正常关闭的http连接 string result = "";//返回结果 HttpWebRequest request = null; HttpWebResponse response = null; Stream reqStream = null; try { //设置最大连接数 ServicePointManager.DefaultConnectionLimit = 200; //设置https验证方式 if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase)) { ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult); } /*************************************************************** * 下面设置HttpWebRequest的相关属性 * ************************************************************/ request = (HttpWebRequest)WebRequest.Create(url); request.Method = "POST"; request.Timeout = timeout * 1000; //设置代理服务器 //WebProxy proxy = new WebProxy(); //定义一个网关对象 //proxy.Address = new Uri(PROXY_URL); //网关服务器端口:端口 //request.Proxy = proxy; //设置POST的数据类型和长度 request.ContentType = "text/xml"; byte[] data = System.Text.Encoding.UTF8.GetBytes(xml); request.ContentLength = data.Length; //是否使用证书 if (isUseCert) { string path = HttpContext.Current.Request.PhysicalApplicationPath; //X509Certificate2 cert = new X509Certificate2(path + SSLCERT_PATH, SSLCERT_PASSWORD); //将上面的改成 X509Certificate2 cert = new X509Certificate2(path + SSLCERT_PATH, SSLCERT_PASSWORD, X509KeyStorageFlags.PersistKeySet | X509KeyStorageFlags.MachineKeySet);//线上发布需要添加 request.ClientCertificates.Add(cert); Vincent._Log.SaveMessage("证书路径:" + (path + SSLCERT_PATH)); //Vincent._Log.SaveMessage("WxPayApi:PostXml used cert"); } //往服务器写入数据 reqStream = request.GetRequestStream(); reqStream.Write(data, 0, data.Length); reqStream.Close(); //获取服务端返回 response = (HttpWebResponse)request.GetResponse(); //获取服务端返回数据 StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8); result = sr.ReadToEnd().Trim(); sr.Close(); } catch (System.Threading.ThreadAbortException e) { Vincent._Log.SaveMessage("HttpService:Thread - caught ThreadAbortException - resetting."); Vincent._Log.SaveMessage("Exception message:" + e.Message); System.Threading.Thread.ResetAbort(); } catch (WebException e) { Vincent._Log.SaveMessage("HttpService" + e.ToString()); if (e.Status == WebExceptionStatus.ProtocolError) { Vincent._Log.SaveMessage("HttpService:StatusCode : " + ((HttpWebResponse)e.Response).StatusCode); Vincent._Log.SaveMessage("HttpService:StatusDescription : " + ((HttpWebResponse)e.Response).StatusDescription); } throw new Exception(e.ToString()); } catch (Exception e) { Vincent._Log.SaveMessage("HttpService" + e.ToString()); throw new Exception(e.ToString()); } finally { //关闭连接和流 if (response != null) { response.Close(); } if (request != null) { request.Abort(); } } return result; } /// <summary> /// 处理http GET请求,返回数据 /// </summary> /// <param name="url">请求的url地址</param> /// <returns>http GET成功后返回的数据,失败抛WebException异常</returns> public static string Get(string url) { System.GC.Collect(); string result = ""; HttpWebRequest request = null; HttpWebResponse response = null; //请求url以获取数据 try { //设置最大连接数 ServicePointManager.DefaultConnectionLimit = 200; //设置https验证方式 if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase)) { ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult); } /*************************************************************** * 下面设置HttpWebRequest的相关属性 * ************************************************************/ request = (HttpWebRequest)WebRequest.Create(url); request.Method = "GET"; //设置代理 WebProxy proxy = new WebProxy(); proxy.Address = new Uri(PROXY_URL); request.Proxy = proxy; //获取服务器返回 response = (HttpWebResponse)request.GetResponse(); //获取HTTP返回数据 StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8); result = sr.ReadToEnd().Trim(); sr.Close(); } catch (System.Threading.ThreadAbortException e) { Vincent._Log.SaveMessage("HttpService:Thread - caught ThreadAbortException - resetting."); Vincent._Log.SaveMessage("Exception message: " + e.Message); System.Threading.Thread.ResetAbort(); } catch (WebException e) { Vincent._Log.SaveMessage("HttpService" + e.ToString()); if (e.Status == WebExceptionStatus.ProtocolError) { Vincent._Log.SaveMessage("HttpService:StatusCode : " + ((HttpWebResponse)e.Response).StatusCode); Vincent._Log.SaveMessage("HttpService:StatusDescription : " + ((HttpWebResponse)e.Response).StatusDescription); } throw new Exception(e.ToString()); } catch (Exception e) { Vincent._Log.SaveMessage("HttpService" + e.ToString()); throw new Exception(e.ToString()); } finally { //关闭连接和流 if (response != null) { response.Close(); } if (request != null) { request.Abort(); } } return result; } } }