573 lines
16 KiB
PHP
Executable File
573 lines
16 KiB
PHP
Executable File
<?php
|
||
/**
|
||
* 微信支付
|
||
*/
|
||
|
||
include_once ("WxException.php");
|
||
include_once ("WxPayConf.php");
|
||
|
||
/**
|
||
* 所有接口的基类
|
||
*/
|
||
|
||
class CommonUtil {
|
||
function __construct() {
|
||
}
|
||
function trimString($value) {
|
||
$ret = null;
|
||
if (null != $value) {
|
||
$ret = $value;
|
||
if (strlen ( $ret ) == 0) {
|
||
$ret = null;
|
||
}
|
||
}
|
||
return $ret;
|
||
}
|
||
|
||
/**
|
||
* 产生随机字符串,不长于32位
|
||
*/
|
||
public function createNoncestr($length = 32) {
|
||
$chars = "abcdefghijklmnopqrstuvwxyz0123456789";
|
||
$str = "";
|
||
for($i = 0; $i < $length; $i ++) {
|
||
$str .= substr ( $chars, mt_rand ( 0, strlen ( $chars ) - 1 ), 1 );
|
||
}
|
||
return $str;
|
||
}
|
||
|
||
/**
|
||
* 格式化参数,签名过程需要使用
|
||
*/
|
||
function formatBizQueryParaMap($paraMap, $urlencode) {
|
||
$buff = "";
|
||
ksort ( $paraMap );
|
||
foreach ( $paraMap as $k => $v ) {
|
||
if ($urlencode) {
|
||
$v = urlencode ( $v );
|
||
}
|
||
$buff .= $k . "=" . $v . "&";
|
||
}
|
||
$reqPar;
|
||
if (strlen ( $buff ) > 0) {
|
||
$reqPar = substr ( $buff, 0, strlen ( $buff ) - 1 );
|
||
}
|
||
return $reqPar;
|
||
}
|
||
|
||
/**
|
||
* 生成签名
|
||
*/
|
||
public function getSign($Obj) {
|
||
foreach ( $Obj as $k => $v ) {
|
||
$Parameters [$k] = $v;
|
||
}
|
||
// 签名步骤一:按字典序排序参数
|
||
ksort ( $Parameters );
|
||
$String = $this->formatBizQueryParaMap ( $Parameters, false );
|
||
// 签名步骤二:在string后加入KEY
|
||
$String = $String . "&key=" . WxPayConf::$KEY;
|
||
// 签名步骤三:MD5加密
|
||
$String = md5 ( $String );
|
||
// 签名步骤四:所有字符转为大写
|
||
$result_ = strtoupper ( $String );
|
||
return $result_;
|
||
}
|
||
|
||
/**
|
||
* array转xml
|
||
*/
|
||
function arrayToXml($arr) {
|
||
$xml = "<xml>";
|
||
foreach ( $arr as $key => $val ) {
|
||
if (is_numeric ( $val )) {
|
||
$xml .= "<" . $key . ">" . $val . "</" . $key . ">";
|
||
} else
|
||
$xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">";
|
||
}
|
||
$xml .= "</xml>";
|
||
return $xml;
|
||
}
|
||
|
||
/**
|
||
* 将xml转为array
|
||
*/
|
||
public function xmlToArray($xml) {
|
||
// 将XML转为array
|
||
libxml_disable_entity_loader(true);
|
||
libxml_use_internal_errors();
|
||
$array_data = json_decode ( json_encode ( simplexml_load_string ( $xml, 'SimpleXMLElement', LIBXML_NOCDATA ) ), true );
|
||
return $array_data;
|
||
}
|
||
|
||
/**
|
||
* 以post方式提交xml到对应的接口url
|
||
*/
|
||
public function postXmlCurl($xml, $url, $second = 30) {
|
||
// 初始化curl
|
||
$ch = curl_init ();
|
||
// 设置超时
|
||
curl_setopt ( $ch, CURLOPT_TIMEOUT, $second );
|
||
// 这里设置代理,如果有的话
|
||
curl_setopt ( $ch, CURLOPT_URL, $url );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYPEER, FALSE );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYHOST, FALSE );
|
||
// 设置header
|
||
curl_setopt ( $ch, CURLOPT_HEADER, FALSE );
|
||
// 要求结果为字符串且输出到屏幕上
|
||
curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, TRUE );
|
||
// post提交方式
|
||
curl_setopt ( $ch, CURLOPT_POST, TRUE );
|
||
curl_setopt ( $ch, CURLOPT_POSTFIELDS, $xml );
|
||
// 运行curl
|
||
$data = curl_exec ( $ch );
|
||
// 返回结果
|
||
if ($data) {
|
||
curl_close ( $ch );
|
||
return $data;
|
||
} else {
|
||
$error = curl_errno ( $ch );
|
||
echo "curl出错,错误码:$error" . "<br>";
|
||
echo "<a href='http://curl.haxx.se/libcurl/c/libcurl-errors.html'>错误原因查询</a></br>";
|
||
curl_close ( $ch );
|
||
return false;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 使用证书,以post方式提交xml到对应的接口url
|
||
*/
|
||
function postXmlSSLCurl($xml, $url, $second = 30) {
|
||
$ch = curl_init ();
|
||
// 超时时间
|
||
curl_setopt ( $ch, CURLOPT_TIMEOUT, $second );
|
||
curl_setopt ( $ch, CURLOPT_URL, $url );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYPEER, FALSE );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYHOST, FALSE );
|
||
// 设置header
|
||
curl_setopt ( $ch, CURLOPT_HEADER, FALSE );
|
||
// 要求结果为字符串且输出到屏幕上
|
||
curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, TRUE );
|
||
// 设置证书
|
||
|
||
curl_setopt ( $ch, CURLOPT_SSLCERT, WxPayConf::$SSLCERT_PATH );
|
||
curl_setopt ( $ch, CURLOPT_SSLKEY, WxPayConf::$SSLKEY_PATH );
|
||
curl_setopt ( $ch, CURLOPT_CAINFO, WxPayConf::$SSLCA_PATH ); // CA根证书(用来验证的网站证书是否是CA颁布)
|
||
|
||
// post提交方式
|
||
curl_setopt ( $ch, CURLOPT_POST, true );
|
||
curl_setopt ( $ch, CURLOPT_POSTFIELDS, $xml );
|
||
$data = curl_exec ( $ch );
|
||
// 返回结果
|
||
if ($data) {
|
||
//curl_close ( $ch );
|
||
return $data;
|
||
} else {
|
||
$error = curl_errno ( $ch );
|
||
echo "curl出错,错误码:$error" . "<br>";
|
||
echo "<a href='http://curl.haxx.se/libcurl/c/libcurl-errors.html'>错误原因查询</a></br>";
|
||
curl_close ( $ch );
|
||
return false;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 打印数组
|
||
*/
|
||
function printErr($wording = '', $err = '') {
|
||
print_r ( '<pre>' );
|
||
echo $wording . "</br>";
|
||
var_dump ( $err );
|
||
print_r ( '</pre>' );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 请求型接口的基类
|
||
*/
|
||
class WxpayClient extends CommonUtil {
|
||
var $parameters; // 请求参数,类型为关联数组
|
||
public $response; // 微信返回的响应
|
||
public $result; // 返回参数,类型为关联数组
|
||
var $url; // 接口链接
|
||
var $curl_timeout; // curl超时时间
|
||
|
||
/**
|
||
* 设置请求参数
|
||
*/
|
||
function setParameter($parameter, $parameterValue) {
|
||
$this->parameters [$this->trimString ( $parameter )] = $this->trimString ( $parameterValue );
|
||
}
|
||
|
||
/**
|
||
* 设置标配的请求参数,生成签名,生成接口参数xml
|
||
*/
|
||
function createXml() {
|
||
$this->parameters ["appid"] = WxPayConf::$APPID; // 公众账号ID
|
||
$this->parameters ["mch_id"] = WxPayConf::$MCHID; // 商户号
|
||
$this->parameters ["nonce_str"] = $this->createNoncestr (); // 随机字符串
|
||
$this->parameters ["sign"] = $this->getSign ( $this->parameters ); // 签名
|
||
print_r ( $this->parameters );
|
||
return $this->arrayToXml ( $this->parameters );
|
||
}
|
||
|
||
/**
|
||
* post请求xml
|
||
*/
|
||
function postXml() {
|
||
$xml = $this->createXml ();
|
||
$this->response = $this->postXmlCurl ( $xml, $this->url, $this->curl_timeout );
|
||
return $this->response;
|
||
}
|
||
|
||
/**
|
||
* 使用证书post请求xml
|
||
*/
|
||
function postXmlSSL() {
|
||
$xml = $this->createXml ();
|
||
$this->response = $this->postXmlSSLCurl ( $xml, $this->url, $this->curl_timeout );
|
||
return $this->response;
|
||
}
|
||
|
||
/**
|
||
* 获取结果,默认不使用证书
|
||
*/
|
||
function getResult() {
|
||
$this->postXml ();
|
||
$this->result = $this->xmlToArray ( $this->response );
|
||
return $this->result;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 统一支付接口类
|
||
*/
|
||
class UnifiedOrder extends WxpayClient {
|
||
function __construct() {
|
||
// 设置接口链接
|
||
$this->url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
|
||
// 设置curl超时时间
|
||
$this->curl_timeout = WxPayConf::$CURL_TIMEOUT;
|
||
}
|
||
|
||
/**
|
||
* 生成接口参数xml
|
||
*/
|
||
function createXml() {
|
||
try {
|
||
// 检测必填参数
|
||
if ($this->parameters ["out_trade_no"] == null) {
|
||
throw new WxException ( "缺少统一支付接口必填参数out_trade_no!" . "<br>" );
|
||
} elseif ($this->parameters ["body"] == null) {
|
||
throw new WxException ( "缺少统一支付接口必填参数body!" . "<br>" );
|
||
} elseif ($this->parameters ["total_fee"] == null) {
|
||
throw new WxException ( "缺少统一支付接口必填参数total_fee!" . "<br>" );
|
||
} elseif ($this->parameters ["notify_url"] == null) {
|
||
throw new WxException ( "缺少统一支付接口必填参数notify_url!" . "<br>" );
|
||
} elseif ($this->parameters ["trade_type"] == null) {
|
||
throw new WxException ( "缺少统一支付接口必填参数trade_type!" . "<br>" );
|
||
} elseif ($this->parameters ["trade_type"] == "JSAPI" && $this->parameters ["openid"] == NULL) {
|
||
throw new WxException ( "统一支付接口中,缺少必填参数openid!trade_type为JSAPI时,openid为必填参数!" . "<br>" );
|
||
}
|
||
$this->parameters ["appid"] = WxPayConf::$APPID; // 公众账号ID
|
||
$this->parameters ["mch_id"] = WxPayConf::$MCHID; // 商户号
|
||
$this->parameters ["spbill_create_ip"] = $_SERVER ['REMOTE_ADDR']; // 终端ip
|
||
$this->parameters ["nonce_str"] = $this->createNoncestr (); // 随机字符串
|
||
$this->parameters ["sign"] = $this->getSign ( $this->parameters ); // 签名
|
||
return $this->arrayToXml ( $this->parameters );
|
||
} catch ( WxException $e ) {
|
||
die ( $e->errorMessage () );
|
||
}
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* 设置jsapi的参数
|
||
*/
|
||
public function getParameters($obj) {
|
||
$appObj ["appid"] = WxPayConf::$APPID;
|
||
$appObj ["partnerid"] = WxPayConf::$MCHID; // 商户号
|
||
$appObj ["prepayid"] = $obj["prepayid"];
|
||
$appObj ["package"] = "Sign=WXPay";
|
||
$appObj ["noncestr"] = $this->createNoncestr ();
|
||
$timeStamp = time ();
|
||
$appObj ["timestamp"] = (string)$timeStamp;
|
||
$appObj ["sign"] = $this->getSign ( $appObj );
|
||
return $appObj;
|
||
}
|
||
|
||
/**
|
||
* 获取prepay_id
|
||
*/
|
||
function getPrepayId() {
|
||
$this->postXml ();
|
||
$this->result = $this->xmlToArray ( $this->response );
|
||
|
||
$prepay_id = $this->result ["prepay_id"];
|
||
|
||
return $prepay_id;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 订单查询接口
|
||
*/
|
||
class OrderQuery extends WxpayClient {
|
||
function __construct() {
|
||
// 设置接口链接
|
||
$this->url = "https://api.mch.weixin.qq.com/pay/orderquery";
|
||
// 设置curl超时时间
|
||
$this->curl_timeout = WxPayConf::$CURL_TIMEOUT;
|
||
}
|
||
|
||
/**
|
||
* 生成接口参数xml
|
||
*/
|
||
function createXml() {
|
||
try {
|
||
// 检测必填参数
|
||
if ($this->parameters ["out_trade_no"] == null && $this->parameters ["transaction_id"] == null) {
|
||
throw new WxException ( "订单查询接口中,out_trade_no、transaction_id至少填一个!" . "<br>" );
|
||
}
|
||
$this->parameters ["appid"] = WxPayConf::$APPID; // 公众账号ID
|
||
$this->parameters ["mch_id"] = WxPayConf::$MCHID; // 商户号
|
||
$this->parameters ["nonce_str"] = $this->createNoncestr (); // 随机字符串
|
||
$this->parameters ["sign"] = $this->getSign ( $this->parameters ); // 签名
|
||
return $this->arrayToXml ( $this->parameters );
|
||
} catch ( WxException $e ) {
|
||
die ( $e->errorMessage () );
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
|
||
/**
|
||
* 响应型接口基类
|
||
*/
|
||
class WxpayServer extends CommonUtil {
|
||
public $data; // 接收到的数据,类型为关联数组
|
||
var $returnParameters; // 返回参数,类型为关联数组
|
||
|
||
/**
|
||
* 将微信的请求xml转换成关联数组,以方便数据处理
|
||
*/
|
||
function saveData($xml) {
|
||
$this->data = $this->xmlToArray ( $xml );
|
||
}
|
||
function checkSign() {
|
||
$tmpData = $this->data;
|
||
unset ( $tmpData ['sign'] );
|
||
$sign = $this->getSign ( $tmpData ); // 本地签名
|
||
if ($this->data ['sign'] == $sign) {
|
||
return TRUE;
|
||
}
|
||
return FALSE;
|
||
}
|
||
|
||
/**
|
||
* 获取微信的请求数据
|
||
*/
|
||
function getData() {
|
||
return $this->data;
|
||
}
|
||
|
||
/**
|
||
* 设置返回微信的xml数据
|
||
*/
|
||
function setReturnParameter($parameter, $parameterValue) {
|
||
$this->returnParameters [$this->trimString ( $parameter )] = $this->trimString ( $parameterValue );
|
||
}
|
||
|
||
/**
|
||
* 生成接口参数xml
|
||
*/
|
||
function createXml() {
|
||
return $this->arrayToXml ( $this->returnParameters );
|
||
}
|
||
|
||
/**
|
||
* 将xml数据返回微信
|
||
*/
|
||
function returnXml() {
|
||
$returnXml = $this->createXml ();
|
||
return $returnXml;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 通用通知接口
|
||
*/
|
||
class Notify extends WxpayServer {
|
||
}
|
||
|
||
|
||
/**
|
||
* 请求商家获取商品信息接口
|
||
*/
|
||
class NativeCall extends WxpayServer {
|
||
/**
|
||
* 生成接口参数xml
|
||
*/
|
||
function createXml() {
|
||
if ($this->returnParameters ["return_code"] == "SUCCESS") {
|
||
$this->returnParameters ["appid"] = WxPayConf::$APPID; // 公众账号ID
|
||
$this->returnParameters ["mch_id"] = WxPayConf::$MCHID; // 商户号
|
||
$this->returnParameters ["nonce_str"] = $this->createNoncestr (); // 随机字符串
|
||
$this->returnParameters ["sign"] = $this->getSign ( $this->returnParameters ); // 签名
|
||
}
|
||
return $this->arrayToXml ( $this->returnParameters );
|
||
}
|
||
|
||
/**
|
||
* 获取product_id
|
||
*/
|
||
function getProductId() {
|
||
$product_id = $this->data ["product_id"];
|
||
return $product_id;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 静态链接二维码
|
||
*/
|
||
class NativeLink extends CommonUtil {
|
||
var $parameters; // 静态链接参数
|
||
var $url; // 静态链接
|
||
function __construct() {
|
||
}
|
||
|
||
/**
|
||
* 设置参数
|
||
*/
|
||
function setParameter($parameter, $parameterValue) {
|
||
$this->parameters [$this->trimString ( $parameter )] = $this->trimString ( $parameterValue );
|
||
}
|
||
|
||
/**
|
||
* 生成Native支付链接二维码
|
||
*/
|
||
function createLink() {
|
||
try {
|
||
if ($this->parameters ["product_id"] == null) {
|
||
throw new WxException ( "缺少Native支付二维码链接必填参数product_id!" . "<br>" );
|
||
}
|
||
$this->parameters ["appid"] = WxPayConf::$APPID; // 公众账号ID
|
||
$this->parameters ["mch_id"] = WxPayConf::$MCHID; // 商户号
|
||
$time_stamp = time ();
|
||
$this->parameters ["time_stamp"] = "$time_stamp"; // 时间戳
|
||
$this->parameters ["nonce_str"] = $this->createNoncestr (); // 随机字符串
|
||
$this->parameters ["sign"] = $this->getSign ( $this->parameters ); // 签名
|
||
$bizString = $this->formatBizQueryParaMap ( $this->parameters, false );
|
||
$this->url = "weixin://wxpay/bizpayurl?" . $bizString;
|
||
} catch ( WxException $e ) {
|
||
die ( $e->errorMessage () );
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 返回链接
|
||
*/
|
||
function getUrl() {
|
||
$this->createLink ();
|
||
return $this->url;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* JSAPI支付——H5网页端调起支付接口
|
||
*/
|
||
class JsApi extends CommonUtil {
|
||
var $code; // code码,用以获取openid
|
||
var $openid; // 用户的openid
|
||
var $parameters; // jsapi参数,格式为json
|
||
var $prepay_id; // 使用统一支付接口得到的预支付id
|
||
var $curl_timeout; // curl超时时间
|
||
function __construct() {
|
||
// 设置curl超时时间
|
||
$this->curl_timeout = WxPayConf::$CURL_TIMEOUT;
|
||
}
|
||
|
||
/**
|
||
* 生成可以获得code的url
|
||
*/
|
||
function createOauthUrlForCode($redirectUrl) {
|
||
$urlObj ["appid"] = WxPayConf::$APPID;
|
||
$urlObj ["redirect_uri"] = "$redirectUrl";
|
||
$urlObj ["response_type"] = "code";
|
||
$urlObj ["scope"] = "snsapi_base";
|
||
$urlObj ["state"] = "STATE" . "#wechat_redirect";
|
||
$bizString = $this->formatBizQueryParaMap ( $urlObj, false );
|
||
return "https://open.weixin.qq.com/connect/oauth2/authorize?" . $bizString;
|
||
}
|
||
|
||
/**
|
||
* 生成可以获得openid的url
|
||
*/
|
||
function createOauthUrlForOpenid() {
|
||
$urlObj ["appid"] = WxPayConf::$APPID;
|
||
$urlObj ["secret"] = WxPayConf::$APPSECRET;
|
||
$urlObj ["code"] = $this->code;
|
||
$urlObj ["grant_type"] = "authorization_code";
|
||
$bizString = $this->formatBizQueryParaMap ( $urlObj, false );
|
||
return "https://api.weixin.qq.com/sns/oauth2/access_token?" . $bizString;
|
||
}
|
||
|
||
/**
|
||
* 通过curl向微信提交code,以获取openid
|
||
*/
|
||
function getOpenid() {
|
||
$url = $this->createOauthUrlForOpenid ();
|
||
// 初始化curl
|
||
$ch = curl_init ();
|
||
// 设置超时
|
||
curl_setopt ( $ch, CURLOPT_TIMEOUT, $this->curl_timeout );
|
||
curl_setopt ( $ch, CURLOPT_URL, $url );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYPEER, FALSE );
|
||
curl_setopt ( $ch, CURLOPT_SSL_VERIFYHOST, FALSE );
|
||
curl_setopt ( $ch, CURLOPT_HEADER, FALSE );
|
||
curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, TRUE );
|
||
// 运行curl,结果以jason形式返回
|
||
$res = curl_exec ( $ch );
|
||
curl_close ( $ch );
|
||
// 取出openid
|
||
$data = json_decode ( $res, true );
|
||
$this->openid = $data ['openid'];
|
||
return $this->openid;
|
||
}
|
||
|
||
/**
|
||
* 设置prepay_id
|
||
*/
|
||
function setPrepayId($prepayId) {
|
||
$this->prepay_id = $prepayId;
|
||
}
|
||
|
||
/**
|
||
* 设置code
|
||
*/
|
||
function setCode($code_) {
|
||
$this->code = $code_;
|
||
}
|
||
|
||
/**
|
||
* 设置jsapi的参数
|
||
*/
|
||
public function getParameters() {
|
||
$jsApiObj ["appId"] = WxPayConf::$APPID;
|
||
$timeStamp = time ();
|
||
$jsApiObj ["timeStamp"] = (string)$timeStamp;
|
||
$jsApiObj ["nonceStr"] = $this->createNoncestr ();
|
||
$jsApiObj ["package"] = "prepay_id=$this->prepay_id";
|
||
$jsApiObj ["signType"] = "MD5";
|
||
$jsApiObj ["paySign"] = $this->getSign ( $jsApiObj );
|
||
$this->parameters = json_encode ( $jsApiObj );
|
||
|
||
return $this->parameters;
|
||
}
|
||
}
|
||
|
||
?>
|