我这个也是参考别人写的。有不明白的朋友可以问我
public function unifiedorder($order_no, $openid, $total_fee, $attach, $order_id, $user_id){
// 当前时间
$time = time()
// 生成随机字符串
$nonceStr = md5($time . $openid)
// API参数
$params = [
'appid' =>$this->appid, //微信分配的小程序id
'attach' =>$attach, //附加数据,在查询API和支付通知中原样返回,可作为自定义参数使用。
'body' =>'会员卡', //募捐描述
'mch_id' =>$this->mchid, //微信支付分配的商户号
'nonce_str' =>$nonceStr, //随机字符串,32位以内
'notify_url' =>$this->notify_url, // base_url() . 'notice.php?s=/task/notify/order/wxapp_id/'.$wxapp_id, // 异步通知地址
'openid' =>$openid, //用户标识;trade_type=JSAPI,此参数必传,用户在商户appid下的唯一标识。
'out_trade_no' =>$order_no, //商户账单号
'spbill_create_ip' =>\request()->ip(), //终端IP;支持IPV4和IPV6两种格式的IP地址。调用微信支付API的机器IP
'total_fee' =>(int)$total_fee * 100, // 价格:单位分 // 价格:单位分
'trade_type' =>'JSAPI', //交易类型
]
// 生成签名
$params['sign'] = $this->makeSign($params) //这个地方最坑,需要的是配置 1、appid和商户号必须是绑定的状态
// 请求API
$url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'
$result = $this->post($url, $this->toXml($params))
$prepay = $this->fromXml($result)
//添加preapay_id
$data = [
'user_id' =>$user_id,
'order_id' =>$order_id,
'attach' =>json_encode($attach),
'prepay_id' =>$prepay['prepay_id'],
]
(new AppleWxPrepay())->addInfo($data)
// 请求失败
if ($prepay['return_code'] === 'FAIL') {
return [API_CODE_NAME =>2000004, API_MSG_NAME =>$prepay['return_msg']]
}
if ($prepay['result_code'] === 'FAIL') {
return [API_CODE_NAME =>2000004, API_MSG_NAME =>$prepay['err_code_des']]
}
// 生成 nonce_str 供前端使用
$paySign = $this->makePaySign($params['nonce_str'], $prepay['prepay_id'], $time)
return [
'prepay_id' =>$prepay['prepay_id'],
'nonceStr' =>$nonceStr,
'timeStamp' =>(string)$time,
'paySign' =>$paySign
]
}
/**
* 生成签名
* @param $values
* @return string 本函数不覆盖sign成员变量,如要设置签名需要调用SetSign方法赋值
*/
private function makeSign($values)
{
//签名步骤一:按字典序排序参数
ksort($values)
$string = $this->toUrlParams($values)
//签名步骤二:在string后加入KEY
$string = $string . '&key=' . $this->apikey
//签名步骤三:MD5加密
$string = md5($string)
//签名步骤四:所有字符转为大写
$result = strtoupper($string)
return $result
}
/**
* 格式化参数格式化成url参数
* @param $values
* @return string
*/
private function toUrlParams($values)
{
$buff = ''
foreach ($values as $k =>$v) {
if ($k != 'sign' &&$v != '' &&!is_array($v)) {
$buff .= $k . '=' . $v . '&'
}
}
return trim($buff, '&')
}
/**
* 模拟POST请求
* @param $url
* @param array $data
* @param bool $useCert
* @param array $sslCert
* @return mixed
*/
public function post($url, $data = [], $useCert = false, $sslCert = [])
{
$header = [
'Content-type: application/jsoncharset=UTF8'
]
$curl = curl_init()
curl_setopt($curl, CURLOPT_URL, $url)
curl_setopt($curl, CURLOPT_HTTPHEADER, $header)
curl_setopt($curl, CURLOPT_HEADER, false)
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1)
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false)
curl_setopt($curl, CURLOPT_POST, TRUE)
curl_setopt($curl, CURLOPT_POSTFIELDS, $data)
if ($useCert == true) {
// 设置证书:cert 与 key 分别属于两个.pem文件
curl_setopt($curl, CURLOPT_SSLCERTTYPE, 'PEM')
curl_setopt($curl, CURLOPT_SSLCERT, $sslCert['certPem'])
curl_setopt($curl, CURLOPT_SSLKEYTYPE, 'PEM')
curl_setopt($curl, CURLOPT_SSLKEY, $sslCert['keyPem'])
}
$result = curl_exec($curl)
curl_close($curl)
return $result
}
/**
* 输出xml字符
* @param $values
* @return bool|string
*/
private function toXml($values)
{
if (!is_array($values) || count($values) <= 0) {
return false
}
$xml = "<xml>"
foreach ($values as $key =>$val) {
if (is_numeric($val)) {
$xml .= "<" . $key . ">" . $val . "</" . $key . ">"
} else {
$xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">"
}
}
$xml .= "</xml>"
return $xml
}
/**
* 将xml转为array
* @param $xml
* @return mixed
*/
private function fromXml($xml)
{
// 禁止引用外部xml实体
libxml_disable_entity_loader(true)
return json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true)
}
/**
* 生成paySign
* @param $nonceStr
* @param $prepay_id
* @param $timeStamp
* @return string
*/
private function makePaySign($nonceStr, $prepay_id, $timeStamp)
{
$data = [
'appId' =>$this->appid,
'nonceStr' =>$nonceStr,
'package' =>'prepay_id=' . $prepay_id,
'signType' =>'MD5',
'timeStamp' =>$timeStamp,
]
// 签名步骤一:按字典序排序参数
ksort($data)
$string = $this->toUrlParams($data)
// 签名步骤二:在string后加入KEY
$string = $string . '&key=' . $this->apikey
// 签名步骤三:MD5加密
$string = md5($string)
// 签名步骤四:所有字符转为大写
$result = strtoupper($string)
return $result
}
/*********************************微信回调**********************/
public function getNotify()
{
if (!$xml = file_get_contents('php://input')) {
$this->returnCode(50000001, 'Not found DATA')
}
// 将服务器返回的XML数据转化为数组
$data = $this->fromXml($xml)
$payLog = new ApplePayLog()
// 记录日志
$payLog->addInfo(['content'=>json_encode($xml)])
$payLog->addInfo(['content'=>json_encode($data)])
// 实例化账单模型
$OrderModel = new AppleOrder()
// 账单信息
$orderInfo = $OrderModel->getInfo(['id'=>$data['attach']],'*')
if (empty($orderInfo)) {
$this->returnCode(50000001, '账单不存在')
}
if($orderInfo['pay_status'] != 1 || !empty($orderInfo['pay_time'])){
$this->returnCode(50000001,'订单已支付,请勿再次支付')
}
// 保存微信服务器返回的签名sign
$dataSign = $data['sign']
$return_code = $data['return_code']
$result_code = $data['result_code']
$data['body'] = '会员卡'
$data['spbill_create_ip'] = \request()->ip()
$data['notify_url'] = $this->notify_url
// sign 与 s 参数 不参与签名算法
unset($data['sign'])
unset($data['transaction_id'])
unset($data['coupon_id'])
unset($data['coupon_type'])
unset($data['coupon_count'])
unset($data['coupon_fee'])
unset($data['time_end'])
unset($data['return_code'])
unset($data['result_code'])
unset($data['is_subscribe'])
unset($data['fee_type'])
unset($data['bank_type'])
unset($data['bank_type'])
// 生成签名
$sign = $this->makeSign($data)
// 判断签名是否正确 判断支付状态
if (($sign === $dataSign) &&($return_code == 'SUCCESS') &&($result_code == 'SUCCESS')) {
$OrderModel->startTrans()
try {
// 账单支付成功业务处理
$appleOrderInfo = $OrderModel->where(['id'=>$orderInfo['id']])->lock(true)->find()
$result = $appleOrderInfo->addInfo(['pay_status'=>2,'pay_time'=>time()],['id'=>$orderInfo['id']])
if(!$result){
$OrderModel->rollback()
$this->returnCode(5000003, '修改订单失败,失败原因:'.$OrderModel->getError())
}
$appleUserModel = new AppleUser()
$appleUserInfo = $appleUserModel->where(['openid'=>$orderInfo['openid']])->lock(true)->find()
$appleUser = $appleUserInfo->where(['openid'=>$orderInfo['openid']])->setInc('moxibustion',$orderInfo['moxibustion'])
if(!$appleUser){
$OrderModel->rollback()
$this->returnCode(5000003, '添加会员针灸次数失败,失败原因:'.$appleUserModel->getError())
}
}catch (\Exception $exception){
$OrderModel->rollback()
$this->returnCode(5000003, ' *** 作失败,失败原因:'.$exception->getMessage())
}
$OrderModel->commit()
// 返回状态
die(json(['code'=>0,'支付成功']))
}
// 返回状态
$this->returnCode(2000003, '签名失败')
}
以thinkphp5.0为例。小程序可以视为前端,使用异步请求(wx.request)的方式获取thinkphp(服务端)的接口。
thinkphp方面。只要设置好各个接口就行了。
在控制器中使用$this->result返回封装后的api数据到客户端(小程序)。返回的是json格式数据。
第一步 先把文件夹的那两个图片 配置成一样的路径 除了域名要改 其他保持一致。第二步 把 Weixinpay 这个文件夹放在 \ThinkPHP\Library\Vendor 将Weixinpay文件夹放置到这个Vendor文件夹中
第三步 把 WxJsAPIController.class.php 这个php文件 \Home\Controller 这里面
第四步 把 WxJsAPI这个文件夹 放置在 \Home\View 这个里面
第五步 成功调用微信公众号支付功能
jsApiCall.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/htmlcharset=utf-8" />
<meta name='viewport' content='width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0'/>
<title>微信安全支付</title>
<script type="text/javascript">
//调用微信JS api 支付
function jsApiCall()
{
WeixinJSBridge.invoke(
'getBrandWCPayRequest',
<?php echo $jsApiParameters?>,
function(res){
WeixinJSBridge.log(res.err_msg)
if(res.err_msg == "get_brand_wcpay_request:ok"){
//alert(res.err_code+res.err_desc+res.err_msg)
/*这里写如果支付成功的话执行什么 *** 作*/
}else{
//返回跳转到订单详情页面
alert(支付失败)
}
}
)
}
function callpay()
{
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', jsApiCall, false)
}else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', jsApiCall)
document.attachEvent('onWeixinJSBridgeReady', jsApiCall)
}
}else{
jsApiCall()
}
}
</script>
</head>
<body>
<button style="width:210pxheight:30pxbackground-color:#FE6714border:0px #FE6714 solidcursor: pointercolor:whitefont-size:16px" type="button" onclick="jsApiCall()" id="asd" >购买</button>
</p>
</body>
</html>
WxJsAPIController.class.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
<?php
namespace Home\Controller
use Think\Controller
class WxJsAPIController extends Controller{
public function _initialize()
{
//引入WxPayPubHelper
vendor('Weixinpay.WxPayPubHelper')
}
public function jsApiCall()
{
//使用jsapi接口
$jsApi = new \JsApi_pub()
//=========步骤1:网页授权获取用户openid============
//通过code获得openid
//
//
if (!isset($_GET['code']))
{
//触发微信返回code码
$rUrl=urlencode(C('WxConfig.JS_API_CALL_URL').'/id/'.$_GET[id])
/*注意一下这个 id 是为了可以成功的传个 id值才这样写 如果你不需要的话也可以直接写成
$rUrl=urlencode(C('WxConfig.JS_API_CALL_URL'))
$this->redirect('WxJsAPI/jsApiCall',array('id' =>55),0, '页面跳转中...')
我是这样穿这个id过来的值得你们也可以自己改成其他的来传这个id 或是删除他不要
*/
$url = $jsApi->createOauthUrlForCode($rUrl)
// echo $urlexit()
Header("Location: $url")
}else
{
//获取code码,以获取openid
$code = $_GET['code']
$jsApi->setCode($code)
$openid = $jsApi->getOpenId()//openid 这里是为了获取用户当前的openid 如果你有做微信登陆的话就可以无视他。
}
//=========步骤2:使用统一支付接口,获取prepay_id============
//使用统一支付接口
$unifiedOrder = new \UnifiedOrder_pub()
/*此处做数据库的查询 这里 *** 作数据库把产品信息显示出来*/
/*此处做数据库的查询 这里 *** 作数据库把产品信息显示出来*/
//设置统一支付接口参数
//设置必填参数
//appid已填,商户无需重复填写
//mch_id已填,商户无需重复填写
//noncestr已填,商户无需重复填写
//spbill_create_ip已填,商户无需重复填写
//sign已填,商户无需重复填写
//
$NOTIFY_URL="你的域名/index.php/Home/WxJsAPI/notify"
$unifiedOrder->setParameter("openid",$openid)//openid
$unifiedOrder->setParameter("body",'商品的名字')//商品描述
$unifiedOrder->setParameter("out_trade_no",'123456789')//商户订单号
$unifiedOrder->setParameter("total_fee",1*100)//总金额 微信的钱1*100等于1
$unifiedOrder->setParameter("notify_url",$NOTIFY_URL)//通知地址
$unifiedOrder->setParameter("trade_type","JSAPI")//交易类型
//非必填参数,商户可根据实际情况选填
//$unifiedOrder->setParameter("sub_mch_id","XXXX")//子商户号
//$unifiedOrder->setParameter("device_info","XXXX")//设备号
//$unifiedOrder->setParameter("attach","XXXX")//附加数据
//$unifiedOrder->setParameter("time_start","XXXX")//交易起始时间
//$unifiedOrder->setParameter("time_expire","XXXX")//交易结束时间
//$unifiedOrder->setParameter("goods_tag","XXXX")//商品标记
//$unifiedOrder->setParameter("openid","XXXX")//用户标识
//$unifiedOrder->setParameter("product_id","XXXX")//商品ID
$prepay_id = $unifiedOrder->getPrepayId()
// echo $prepay_idexit()
//=========步骤3:使用jsapi调起支付============
$jsApi->setPrepayId($prepay_id)
$jsApiParameters = $jsApi->getParameters()
$WEB_HOST='你的域名'//填写的话 如 <a rel="nofollow" href="http://nicaicai.imwork.net" target="_blank">http://nicaicai.imwork.net</a>最后面不用加 /
$this->assign('HOSTS',$WEB_HOST)
$this->assign('jsApiParameters',$jsApiParameters)
$this->display()
//echo $jsApiParameters
}
public function notify()
{
//使用通用通知接口
$notify = new \Notify_pub()
//存储微信的回调
$xml = $GLOBALS['HTTP_RAW_POST_DATA']
$notify->saveData($xml)
//验证签名,并回应微信。
//对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,
//微信会通过一定的策略(如30分钟共8次)定期重新发起通知,
//尽可能提高通知的成功率,但微信不保证通知最终能成功。
if($notify->checkSign() == FALSE){
$notify->setReturnParameter("return_code","FAIL")//返回状态码
$notify->setReturnParameter("return_msg","签名失败了啊")//返回信息
}else{
$notify->setReturnParameter("return_code","SUCCESS")//设置返回码
}
$returnXml = $notify->returnXml()
echo $returnXml
//==商户根据实际情况设置相应的处理流程,此处仅作举例=======
//以log文件形式记录回调信息
// $log_ = new Log_()
$log_name= ROOT."/Public/notify_url.log"//log文件路径
if($notify->checkSign() == TRUE)
{
if ($notify->data["return_code"] == "FAIL") {
//此处应该更新一下订单状态,商户自行增删 *** 作
log_result($log_name,"【通信出错】:\n".$xml."\n")
}
elseif($notify->data["result_code"] == "FAIL"){
//此处应该更新一下订单状态,商户自行增删 *** 作
log_result($log_name,"【业务出错】:\n".$xml."\n")
}
else{
/*查看支付成功的返回值请去 <a rel="nofollow" href="https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1" target="_blank">https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1</a>*/
$xmlss=$notify->data["out_trade_no"]//订单号
$total_fee=$notify->data['total_fee']//订单总金额,单位为分,详见支付金额
/*更新订单状态这里写数据库的 *** 作*/
/*更新订单状态这里写数据库的 *** 作*/
//此处应该更新一下订单状态,商户自行增删 *** 作
log_result($log_name,"【支付成功】:\n".$xml."\n")
}
//商户自行增加处理流程,
//例如:更新订单状态
//例如:数据库 *** 作
//例如:推送支付完成信息
}
}
// 打印log
public function log_result($file,$word)
{
$fp = fopen($file,"a")
flock($fp, LOCK_EX)
fwrite($fp,"执行日期:".strftime("%Y-%m-%d-%H:%M:%S",time())."\n".$word."\n\n")
flock($fp, LOCK_UN)
fclose($fp)
}
}
?>
希望我的回答可以帮到您哦
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)