123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441 |
- <?php
- namespace fast\payment;
- use Exception;
- use think\Config;
- /**
- * @link https://github.com/zhangv/wechat-pay
- */
- class Wechat
- {
- const TRADETYPE_JSAPI = 'JSAPI', TRADETYPE_NATIVE = 'NATIVE', TRADETYPE_APP = 'APP';
- const URL_UNIFIEDORDER = "https://api.mch.weixin.qq.com/pay/unifiedorder";
- const URL_ORDERQUERY = "https://api.mch.weixin.qq.com/pay/orderquery";
- const URL_CLOSEORDER = 'https://api.mch.weixin.qq.com/pay/closeorder';
- const URL_REFUND = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
- const URL_REFUNDQUERY = 'https://api.mch.weixin.qq.com/pay/refundquery';
- const URL_DOWNLOADBILL = 'https://api.mch.weixin.qq.com/pay/downloadbill';
- const URL_REPORT = 'https://api.mch.weixin.qq.com/payitil/report';
- const URL_SHORTURL = 'https://api.mch.weixin.qq.com/tools/shorturl';
- const URL_MICROPAY = 'https://api.mch.weixin.qq.com/pay/micropay';
- /**
- * 错误信息
- */
- public $error = null;
- /**
- * 错误信息XML
- */
- public $errorXML = null;
- /**
- * 微信支付配置数组
- * appid 公众账号appid
- * mch_id 商户号
- * apikey 加密key
- * appsecret 公众号appsecret
- * sslcertPath 证书路径(apiclient_cert.pem)
- * sslkeyPath 密钥路径(apiclient_key.pem)
- */
- private $_config;
- /**
- * @param $options 微信支付配置数组
- */
- public function __construct($options = [])
- {
- if ($config = Config::get('payment.wechat'))
- {
- $this->config = array_merge($this->config, $config);
- }
- $this->_config = array_merge($this->_config, is_array($options) ? $options : []);
- }
- /**
- * JSAPI获取prepay_id
- *
- * @param string $body
- * @param string $out_trade_no
- * @param int $total_fee
- * @param string $openid
- * @param array $ext
- * @return string
- */
- public function getPrepayId($body, $out_trade_no, $total_fee, $openid, $ext = null)
- {
- $data = $ext? : [];
- $data["nonce_str"] = $this->getNonceStr();
- $data["body"] = $body;
- $data["out_trade_no"] = $out_trade_no;
- $data["total_fee"] = $total_fee;
- $data["spbill_create_ip"] = $_SERVER["REMOTE_ADDR"];
- $data["trade_type"] = self::TRADETYPE_JSAPI;
- $data["openid"] = $openid;
- $result = $this->unifiedOrder($data);
- if ($result["return_code"] == "SUCCESS" && $result["result_code"] == "SUCCESS")
- {
- return $result["prepay_id"];
- }
- else
- {
- $this->error = $result["return_code"] == "SUCCESS" ? $result["err_code_des"] : $result["return_msg"];
- $this->errorXML = $this->array2xml($result);
- return null;
- }
- }
- private function getNonceStr()
- {
- return substr(str_shuffle("abcdefghijklmnopqrstuvwxyz0123456789"), 0, 32);
- }
- /**
- * 统一下单接口
- */
- public function unifiedOrder($params)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["device_info"] = (isset($params['device_info']) && trim($params['device_info']) != '') ? $params['device_info'] : null;
- $data["nonce_str"] = $this->getNonceStr();
- $data["body"] = $params['body'];
- $data["detail"] = isset($params['detail']) ? $params['detail'] : null; //optional
- $data["attach"] = isset($params['attach']) ? $params['attach'] : null; //optional
- $data["out_trade_no"] = isset($params['out_trade_no']) ? $params['out_trade_no'] : null;
- $data["fee_type"] = isset($params['fee_type']) ? $params['fee_type'] : 'CNY';
- $data["total_fee"] = $params['total_fee'];
- $data["spbill_create_ip"] = $params['spbill_create_ip'];
- $data["time_start"] = isset($params['time_start']) ? $params['time_start'] : null; //optional
- $data["time_expire"] = isset($params['time_expire']) ? $params['time_expire'] : null; //optional
- $data["goods_tag"] = isset($params['goods_tag']) ? $params['goods_tag'] : null;
- $data["notify_url"] = isset($params['notify_url']) ? $params['notify_url'] : $this->_config['notify_url'];
- $data["trade_type"] = $params['trade_type'];
- $data["product_id"] = isset($params['product_id']) ? $params['product_id'] : null; //required when trade_type = NATIVE
- $data["openid"] = isset($params['openid']) ? $params['openid'] : null; //required when trade_type = JSAPI
- $result = $this->post(self::URL_UNIFIEDORDER, $data);
- return $result;
- }
- private function post($url, $data, $cert = false)
- {
- if (!isset($data['sign']))
- $data["sign"] = $this->sign($data);
- $xml = $this->array2xml($data);
- $ch = curl_init();
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
- curl_setopt($ch, CURLOPT_POST, 1);
- curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_URL, $url);
- if ($cert == true)
- {
- //使用证书:cert 与 key 分别属于两个.pem文件
- curl_setopt($ch, CURLOPT_SSLCERTTYPE, 'PEM');
- curl_setopt($ch, CURLOPT_SSLCERT, $this->_config['sslcertPath']);
- curl_setopt($ch, CURLOPT_SSLKEYTYPE, 'PEM');
- curl_setopt($ch, CURLOPT_SSLKEY, $this->_config['sslkeyPath']);
- }
- $content = curl_exec($ch);
- $array = $this->xml2array($content);
- return $array;
- }
- /**
- * 扫码支付(模式二)获取支付二维码
- *
- * @param string $body
- * @param string $out_trade_no
- * @param int $total_fee
- * @param string $product_id
- * @param array $ext
- * @return string
- */
- public function getCodeUrl($body, $out_trade_no, $total_fee, $product_id, $ext = null)
- {
- $data = $ext ? $ext : [];
- $data["nonce_str"] = $this->getNonceStr();
- $data["body"] = $body;
- $data["out_trade_no"] = $out_trade_no;
- $data["total_fee"] = $total_fee;
- $data["spbill_create_ip"] = $_SERVER["SERVER_ADDR"];
- $data["trade_type"] = self::TRADETYPE_NATIVE;
- $data["product_id"] = $product_id;
- $result = $this->unifiedOrder($data);
- if ($result["return_code"] == "SUCCESS" && $result["result_code"] == "SUCCESS")
- {
- return $result["code_url"];
- }
- else
- {
- $this->error = $result["return_code"] == "SUCCESS" ? $result["err_code_des"] : $result["return_msg"];
- return null;
- }
- }
- /**
- * 查询订单
- * @param $transaction_id
- * @param $out_trade_no
- * @return array
- */
- public function orderQuery($transaction_id, $out_trade_no)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["transaction_id"] = $transaction_id;
- $data["out_trade_no"] = $out_trade_no;
- $data["nonce_str"] = $this->getNonceStr();
- $result = $this->post(self::URL_ORDERQUERY, $data);
- return $result;
- }
- /**
- * 关闭订单
- * @param $out_trade_no
- * @return array
- */
- public function closeOrder($out_trade_no)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["out_trade_no"] = $out_trade_no;
- $data["nonce_str"] = $this->getNonceStr();
- $result = $this->post(self::URL_CLOSEORDER, $data);
- return $result;
- }
- /**
- * 申请退款 - 使用商户订单号
- * @param $out_trade_no 商户订单号
- * @param $out_refund_no 退款单号
- * @param $total_fee 总金额(单位:分)
- * @param $refund_fee 退款金额(单位:分)
- * @param $op_user_id 操作员账号
- * @return array
- */
- public function refund($out_trade_no, $out_refund_no, $total_fee, $refund_fee, $op_user_id)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["nonce_str"] = $this->getNonceStr();
- $data["out_trade_no"] = $out_trade_no;
- $data["out_refund_no"] = $out_refund_no;
- $data["total_fee"] = $total_fee;
- $data["refund_fee"] = $refund_fee;
- $data["op_user_id"] = $op_user_id;
- $result = $this->post(self::URL_REFUND, $data, true);
- return $result;
- }
- /**
- * 申请退款 - 使用微信订单号
- * @param $transaction_id 微信订单号
- * @param $out_refund_no 退款单号
- * @param $total_fee 总金额(单位:分)
- * @param $refund_fee 退款金额(单位:分)
- * @param $op_user_id 操作员账号
- * @return array
- */
- public function refundByTransId($transaction_id, $out_refund_no, $total_fee, $refund_fee, $op_user_id)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["nonce_str"] = $this->getNonceStr();
- $data["transaction_id"] = $transaction_id;
- $data["out_refund_no"] = $out_refund_no;
- $data["total_fee"] = $total_fee;
- $data["refund_fee"] = $refund_fee;
- $data["op_user_id"] = $op_user_id;
- $result = $this->post(self::URL_REFUND, $data, true);
- return $result;
- }
- /**
- * 下载对账单
- * @param $bill_date 下载对账单的日期,格式:20140603
- * @param string $bill_type 类型
- * @return array
- */
- public function downloadBill($bill_date, $bill_type = 'ALL')
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["bill_date"] = $bill_date;
- $data["bill_type"] = $bill_type;
- $data["nonce_str"] = $this->getNonceStr();
- $result = $this->post(self::URL_DOWNLOADBILL, $data);
- return $result;
- }
- /**
- * 扫码原生支付模式一中的二维码链接转成短链接
- * @param $long_url 需要转换的URL,签名用原串,传输需URLencode
- * @return array
- */
- public function shortUrl($long_url)
- {
- $data = array();
- $data["appid"] = $this->_config["appid"];
- $data["mch_id"] = $this->_config["mch_id"];
- $data["long_url"] = $long_url;
- $data["nonce_str"] = $this->getNonceStr();
- $data["sign"] = $this->sign($data);
- $data["long_url"] = urlencode($long_url);
- $result = $this->post(self::URL_SHORTURL, $data);
- return $result;
- }
- /**
- * 获取jsapi支付所需参数
- *
- * @param string $prepay_id
- * @return array
- */
- public function getPackageData($prepay_id)
- {
- $data = array();
- $data["appId"] = $this->_config["appid"];
- //解决微信支付调用JSAPI缺少参数:timeStamp
- $data["timeStamp"] = time();
- $data["nonceStr"] = $this->getNonceStr();
- $data["package"] = "prepay_id=$prepay_id";
- $data["signType"] = "MD5";
- $data["paySign"] = $this->sign($data);
- return $data;
- }
- /**
- * 获取发送到通知地址的数据(在通知地址内使用)
- * @return string 结果数组,如果不是微信服务器发送的数据返回null
- * appid
- * bank_type
- * cash_fee
- * fee_type
- * is_subscribe
- * mch_id
- * nonce_str
- * openid
- * out_trade_no 商户订单号
- * result_code
- * return_code
- * sign
- * time_end
- * total_fee 总金额
- * trade_type
- * transaction_id 微信支付订单号
- */
- public function getNotifyData()
- {
- $xml = file_get_contents("php://input");
- $data = $this->xml2array($xml);
- if ($this->validate($data))
- {
- return $data;
- }
- else
- {
- return null;
- }
- }
- /**
- * 验证数据签名
- * @param $data 数据数组
- * @return 数据校验结果
- */
- public function validate($data)
- {
- if (!isset($data["sign"]))
- {
- return false;
- }
- $sign = $data["sign"];
- unset($data["sign"]);
- return $this->sign($data) == $sign;
- }
- /**
- * 响应微信支付后台通知
- * @param string $return_code 返回状态码 SUCCESS/FAIL
- * @param $return_msg 返回信息
- */
- public function response_back($return_code = "SUCCESS", $return_msg = null)
- {
- $data = array();
- $data["return_code"] = $return_code;
- if ($return_msg)
- {
- $data["return_msg"] = $return_msg;
- }
- $xml = $this->array2xml($data);
- print $xml;
- }
- /**
- * 数据签名
- * @param $data
- * @return string
- */
- private function sign($data)
- {
- ksort($data);
- $string1 = "";
- foreach ($data as $k => $v)
- {
- if ($v && trim($v) != '')
- {
- $string1 .= "$k=$v&";
- }
- }
- $stringSignTemp = $string1 . "key=" . $this->_config["apikey"];
- $sign = strtoupper(md5($stringSignTemp));
- return $sign;
- }
- private function array2xml($array)
- {
- $xml = "<xml>" . PHP_EOL;
- foreach ($array as $k => $v)
- {
- if ($v && trim($v) != '')
- $xml .= "<$k><![CDATA[$v]]></$k>" . PHP_EOL;
- }
- $xml .= "</xml>";
- return $xml;
- }
- private function xml2array($xml)
- {
- $array = array();
- $tmp = null;
- try
- {
- $tmp = (array) simplexml_load_string($xml);
- }
- catch (Exception $e)
- {
- }
- if ($tmp && is_array($tmp))
- {
- foreach ($tmp as $k => $v)
- {
- $array[$k] = (string) $v;
- }
- }
- return $array;
- }
- }
|