alipay_submit.class.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. <?php
  2. /* *
  3. * 类名:AlipaySubmit
  4. * 功能:支付宝各接口请求提交类
  5. * 详细:构造支付宝各接口表单HTML文本,获取远程HTTP数据
  6. * 版本:3.3
  7. * 日期:2012-07-23
  8. * 说明:
  9. * 以下代码只是为了方便商户测试而提供的样例代码,商户可以根据自己网站的需要,按照技术文档编写,并非一定要使用该代码。
  10. * 该代码仅供学习和研究支付宝接口使用,只是提供一个参考。
  11. */
  12. require_once("alipay_core.function.php");
  13. require_once("alipay_rsa.function.php");
  14. require_once("alipay_md5.function.php");
  15. class AlipaySubmit {
  16. var $alipay_config;
  17. /**
  18. *支付宝网关地址
  19. */
  20. //var $alipay_gateway_new = 'https://mapi.alipay.com/gateway.do?';
  21. var $alipay_gateway_new = 'http://wappaygw.alipay.com/service/rest.htm?';
  22. function __construct($alipay_config){
  23. $this->alipay_config = $alipay_config;
  24. }
  25. function AlipaySubmit($alipay_config) {
  26. $this->__construct($alipay_config);
  27. }
  28. /**
  29. * 生成签名结果
  30. * @param $para_sort 已排序要签名的数组
  31. * return 签名结果字符串
  32. */
  33. function buildRequestMysign($para_sort) {
  34. //把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
  35. $prestr = createLinkstring($para_sort);
  36. $mysign = "";
  37. switch (strtoupper(trim($this->alipay_config['sign_type']))) {
  38. case "MD5" :
  39. $mysign = md5Sign($prestr, $this->alipay_config['key']);
  40. break;
  41. case "RSA" :
  42. $mysign = rsaSign($prestr, $this->alipay_config['private_key_path']);
  43. break;
  44. case "0001" :
  45. $mysign = rsaSign($prestr, $this->alipay_config['private_key_path']);
  46. break;
  47. default :
  48. $mysign = "";
  49. }
  50. return $mysign;
  51. }
  52. /**
  53. * 生成要请求给支付宝的参数数组
  54. * @param $para_temp 请求前的参数数组
  55. * @return 要请求的参数数组
  56. */
  57. function buildRequestPara($para_temp) {
  58. //除去待签名参数数组中的空值和签名参数
  59. $para_filter = paraFilter($para_temp);
  60. //对待签名参数数组排序
  61. $para_sort = argSort($para_filter);
  62. //生成签名结果
  63. $mysign = $this->buildRequestMysign($para_sort);
  64. //签名结果与签名方式加入请求提交参数组中
  65. $para_sort['sign'] = $mysign;
  66. if($para_sort['service'] != 'alipay.wap.trade.create.direct' && $para_sort['service'] != 'alipay.wap.auth.authAndExecute') {
  67. $para_sort['sign_type'] = strtoupper(trim($this->alipay_config['sign_type']));
  68. }
  69. return $para_sort;
  70. }
  71. /**
  72. * 生成要请求给支付宝的参数数组
  73. * @param $para_temp 请求前的参数数组
  74. * @return 要请求的参数数组字符串
  75. */
  76. function buildRequestParaToString($para_temp) {
  77. //待请求参数数组
  78. $para = $this->buildRequestPara($para_temp);
  79. //把参数组中所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串,并对字符串做urlencode编码
  80. $request_data = createLinkstringUrlencode($para);
  81. return $request_data;
  82. }
  83. /**
  84. * 建立请求,以表单HTML形式构造(默认)
  85. * @param $para_temp 请求参数数组
  86. * @param $method 提交方式。两个值可选:post、get
  87. * @param $button_name 确认按钮显示文字
  88. * @return 提交表单HTML文本
  89. */
  90. function buildRequestForm($para_temp, $method, $button_name) {
  91. //待请求参数数组
  92. $para = $this->buildRequestPara($para_temp);
  93. $sHtml = "<form id='alipaysubmit' name='alipaysubmit' action='".$this->alipay_gateway_new."_input_charset=".trim(strtolower($this->alipay_config['input_charset']))."' method='".$method."'>";
  94. while (list ($key, $val) = each ($para)) {
  95. $sHtml.= "<input type='hidden' name='".$key."' value='".$val."'/>";
  96. }
  97. //submit按钮控件请不要含有name属性
  98. $sHtml = $sHtml."<input type='submit' value='".$button_name."'></form>";
  99. $sHtml = $sHtml."<script>document.forms['alipaysubmit'].submit();</script>";
  100. return $sHtml;
  101. }
  102. /**
  103. * 建立请求,以模拟远程HTTP的POST请求方式构造并获取支付宝的处理结果
  104. * @param $para_temp 请求参数数组
  105. * @return 支付宝处理结果
  106. */
  107. function buildRequestHttp($para_temp) {
  108. $sResult = '';
  109. //待请求参数数组字符串
  110. $request_data = $this->buildRequestPara($para_temp);
  111. //远程获取数据
  112. $sResult = getHttpResponsePOST($this->alipay_gateway_new, $this->alipay_config['cacert'],$request_data,trim(strtolower($this->alipay_config['input_charset'])));
  113. return $sResult;
  114. }
  115. /**
  116. * 建立请求,以模拟远程HTTP的POST请求方式构造并获取支付宝的处理结果,带文件上传功能
  117. * @param $para_temp 请求参数数组
  118. * @param $file_para_name 文件类型的参数名
  119. * @param $file_name 文件完整绝对路径
  120. * @return 支付宝返回处理结果
  121. */
  122. function buildRequestHttpInFile($para_temp, $file_para_name, $file_name) {
  123. //待请求参数数组
  124. $para = $this->buildRequestPara($para_temp);
  125. $para[$file_para_name] = "@".$file_name;
  126. //远程获取数据
  127. $sResult = getHttpResponsePOST($this->alipay_gateway_new, $this->alipay_config['cacert'],$para,trim(strtolower($this->alipay_config['input_charset'])));
  128. return $sResult;
  129. }
  130. /**
  131. * 解析远程模拟提交后返回的信息
  132. * @param $str_text 要解析的字符串
  133. * @return 解析结果
  134. */
  135. function parseResponse($str_text) {
  136. //以“&”字符切割字符串
  137. $para_split = explode('&',$str_text);
  138. //把切割后的字符串数组变成变量与数值组合的数组
  139. foreach ($para_split as $item) {
  140. //获得第一个=字符的位置
  141. $nPos = strpos($item,'=');
  142. //获得字符串长度
  143. $nLen = strlen($item);
  144. //获得变量名
  145. $key = substr($item,0,$nPos);
  146. //获得数值
  147. $value = substr($item,$nPos+1,$nLen-$nPos-1);
  148. //放入数组中
  149. $para_text[$key] = $value;
  150. }
  151. if( ! empty ($para_text['res_data'])) {
  152. //解析加密部分字符串
  153. if($this->alipay_config['sign_type'] == '0001') {
  154. $para_text['res_data'] = rsaDecrypt($para_text['res_data'], $this->alipay_config['private_key_path']);
  155. }
  156. //token从res_data中解析出来(也就是说res_data中已经包含token的内容)
  157. $doc = new DOMDocument();
  158. $doc->loadXML($para_text['res_data']);
  159. $para_text['request_token'] = $doc->getElementsByTagName( "request_token" )->item(0)->nodeValue;
  160. }
  161. return $para_text;
  162. }
  163. /**
  164. * 用于防钓鱼,调用接口query_timestamp来获取时间戳的处理函数
  165. * 注意:该功能PHP5环境及以上支持,因此必须服务器、本地电脑中装有支持DOMDocument、SSL的PHP配置环境。建议本地调试时使用PHP开发软件
  166. * return 时间戳字符串
  167. */
  168. function query_timestamp() {
  169. $url = $this->alipay_gateway_new."service=query_timestamp&partner=".trim(strtolower($this->alipay_config['partner']))."&_input_charset=".trim(strtolower($this->alipay_config['input_charset']));
  170. $encrypt_key = "";
  171. $doc = new DOMDocument();
  172. $doc->load($url);
  173. $itemEncrypt_key = $doc->getElementsByTagName( "encrypt_key" );
  174. $encrypt_key = $itemEncrypt_key->item(0)->nodeValue;
  175. return $encrypt_key;
  176. }
  177. }
  178. ?>