WeChatFunction.cs 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. using System;
  2. using System.Collections.Generic;
  3. using Library;
  4. using LitJson;
  5. using System.Linq;
  6. using System.Security.Cryptography.X509Certificates;
  7. using System.Security.Cryptography;
  8. using System.Text;
  9. using System.IO;
  10. using System.Net;
  11. using MySystem.PxcModels;
  12. using Org.BouncyCastle.Crypto.Modes;
  13. using Org.BouncyCastle.Crypto.Parameters;
  14. using Org.BouncyCastle.Crypto.Engines;
  15. namespace MySystem
  16. {
  17. public class WeChatFunction
  18. {
  19. public WeChatFunction()
  20. {
  21. }
  22. #region 服务商平台商户进件提交申请单API
  23. public string MerchantCreate(PxcModels.MerchantAddInfo info, PxcModels.MerchantInfo merchant)
  24. {
  25. // pubkey = GetPublicKey();
  26. Dictionary<string, object> result = new Dictionary<string, object>();
  27. result.Add("business_code", info.BusinessCode); //业务申请编号
  28. PxcModels.WebCMSEntities dbpxc = new PxcModels.WebCMSEntities();
  29. //超级管理员信息
  30. Dictionary<string, object> contact_info = new Dictionary<string, object>();
  31. contact_info.Add("contact_type", "LEGAL"); //超级管理员姓名
  32. contact_info.Add("contact_name", RSAEncrypt(info.ContactName)); //超级管理员姓名
  33. contact_info.Add("contact_id_number", RSAEncrypt(info.ContactIdNumber)); //超级管理员身份证件号码
  34. // contact_info.Add("openid", info.OpenId); //超级管理员微信openid
  35. contact_info.Add("mobile_phone", RSAEncrypt(info.MobilePhone)); //联系手机
  36. contact_info.Add("contact_email", RSAEncrypt(info.ContactEmail)); //联系邮箱
  37. result.Add("contact_info", contact_info);
  38. //主体资料
  39. Dictionary<string, object> subject_info = new Dictionary<string, object>();
  40. subject_info.Add("subject_type", info.SubjectType); //主体类型
  41. Dictionary<string, object> business_license_info = new Dictionary<string, object>();
  42. business_license_info.Add("license_copy", GetMediaId("/" + info.LicenseCopy)); //营业执照照片
  43. business_license_info.Add("license_number", info.LicenseNumber); //注册号/统一社会信用代码
  44. business_license_info.Add("merchant_name", merchant.Name); //商户名称
  45. business_license_info.Add("legal_person", info.LegalPerson); //个体户经营者/法人姓名
  46. subject_info.Add("business_license_info", business_license_info); //营业执照
  47. // Dictionary<string, object> certificate_info = new Dictionary<string, object>();
  48. // certificate_info.Add("cert_copy", GetMediaId("/" + info.CertCopy)); //登记证书照片
  49. // certificate_info.Add("cert_type", info.CertType); //登记证书类型
  50. // certificate_info.Add("cert_number", info.CertNumber); //证书号
  51. // certificate_info.Add("merchant_name", info.CertMerchantName); //商户名称
  52. // certificate_info.Add("company_address", info.CompanyAddress); //注册地址
  53. // certificate_info.Add("legal_person", info.CertLegalPerson); //法人姓名
  54. // certificate_info.Add("period_begin", CheckForever(info.PeriodBegin)); //有效期限开始日期
  55. // certificate_info.Add("period_end", CheckForever(info.PeriodEnd)); //有效期限结束日期
  56. // subject_info.Add("certificate_info", certificate_info); //登记证书
  57. // Dictionary<string, object> organization_info = new Dictionary<string, object>();
  58. // organization_info.Add("organization_copy", GetMediaId("/" + info.OrganizationCopy)); //组织机构代码证照片
  59. // organization_info.Add("organization_code", info.OrganizationCode); //组织机构代码
  60. // organization_info.Add("org_period_begin", CheckForever(info.OrgPeriodBegin)); //组织机构代码证有效期开始日期
  61. // organization_info.Add("org_period_end", CheckForever(info.OrgPeriodEnd)); //组织机构代码证有效期结束日期
  62. // subject_info.Add("organization_info", organization_info); //组织机构代码证
  63. // subject_info.Add("certificate_letter_copy", GetMediaId("/" + info.CertificateLetterCopy)); //单位证明函照片
  64. Dictionary<string, object> identity_info = new Dictionary<string, object>();
  65. info.IdDocType = "IDENTIFICATION_TYPE_IDCARD";
  66. identity_info.Add("id_holder_type", "LEGAL"); //证件持有人类型
  67. identity_info.Add("id_doc_type", info.IdDocType); //证件类型
  68. Dictionary<string, object> id_card_info = new Dictionary<string, object>();
  69. id_card_info.Add("id_card_copy", GetMediaId("/" + info.IdCardCopy)); //身份证人像面照片
  70. id_card_info.Add("id_card_national", GetMediaId("/" + info.IdCardNational)); //身份证人像面照片
  71. id_card_info.Add("id_card_name", RSAEncrypt(info.IdCardName)); //身份证姓名
  72. id_card_info.Add("id_card_number", RSAEncrypt(info.IdCardNumber)); //身份证号码
  73. id_card_info.Add("id_card_address", RSAEncrypt(info.IdCardAddress)); //身份证居住地址
  74. id_card_info.Add("card_period_begin", CheckForever(info.CardPeriodBegin)); //身份证有效期开始时间
  75. id_card_info.Add("card_period_end", CheckForever(info.CardPeriodEnd)); //身份证有效期结束时间
  76. identity_info.Add("id_card_info", id_card_info); //身份证信息
  77. // identity_info.Add("owner", info.Owner == 1 ? true : false); //经营者/法人是否为受益人
  78. identity_info.Add("owner", true);
  79. subject_info.Add("identity_info", identity_info); //经营者/法人身份证件
  80. // if (info.Owner == 0)
  81. // {
  82. // Dictionary<string, object> ubo_info = new Dictionary<string, object>();
  83. // ubo_info.Add("id_type", info.UboInfoIdType); //证件类型
  84. // ubo_info.Add("id_card_copy", GetMediaId("/" + info.UboInfoIdCardCopy)); //身份证人像面照片
  85. // ubo_info.Add("id_card_national", GetMediaId("/" + info.UboInfoIdCardNational)); //身份证国徽面照片
  86. // ubo_info.Add("id_doc_copy", GetMediaId("/" + info.UboInfoIdDocCopy)); //证件照片
  87. // ubo_info.Add("name", RSAEncrypt(info.UboInfoName)); //受益人姓名
  88. // ubo_info.Add("id_number", RSAEncrypt(info.UboInfoIdNumber)); //证件号码
  89. // ubo_info.Add("id_period_begin", CheckForever(info.UboInfoIdPeriodBegin)); //证件有效期开始时间
  90. // ubo_info.Add("id_period_end", CheckForever(info.UboInfoIdPeriodEnd)); //证件有效期结束时间
  91. // subject_info.Add("ubo_info", ubo_info); //最终受益人信息(UBO)
  92. // }
  93. result.Add("subject_info", subject_info);
  94. //经营资料
  95. Dictionary<string, object> business_info = new Dictionary<string, object>();
  96. business_info.Add("merchant_shortname", info.MerchantShortname); //商户简称
  97. business_info.Add("service_phone", info.ServicePhone); //客服电话
  98. Dictionary<string, object> sales_info = new Dictionary<string, object>();
  99. info.SalesScenesType = "SALES_SCENES_STORE";
  100. sales_info.Add("sales_scenes_type", info.SalesScenesType.Split(',').ToList()); //经营场景类型
  101. Dictionary<string, object> biz_store_info = new Dictionary<string, object>();
  102. biz_store_info.Add("biz_store_name", info.BizStoreName); //门店名称
  103. PxcModels.OpenBankCityTable BizAddress = dbpxc.OpenBankCityTable.FirstOrDefault(m => m.CityName.EndsWith(info.BizAddressCode)) ?? new PxcModels.OpenBankCityTable();
  104. biz_store_info.Add("biz_address_code", BizAddress.Code); //门店省市编码
  105. biz_store_info.Add("biz_store_address", info.BizStoreAddress); //门店地址
  106. List<string> store_entrance_pic = new List<string>();
  107. store_entrance_pic.Add(GetMediaId("/" + info.StoreEntrancePic));
  108. biz_store_info.Add("store_entrance_pic", store_entrance_pic); //门店门头照片
  109. List<string> indoor_pic = new List<string>();
  110. indoor_pic.Add(GetMediaId("/" + info.IndoorPic));
  111. biz_store_info.Add("indoor_pic", indoor_pic); //店内环境照片
  112. biz_store_info.Add("biz_sub_appid", info.BizSubAppid); //线下场所对应的商家APPID
  113. sales_info.Add("biz_store_info", biz_store_info); //线下门店场景
  114. // Dictionary<string, object> mp_info = new Dictionary<string, object>();
  115. // mp_info.Add("mp_appid", info.MpAppid); //服务商公众号APPID
  116. // mp_info.Add("mp_sub_appid", info.MpSubAppid); //商家公众号APPID
  117. // mp_info.Add("mp_pics", GetMediaId("/" + info.MpPics)); //公众号页面截图
  118. // sales_info.Add("mp_info", mp_info); //公众号场景
  119. // Dictionary<string, object> mini_program_info = new Dictionary<string, object>();
  120. // mini_program_info.Add("mini_program_appid", info.MiniProgramAppid); //服务商小程序APPID
  121. // mini_program_info.Add("mini_program_sub_appid", info.MiniProgramSubAppid); //商家小程序APPID
  122. // mini_program_info.Add("mini_program_pics", GetMediaId("/" + info.MiniProgramPics)); //小程序截图
  123. // sales_info.Add("mini_program_info", mini_program_info); //小程序场景
  124. // Dictionary<string, object> app_info = new Dictionary<string, object>();
  125. // app_info.Add("app_appid", info.AppAppid); //服务商应用APPID
  126. // app_info.Add("app_sub_appid", info.AppSubAppid); //商家应用APPID
  127. // app_info.Add("app_pics", GetMediaId("/" + info.AppPics)); //商家应用APPID
  128. // sales_info.Add("app_info", app_info); //APP场景
  129. // Dictionary<string, object> web_info = new Dictionary<string, object>();
  130. // web_info.Add("domain", info.WebDomain); //互联网网站域名
  131. // web_info.Add("web_authorisation", GetMediaId("/" + info.WebAuthorisation)); //网站授权函
  132. // web_info.Add("web_appid", info.WebAppId); //互联网网站对应的商家APPID
  133. // sales_info.Add("web_info", web_info); //互联网网站场景
  134. // Dictionary<string, object> wework_info = new Dictionary<string, object>();
  135. // wework_info.Add("sub_corp_id", info.SubCorpId); //商家企业微信CorpID
  136. // wework_info.Add("wework_pics", GetMediaId("/" + info.WeworkPics)); //企业微信页面截图
  137. // sales_info.Add("wework_info", wework_info); //企业微信场景
  138. business_info.Add("sales_info", sales_info); //经营场景
  139. result.Add("business_info", business_info);
  140. //结算规则
  141. Dictionary<string, object> settlement_info = new Dictionary<string, object>();
  142. settlement_info.Add("settlement_id", info.SettlementId); //入驻结算规则ID
  143. string QualificationType = function.CheckNull(info.QualificationType);
  144. if (QualificationType.Contains(","))
  145. {
  146. QualificationType = QualificationType.Split(',')[0];
  147. }
  148. settlement_info.Add("qualification_type", QualificationType); //所属行业
  149. if (!string.IsNullOrEmpty(info.Qualifications))
  150. {
  151. List<string> Qualifications = new List<string>();
  152. Qualifications.Add(GetMediaId("/" + info.Qualifications));
  153. settlement_info.Add("qualifications", Qualifications); //特殊资质图片
  154. }
  155. settlement_info.Add("activities_id", info.ActivitiesId); //优惠费率活动ID
  156. settlement_info.Add("activities_rate", info.ActivitiesRate); //优惠费率活动值
  157. result.Add("settlement_info", settlement_info);
  158. //结算银行账户
  159. Dictionary<string, object> bank_account_info = new Dictionary<string, object>();
  160. bank_account_info.Add("bank_account_type", info.BankAccountType); //账户类型
  161. bank_account_info.Add("account_name", RSAEncrypt(info.AccountName)); //开户名称
  162. bank_account_info.Add("account_bank", info.AccountBank); //开户银行
  163. PxcModels.OpenBankCityTable BankAddress = dbpxc.OpenBankCityTable.FirstOrDefault(m => m.CityName.EndsWith(info.BankAddressCode)) ?? new PxcModels.OpenBankCityTable();
  164. bank_account_info.Add("bank_address_code", BankAddress.Code); //开户银行省市编码
  165. PxcModels.OpenBankTable BankBranch = dbpxc.OpenBankTable.FirstOrDefault(m => m.BankName == info.BankName) ?? new PxcModels.OpenBankTable();
  166. bank_account_info.Add("bank_branch_id", BankBranch.BankCode); //开户银行联行号
  167. bank_account_info.Add("bank_name", info.BankName); //开户银行全称(含支行)
  168. bank_account_info.Add("account_number", RSAEncrypt(info.AccountNumber)); //银行账号
  169. result.Add("bank_account_info", bank_account_info);
  170. dbpxc.Dispose();
  171. string req = Newtonsoft.Json.JsonConvert.SerializeObject(result);
  172. Dictionary<string, string> header = new Dictionary<string, string>();
  173. // 如何查看证书序列号?
  174. // 登陆商户平台【API安全】->【API证书】->【查看证书】,可查看商户API证书序列号。
  175. // 商户API证书和微信支付平台证书均可以使用第三方的证书解析工具,查看证书内容。或者使用openssl命令行工具查看证书序列号。
  176. // $ openssl x509 -in 1900009191_20180326_cert.pem -noout -serial
  177. // serial=1DDE55AD98ED71D6EDD4A4A16996DE7B47773A8C
  178. string merchantId = "1611167423"; //商户号
  179. string serialNo = "534165421BE2814DEC08F8D00BDAB9944B13646C"; //证书编号
  180. string resp = postJson("https://api.mch.weixin.qq.com/v3/applyment4sub/applyment/", req, prikey, merchantId, serialNo);
  181. function.WriteLog(resp, "商户进件日志");
  182. return resp;
  183. }
  184. #endregion
  185. #region 服务商平台查询申请单状态API
  186. public Dictionary<string, object> QueryMerchant(string BusinessCode)
  187. {
  188. Dictionary<string, object> return_result = new Dictionary<string, object>();
  189. try
  190. {
  191. string merchantId = "1611167423"; //商户号
  192. string serialNo = "534165421BE2814DEC08F8D00BDAB9944B13646C"; //证书编号
  193. string result = postJson("https://api.mch.weixin.qq.com/v3/applyment4sub/applyment/business_code/" + BusinessCode, "", prikey, merchantId, serialNo, "GET");
  194. function.WriteLog(result, "服务商平台查询申请单状态API");
  195. //{\"applyment_id\":2000002247709762,\"applyment_state\":\"APPLYMENT_STATE_FINISHED\",\"applyment_state_msg\":\"商户入驻申请已完成\",\"audit_detail\":[],\"business_code\":\"0123456789\",\"sign_url\":\"https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket=gQFv7zwAAAAAAAAAAS5odHRwOi8vd2VpeGluLnFxLmNvbS9xLzAyUXE1czkzb3JlUjIxZHpXbTF5Y2YAAgRjbe9hAwQAjScA\",\"sub_mchid\":\"1619775262\"}
  196. JsonData jsonObj = JsonMapper.ToObject(result);
  197. return_result.Add("applyment_id", jsonObj["applyment_id"].ToString()); //微信支付申请单号
  198. return_result.Add("applyment_state", jsonObj["applyment_state"].ToString()); //申请单状态
  199. return_result.Add("applyment_state_msg", jsonObj["applyment_state_msg"].ToString()); //申请状态描述
  200. if (jsonObj["applyment_state"].ToString() == "APPLYMENT_STATE_REJECTED")
  201. {
  202. List<Dictionary<string, string>> audit_detail = new List<Dictionary<string, string>>();
  203. JsonData auditObj = jsonObj["audit_detail"]; //驳回原因详情
  204. // for (int i = 0; i < auditObj.Count; i++)
  205. // {
  206. // Dictionary<string, string> row = new Dictionary<string, string>();
  207. // // row.Add("field", auditObj[i]["field"].ToString()); //字段名
  208. // // row.Add("field_name", auditObj[i]["field_name"].ToString()); //字段名称
  209. // row.Add("reject_reason", auditObj[i]["reject_reason"].ToString()); //驳回原因
  210. // audit_detail.Add(row);
  211. // }
  212. return_result.Add("reject_reason", auditObj[0]["reject_reason"].ToString());
  213. return_result.Add("audit_detail", audit_detail); //申请状态描述
  214. }
  215. if (result.Contains("\"sign_url\":"))
  216. {
  217. return_result.Add("sign_url", jsonObj["sign_url"].ToString()); //进件通过申请,待商家签约的地址
  218. }
  219. }
  220. catch (Exception ex)
  221. {
  222. function.WriteLog(DateTime.Now.ToString() + "\r\n" + ex.ToString(), "服务商平台查询申请单状态API异常");
  223. }
  224. return return_result;
  225. }
  226. #endregion
  227. #region 服务商平台申请退款API
  228. public Dictionary<string, string> Refund(string AppId, string MchId, string SubMchId, decimal RefundMoney, decimal TotalMoney, string OrderNo, string RefundNo)
  229. {
  230. Dictionary<string, string> return_result = new Dictionary<string, string>();
  231. int refund = (int)(RefundMoney * 100);
  232. int amount = (int)(TotalMoney * 100);
  233. //具体请求参数
  234. SortedList<string, object> par = new SortedList<string, object>();
  235. par.Add("sub_mchid", SubMchId); //子商户号
  236. par.Add("out_trade_no", OrderNo); //商户订单号
  237. par.Add("out_refund_no", RefundNo); //商户退款单号
  238. Dictionary<string, object> amountDic = new Dictionary<string, object>();
  239. amountDic.Add("refund", refund); //退款金额
  240. amountDic.Add("total", amount); //原订单金额
  241. amountDic.Add("currency", "CNY"); //退款币种
  242. par.Add("amount", amountDic); //金额信息
  243. string req = Newtonsoft.Json.JsonConvert.SerializeObject(par);
  244. function.WriteLog(req, "服务商平台申请退款API");
  245. try
  246. {
  247. string merchantId = MchId; //商户号
  248. string serialNo = "534165421BE2814DEC08F8D00BDAB9944B13646C"; //证书编号
  249. string result = postJson("https://api.mch.weixin.qq.com/v3/refund/domestic/refunds", req, prikey, merchantId, serialNo);
  250. function.WriteLog(result, "服务商平台申请退款API");
  251. JsonData jsonObj = JsonMapper.ToObject(result);
  252. return_result.Add("refund_id", jsonObj["refund_id"].ToString()); //微信支付退款单号
  253. return_result.Add("channel", jsonObj["channel"].ToString()); //退款渠道
  254. return_result.Add("user_received_account", jsonObj["user_received_account"].ToString()); //退款入账账户
  255. return_result.Add("status", jsonObj["status"].ToString()); //退款状态
  256. return_result.Add("payer_total", jsonObj["amount"]["payer_total"].ToString()); //用户支付金额
  257. return_result.Add("payer_refund", jsonObj["amount"]["payer_refund"].ToString()); //用户退款金额
  258. return_result.Add("settlement_refund", jsonObj["amount"]["settlement_refund"].ToString()); //应结退款金额
  259. return_result.Add("settlement_total", jsonObj["amount"]["settlement_total"].ToString()); //应结订单金额
  260. return_result.Add("discount_refund", jsonObj["amount"]["discount_refund"].ToString()); //优惠退款金额
  261. }
  262. catch (Exception ex)
  263. {
  264. function.WriteLog(DateTime.Now.ToString() + "\r\n" + ex.ToString(), "服务商平台申请退款API异常");
  265. }
  266. return return_result;
  267. }
  268. #endregion
  269. #region 敏感信息加密
  270. // string pubkey = "MIIBCgKCAQEAxK9K2ElQ0rUMwNFW4xMx7kmkpDlp7WGQ8VffXLpudJBIrRVFWJN7dlinCGu67ZATf+/GZvLSSzQPWejHEsI9vUp9ej0x5iZDHGc9/shNUjd03ORkKo/Ohj3Ju19RzX4mh+GoinKGiuKX8CdGPXPe7UObKWyETrDcxrit6VCJQjE604aNtVeAUE+41mOpCLGzTgAKt4psfHxsBRtqlveH4cH/51ip+cGkC/u36gfDMKaRluVfFO3ETxDkI94BwNtnthcA4WDKha+wjtB6HIJ5xxHV550+cfdkl4j8UKU6pB+C5JRNgvjAf6ljg/TF+cGF1d8C6IbYYodYKloqN2DzrwIDAQAB"; //公钥
  271. string pubkey = "MIID3DCCAsSgAwIBAgIUU0FlQhvigU3sCPjQC9q5lEsTZGwwDQYJKoZIhvcNAQELBQAwXjELMAkGA1UEBhMCQ04xEzARBgNVBAoTClRlbnBheS5jb20xHTAbBgNVBAsTFFRlbnBheS5jb20gQ0EgQ2VudGVyMRswGQYDVQQDExJUZW5wYXkuY29tIFJvb3QgQ0EwHhcNMjEwNjMwMDMwOTA5WhcNMjYwNjI5MDMwOTA5WjBuMRgwFgYDVQQDDA9UZW5wYXkuY29tIHNpZ24xEzARBgNVBAoMClRlbnBheS5jb20xHTAbBgNVBAsMFFRlbnBheS5jb20gQ0EgQ2VudGVyMQswCQYDVQQGDAJDTjERMA8GA1UEBwwIU2hlblpoZW4wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC64SIwN1afgge2pjFeNZAlKz5ZdJw8W/hisaIJnkzbE/1oG5n+SgtbMLfBLBFp0zMyT5SP5ve5aJAvUjwnRwQoDG+4z3jE8vLyjFLxoy5sWGex8+FBC5hs81oAUHuazigi5NcSH9Ji/8Lq3zFh8Az4+ZchQ14SFhSGTifSZfdTX5wYDEv3DKDYgQvvAgDgjBF1Uj91NpVD7ruQjAJ7MwZvUX7rnYFnQggFQwIBWazNXaGa7OpO6Xbf+uNtbl436My7WQX+ffjmtYmcgND60mgn/rFQMg5r+Q+J8mDgIaAzCmr3xW31OIvAD1iYV0ksTyQntD5pOzCYdHZPDOZCqYlHAgMBAAGjgYEwfzAJBgNVHRMEAjAAMAsGA1UdDwQEAwIE8DBlBgNVHR8EXjBcMFqgWKBWhlRodHRwOi8vZXZjYS5pdHJ1cy5jb20uY24vcHVibGljL2l0cnVzY3JsP0NBPTFCRDQyMjBFNTBEQkMwNEIwNkFEMzk3NTQ5ODQ2QzAxQzNFOEVCRDIwDQYJKoZIhvcNAQELBQADggEBAEgXlFe4MpFUUnj6gJt5KoPpzLTNOXDHsglUrAobq/HlmDBOF8yCUFvtbRaJxUSQVFfo/2zag8J57YNMo2If1HkD1LMEGAcPGNS+aho9SgCFbLG+c82nNICl8g8oe1HNS7IpRjacSfrKIYt5VL5lHHC5ZJaKq0Ez9c7bfPGC6CaA8WDnApFZut5qhYNMBUYZZjAMVIZ2G/kIsxbo1oUpVEnQq5YQxPUDWdYKxdTvo7uXXpWv6qo761Hid24R7mdrGEuHZNBfxYCt1wJQNlcGVgCjk8PPo73mO0zYRwV0x7pNkFFbiP7aetf6RB82YUbuNkvql83Ox+e22ByMi8lOUZo=";
  272. string prikey = "MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQD7f7VmtjtLJaY+oJkMgDN9dDE082ZOgquHorPrtJNEdf7VjRFLMoKOoiQsT4OgVq3lNN2KKndd5ahsegSq7Ng7aYJq7IDH584RWEGbekEn0M10aXThdk9dPSCMXBPFR6y4gMvV4R6pVbPBxoSFMI6ty+EqfZLP9Cxvxx7msIKEB46r2R6gYL7j061ytoURavUxGAZxMXnCAEckbrssiFwhLTmX6TtC8MVPyUfLPUh07ArRZxK2F0hpX35Te0Jda7xkulbWtz4EB8KPXr0bJGO/oUo+Prqpdjy0+zKMfMvBP6NG1Hnh3z7yBhlYIe4vymFQU1Wy0OtrEbrU1OdJiq5XAgMBAAECggEAJ7biAWczc8ltOCSKJpQ0DVKE1rS86hbuWUvTF5x7SOtNI5HUMki2nylxhuHDilVFvpUbis7r7JLwhri59ffSqYcexh20tFyhiNGkApZKpavuVby1bUkOfU3CzwdXQ+bumnAUQCD9g4X6DdZ4ZMPnNACm/Nu5jEcloT0TXyBua8SpmlFMYgo/x8XOTT4dEeubwNsfEpmSeWcQRJLJI5ujGgmmZLiAs7F3VlCJFNtQa1KT3a1EDhD5jajeQAzsPpQ48i/joOqE71RlVr77z4Ethr8o1Cc16K9Md2AtVesMZ6eZNqKkQ/qC3vnxk5Sy25tuVmOCXa81ErIf9Tf/JwhymQKBgQD/TXnYxEzYJCqa7/Ck6QqBHQf4ak1OVz7/NY7bmxYT5ez4eJKzrh+xQYWBq/UmKeEiH5QI6T3MVo7L20tbeuYLQwYpgb+BpOKHFne2ODOgDYreNREB06Ww2qAQYCqwyBnTqt7UthDmcXhcPojwxNLSJAOCQhld9TOOmK1db33PIwKBgQD8L5KiN9XRFCvUH6rhUcs/8lmGR2N8zSTts+BNCwIPGQWE0UvbjroyNxVbq6cXXlZZ6H07vEAgaIrLIkwLgen2IwsuOiyaEMvRP9oGwC+LnpP+9sxbW5w5do1hfyeGiSPPgSYIWZMOOwX+qfqXDMM3whJP1hk4El87kC+kl2URPQKBgQC4kTc8ebWgMI1Ik5wibIN2khlYAum+cB6P/BU6OCU+32NwwiI0mfY5fEfbVgiVvYMP20p3Ir/VgOIdCH2zNecnTx4eVbmwLRbDWH/VEU5pMEEroauY6XyInZVIIW3bfd4GAyqzFqijIHdGIMN5ZKGQ4A5VVhnmrkTGmyQCP37xhwKBgQCaLW+Fc4NqTt240x+Cn6IWV1E6UfzV+QmBtb4E/BznJBIGsHqjIbBeMOUC5X/DPq6mSwf9dEr0RVXNeQtq4aMrEyOrwG8yReCF7naaN/cZcfrjHvmzRN/PFda766Q7kS5ul3OsGS9ZvOYeWsdUf73+Ic1WN/u1Fy5X2P/3rYNhtQKBgAIioAq8psFdy/itpJryaWdmDM6fM6ctp5P6VBWOVp2JGkOzHboiXFToJJBtmi7Rdp8AN1aYBbtVX1o0+RmfEiHqXn84GpSNT6tGirHvARi5ab3jvn+bkUWjZYOGd96hus8GirA+wQb/V92Ms3jS5VUS1HBfHaTGT5G2Slrr8YZu"; //私钥
  273. public string RSAEncrypt(string text)
  274. {
  275. byte[] publicKey = Convert.FromBase64String(pubkey);
  276. // var rsa = RSA.Create();
  277. // rsa.ImportRSAPublicKey(publicKey, out _);
  278. // var buff = rsa.Encrypt(Encoding.UTF8.GetBytes(text), RSAEncryptionPadding.OaepSHA1);
  279. // return Convert.ToBase64String(buff);
  280. using (var x509 = new X509Certificate2(publicKey))
  281. {
  282. using (var rsa = x509.GetRSAPublicKey())
  283. {
  284. var buff = rsa.Encrypt(Encoding.UTF8.GetBytes(text), RSAEncryptionPadding.OaepSHA1);
  285. return Convert.ToBase64String(buff);
  286. }
  287. }
  288. }
  289. // public string RSADecrypt(string text)
  290. // {
  291. // // byte[] key = Encoding.UTF8.GetBytes(AppConfig.WeChatParam.AesGemKey);
  292. // // byte[] nonce = Encoding.UTF8.GetBytes("1234567890ab");
  293. // // byte[] ciphertext = Convert.FromBase64String(text);
  294. // // AesGcm aes = new AesGcm(key);
  295. // // byte[] plaintext = new byte[ciphertext.Length];
  296. // // aes.Decrypt(nonce, ciphertext, plaintext, null);
  297. // // return Encoding.UTF8.GetString(plaintext);
  298. // }
  299. public string AesGcmDecrypt(string associatedData, string nonce, string ciphertext)
  300. {
  301. // GcmBlockCipher gcmBlockCipher = new GcmBlockCipher(new AesEngine());
  302. // AeadParameters aeadParameters = new AeadParameters(
  303. // new KeyParameter(Encoding.UTF8.GetBytes(AppConfig.WeChatParam.AesGemKey)),
  304. // 128,
  305. // Encoding.UTF8.GetBytes(nonce),
  306. // Encoding.UTF8.GetBytes(associatedData));
  307. // gcmBlockCipher.Init(false, aeadParameters);
  308. // byte[] data = Convert.FromBase64String(ciphertext);
  309. // byte[] plaintext = new byte[gcmBlockCipher.GetOutputSize(data.Length)];
  310. // int length = gcmBlockCipher.ProcessBytes(data, 0, data.Length, plaintext, 0);
  311. // gcmBlockCipher.DoFinal(plaintext, length);
  312. // return Encoding.UTF8.GetString(plaintext);
  313. byte[] key = Encoding.UTF8.GetBytes(AppConfig.WeChatParam.AesGemKey); // 256-bit key
  314. byte[] nonceByte = Encoding.UTF8.GetBytes(nonce); // 96-bit nonce
  315. byte[] cipherByte = Convert.FromBase64String(ciphertext);
  316. byte[] associatedByte = Encoding.UTF8.GetBytes(associatedData);
  317. GcmBlockCipher cipher = new GcmBlockCipher(new AesEngine());
  318. AeadParameters parameters = new AeadParameters(new KeyParameter(key), 128, nonceByte, associatedByte);
  319. cipher.Init(false, parameters);
  320. byte[] plaintext = new byte[cipher.GetOutputSize(cipherByte.Length)];
  321. int len = cipher.ProcessBytes(cipherByte, 0, cipherByte.Length, plaintext, 0);
  322. cipher.DoFinal(plaintext, len);
  323. return Encoding.UTF8.GetString(plaintext);
  324. }
  325. public string postJson(string url, string postData, string privateKey, string merchantId, string serialNo, string method = "POST")
  326. {
  327. HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
  328. request.Method = method;
  329. request.ContentType = "application/json;charset=UTF-8";
  330. request.UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3100.0 Safari/537.36";
  331. request.Accept = "application/json";
  332. string Authorization = GetAuthorization(url, method, postData, privateKey, merchantId, "62B64854B768D344249FDB9E6EF4A1232C000E84");
  333. request.Headers.Add("Authorization", Authorization);
  334. request.Headers.Add("Wechatpay-Serial", serialNo);
  335. if (!string.IsNullOrEmpty(postData))
  336. {
  337. byte[] paramJsonBytes;
  338. paramJsonBytes = System.Text.Encoding.UTF8.GetBytes(postData);
  339. request.ContentLength = paramJsonBytes.Length;
  340. Stream writer;
  341. try
  342. {
  343. writer = request.GetRequestStream();
  344. }
  345. catch (Exception)
  346. {
  347. writer = null;
  348. Console.Write("连接服务器失败!");
  349. }
  350. writer.Write(paramJsonBytes, 0, paramJsonBytes.Length);
  351. writer.Close();
  352. }
  353. HttpWebResponse response;
  354. try
  355. {
  356. response = (HttpWebResponse)request.GetResponse();
  357. }
  358. catch (WebException ex)
  359. {
  360. response = ex.Response as HttpWebResponse;
  361. }
  362. Stream resStream = response.GetResponseStream();
  363. StreamReader reader = new StreamReader(resStream);
  364. string text = reader.ReadToEnd();
  365. return text;
  366. //{\"code\":\"PARAM_ERROR\",\"message\":\"请确认待处理的消息是否为加密后的密文\"}
  367. //{\"applyment_id\":2000002247709762}
  368. }
  369. #endregion
  370. #region 图片上传
  371. public string GetMediaId(string imgPath)
  372. {
  373. if (string.IsNullOrEmpty(imgPath))
  374. {
  375. return "";
  376. }
  377. if (imgPath.Contains(","))
  378. {
  379. imgPath = imgPath.Split(',')[0];
  380. }
  381. string key = "wechatpic:" + function.MD5_16(imgPath);
  382. string media_id = RedisDbconn.Instance.Get<string>(key);
  383. if (!string.IsNullOrEmpty(media_id))
  384. {
  385. return media_id;
  386. }
  387. string filePath = function.getPath(imgPath);
  388. var filename = Path.GetFileName(filePath);
  389. FileStream fs = new FileStream(filePath, FileMode.Open, FileAccess.Read);
  390. Byte[] imgBytesIn = new Byte[fs.Length];
  391. fs.Read(imgBytesIn, 0, imgBytesIn.Length);
  392. fs.Close();
  393. byte[] hash = SHA256Managed.Create().ComputeHash(imgBytesIn);
  394. StringBuilder builder = new StringBuilder();
  395. for (int i = 0; i < hash.Length; i++)
  396. {
  397. builder.Append(hash[i].ToString("x2"));
  398. }
  399. var sha256 = builder.ToString();
  400. string metaStr = "{\"filename\":\""+ filename + "\",\"sha256\":\"" + sha256 + "\"}";
  401. media_id = UploadImgApi(metaStr, imgBytesIn, filename);
  402. RedisDbconn.Instance.Set(key, media_id);
  403. return media_id;
  404. }
  405. public string UploadImgApi(string metaStr, Byte[] imgBytesIn,string filename)
  406. {
  407. string url = "https://api.mch.weixin.qq.com/v3/merchant/media/upload";
  408. string merchantId = "1611167423"; //商户号
  409. string serialNo = "62B64854B768D344249FDB9E6EF4A1232C000E84"; //证书编号
  410. string privateKey = prikey;
  411. #region 定义请求体中的内容 并转成二进制
  412. string boundary = "lc199aecd61b4653ef";
  413. string Enter = "\r\n";
  414. string campaignIDStr1
  415. = "--" + boundary
  416. + Enter
  417. + "Content-Disposition: form-data; name=\"meta\";"
  418. + Enter
  419. + "Content-Type:application/json;"
  420. + Enter
  421. + Enter
  422. + metaStr
  423. + Enter
  424. + "--" + boundary
  425. + Enter
  426. + "Content-Disposition:form-data;name=\"file\";filename=\""+ filename + "\";"
  427. + Enter
  428. + "Content-Type:image/jpeg"
  429. + Enter
  430. + Enter;
  431. byte[] byteData2
  432. = imgBytesIn;
  433. string campaignIDStr3
  434. = Enter
  435. + "--" + boundary
  436. + Enter;
  437. var byteData1 = System.Text.Encoding.UTF8.GetBytes(campaignIDStr1);
  438. var byteData3 = System.Text.Encoding.UTF8.GetBytes(campaignIDStr3);
  439. #endregion
  440. string transactionsResponse = UploadImg_postJson(url, byteData1, byteData2, byteData3, metaStr, privateKey, merchantId, serialNo, boundary, "POST");
  441. var result=JsonMapper.ToObject(transactionsResponse);
  442. return result["media_id"].ToString();
  443. }
  444. public string UploadImg_postJson(string url, byte[] b1, byte[] b2, byte[] b3, string metaStr, string privateKey, string merchantId, string serialNo, string boundary, string method = "POST")
  445. {
  446. HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
  447. request.Method = method;
  448. //request.ContentType = "application/json;charset=UTF-8";
  449. request.ContentType = "multipart/form-data;boundary=" + boundary;
  450. request.UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3100.0 Safari/537.36";
  451. request.Accept = "application/json";
  452. string Authorization = GetAuthorization(url, method, metaStr, privateKey, merchantId, serialNo);
  453. request.Headers.Add("Authorization", Authorization);
  454. Stream writer;
  455. try
  456. {
  457. writer = request.GetRequestStream();
  458. }
  459. catch (Exception)
  460. {
  461. writer = null;
  462. }
  463. writer.Write(b1, 0, b1.Length);
  464. writer.Write(b2, 0, b2.Length);
  465. writer.Write(b3, 0, b3.Length);
  466. writer.Close();
  467. HttpWebResponse response;
  468. try
  469. {
  470. response = (HttpWebResponse)request.GetResponse();
  471. }
  472. catch (WebException ex)
  473. {
  474. response = ex.Response as HttpWebResponse;
  475. }
  476. Stream resStream = response.GetResponseStream();
  477. StreamReader reader = new StreamReader(resStream);
  478. string text = reader.ReadToEnd();
  479. return text;
  480. }
  481. protected string GetAuthorization(string url, string method, string jsonParame, string privateKey, string merchantId, string serialNo)
  482. {
  483. var uri = new Uri(url);
  484. string urlPath = uri.PathAndQuery;
  485. string nonce = Guid.NewGuid().ToString();
  486. var timestamp = DateTimeOffset.Now.ToUnixTimeSeconds();
  487. //数据签名 HTTP请求方法\n接口地址的url\n请求时间戳\n请求随机串\n请求报文主体\n
  488. method = string.IsNullOrEmpty(method) ? "" : method;
  489. string message = string.Format("{0}\n{1}\n{2}\n{3}\n{4}\n", method, urlPath, timestamp, nonce, jsonParame);
  490. string signTxt = Sign(message, privateKey);
  491. //Authorization和格式
  492. string authorzationTxt = string.Format("WECHATPAY2-SHA256-RSA2048 mchid=\"{0}\",nonce_str=\"{1}\",timestamp=\"{2}\",serial_no=\"{3}\",signature=\"{4}\"",
  493. merchantId,
  494. nonce,
  495. timestamp,
  496. serialNo,
  497. signTxt
  498. );
  499. return authorzationTxt;
  500. }
  501. protected string Sign(string message, string privateKey)
  502. {
  503. byte[] keyData = Convert.FromBase64String(privateKey);
  504. byte[] data = System.Text.Encoding.UTF8.GetBytes(message);
  505. var rsa = RSA.Create();
  506. rsa.ImportPkcs8PrivateKey(keyData, out _);
  507. return Convert.ToBase64String(rsa.SignData(data, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1));
  508. // using (CngKey cngKey = CngKey.Import(keyData, CngKeyBlobFormat.Pkcs8PrivateBlob))
  509. // using (RSACng rsa = new RSACng(cngKey))
  510. // {
  511. // return Convert.ToBase64String(rsa.SignData(data, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1));
  512. // }
  513. }
  514. #endregion
  515. #region 获取文件sha256
  516. public string GetSha256(FileStream stream)
  517. {
  518. using (SHA256 mySHA256 = SHA256.Create())
  519. {
  520. byte[] hashValue = mySHA256.ComputeHash(stream);
  521. return Encoding.UTF8.GetString(hashValue);
  522. }
  523. }
  524. #endregion
  525. #region 判断长期
  526. public string CheckForever(DateTime? time)
  527. {
  528. if (time == null)
  529. {
  530. return "";
  531. }
  532. if (time.Value.Year >= 2050)
  533. {
  534. return "长期";
  535. }
  536. return time.Value.ToString("yyyy-MM-dd");
  537. }
  538. #endregion
  539. }
  540. }