pay/plugins/sandpay/sandpay_plugin.php
2025-11-28 10:08:12 +08:00

506 lines
17 KiB
PHP
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<?php
class sandpay_plugin
{
static public $info = [
'name' => 'sandpay', //支付插件英文名称,需和目录名称一致,不能有重复
'showname' => '杉德支付', //支付插件显示名称
'author' => '杉德', //支付插件作者
'link' => 'https://www.sandpay.com.cn/', //支付插件作者链接
'types' => ['alipay','wxpay','bank'], //支付插件支持的支付方式可选的有alipay,qqpay,wxpay,bank
'transtypes' => ['bank'], //支付插件支持的转账方式可选的有alipay,qqpay,wxpay,bank
'inputs' => [ //支付插件要求传入的参数以及参数显示名称可选的有appid,appkey,appsecret,appurl,appmchid
'appid' => [
'name' => '商户编号',
'type' => 'input',
'note' => '',
],
'appkey' => [
'name' => '私钥证书密码',
'type' => 'input',
'note' => '',
],
'appswitch' => [
'name' => '环境选择',
'type' => 'select',
'options' => [0=>'生产环境',1=>'测试环境'],
],
'product' => [
'name' => '市场产品',
'type' => 'select',
'options' => ['QZF'=>'标准线上收款','CSDB'=>'企业杉德宝'],
],
],
'select_alipay' => [
'1' => '扫码支付',
'2' => 'JS支付',
],
'select_bank' => [
'1' => '银联聚合码',
'2' => '快捷支付',
],
'select' => null,
'note' => '将杉德公钥证书sand.cer、商户私钥证书client.pfx或商户编号.pfx上传到 /plugins/sandpay/cert/', //支付密钥填写说明
'bindwxmp' => true, //是否支持绑定微信公众号
'bindwxa' => true, //是否支持绑定微信小程序
];
static public function submit(){
global $siteurl, $channel, $order, $sitename;
if($order['typename']=='alipay'){
if(checkalipay() && in_array('2',$channel['apptype'])){
return ['type'=>'jump','url'=>'/pay/alipayjs/'.TRADE_NO.'/?d=1'];
}else{
return ['type'=>'jump','url'=>'/pay/alipay/'.TRADE_NO.'/'];
}
}elseif($order['typename']=='wxpay'){
if(checkwechat() && $channel['appwxmp']>0){
return ['type'=>'jump','url'=>'/pay/wxjspay/'.TRADE_NO.'/?d=1'];
}elseif(checkmobile() && $channel['appwxa']>0){
return ['type'=>'jump','url'=>'/pay/wxwappay/'.TRADE_NO.'/'];
}else{
return ['type'=>'jump','url'=>'/pay/wxpay/'.TRADE_NO.'/'];
}
}elseif($order['typename']=='bank'){
if(in_array('2',$channel['apptype'])){
return ['type'=>'jump','url'=>'/pay/fastpay/'.TRADE_NO.'/'];
}else{
return ['type'=>'jump','url'=>'/pay/bank/'.TRADE_NO.'/'];
}
}
}
static public function mapi(){
global $siteurl, $channel, $order, $conf, $device, $mdevice, $method;
if($order['typename']=='alipay'){
if($mdevice=='alipay' && in_array('2',$channel['apptype'])){
return ['type'=>'jump','url'=>$siteurl.'pay/alipayjs/'.TRADE_NO.'/?d=1'];
}else{
return self::alipay();
}
}elseif($order['typename']=='wxpay'){
if($mdevice=='wechat' && $channel['appwxmp']>0){
return ['type'=>'jump','url'=>$siteurl.'/pay/wxjspay/'.TRADE_NO.'/?d=1'];
}elseif($device=='mobile' && $channel['appwxa']>0){
return ['type'=>'jump','url'=>$siteurl.'/pay/wxwappay/'.TRADE_NO.'/'];
}else{
return self::wxpay();
}
}elseif($order['typename']=='bank'){
if(in_array('2',$channel['apptype'])){
return self::fastpay();
}else{
return self::bank();
}
}
}
//统一下单
static private function addOrder($pay_type, $pay_mode, $sub_openid = null, $sub_appid = null){
global $channel, $order, $ordername, $conf, $clientip, $siteurl;
require(PAY_ROOT."inc/SandpayClient.php");
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$params = [
'marketProduct' => $channel['product'],
'outReqTime' => date('YmdHis'),
'mid' => $channel['appid'],
'outOrderNo' => TRADE_NO,
'description' => $ordername,
'goodsClass' => '01',
'amount' => $order['realmoney'],
'payType' => $pay_type,
'payMode' => $pay_mode,
'payerInfo' => [
'payAccLimit' => '',
],
'notifyUrl' => $conf['localurl'].'pay/notify/'.TRADE_NO.'/',
'riskmgtInfo' => [
'sourceIp' => $clientip,
],
];
if($sub_openid && $sub_appid){
$params['payerInfo'] = [
'subAppId' => $sub_appid,
'subUserId' => $sub_openid,
'frontUrl' => $siteurl.'pay/return/'.TRADE_NO.'/',
];
}elseif($sub_openid){
$params['payerInfo'] = [
'userId' => $sub_openid,
'frontUrl' => $siteurl.'pay/return/'.TRADE_NO.'/',
];
}
return \lib\Payment::lockPayData(TRADE_NO, function() use($client, $params) {
global $channel;
$result = $client->execute('/v4/sd-receipts/api/trans/trans.order.create', $params);
\lib\Payment::updateOrder(TRADE_NO, $result['sandSerialNo']);
if($channel['appswitch']==1){
$log = "商户订单号:".TRADE_NO."\r\n【统一下单接口】请求报文:\r\n".$client->request_body."\r\n【统一下单接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
return $result['credential'];
});
}
//支付宝扫码支付
static public function alipay(){
global $channel, $device, $mdevice, $siteurl;
if(in_array('2',$channel['apptype']) && !in_array('1',$channel['apptype'])){
$code_url = $siteurl.'pay/alipayjs/'.TRADE_NO.'/';
}else{
try{
$result = self::addOrder('ALIPAY','QR');
$code_url = $result['qrCode'];
}catch(Exception $ex){
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
}
}
if(checkalipay() || $mdevice=='alipay'){
return ['type'=>'jump','url'=>$code_url];
}else{
return ['type'=>'qrcode','page'=>'alipay_qrcode','url'=>$code_url];
}
}
static public function alipayjs(){
[$user_type, $user_id] = alipay_oauth();
$blocks = checkBlockUser($user_id, TRADE_NO);
if($blocks) return $blocks;
if($user_type == 'openid'){
return ['type'=>'error','msg'=>'支付宝快捷登录获取uid失败需将用户标识切换到uid模式'];
}
try{
$result = self::addOrder('ALIPAY', 'JSAPI', $user_id);
}catch(Exception $ex){
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
}
if($_GET['d']=='1'){
$redirect_url='data.backurl';
}else{
$redirect_url='\'/pay/ok/'.TRADE_NO.'/\'';
}
return ['type'=>'page','page'=>'alipay_jspay','data'=>['alipay_trade_no'=>$result['tradeNo'], 'redirect_url'=>$redirect_url]];
}
//微信扫码支付
static public function wxpay(){
global $channel, $siteurl, $device, $mdevice;
try{
$result = self::addOrder('WXPAY','QR');
$code_url = $result['qrCode'];
}catch(Exception $ex){
return ['type'=>'error','msg'=>'微信支付下单失败!'.$ex->getMessage()];
}
if (checkwechat() || $mdevice=='wechat') {
return ['type'=>'jump','url'=>$code_url];
} elseif (checkmobile() || $device == 'mobile') {
return ['type'=>'qrcode','page'=>'wxpay_wap','url'=>$code_url];
} else {
return ['type'=>'qrcode','page'=>'wxpay_qrcode','url'=>$code_url];
}
}
//微信公众号
static public function wxjspay(){
global $siteurl, $channel, $order, $ordername, $conf, $clientip;
$wxinfo = \lib\Channel::getWeixin($channel['appwxmp']);
if(!$wxinfo) return ['type'=>'error','msg'=>'支付通道绑定的微信公众号不存在'];
try{
$tools = new \WeChatPay\JsApiTool($wxinfo['appid'], $wxinfo['appsecret']);
$openid = $tools->GetOpenid();
}catch(Exception $e){
return ['type'=>'error','msg'=>$e->getMessage()];
}
$blocks = checkBlockUser($openid, TRADE_NO);
if($blocks) return $blocks;
try{
$payinfo = self::addOrder('WXPAY','JSAPI',$openid,$wxinfo['appid']);
}catch(Exception $ex){
return ['type'=>'error','msg'=>'微信支付下单失败!'.$ex->getMessage()];
}
if($_GET['d']==1){
$redirect_url='data.backurl';
}else{
$redirect_url='\'/pay/ok/'.TRADE_NO.'/\'';
}
return ['type'=>'page','page'=>'wxpay_jspay','data'=>['jsApiParameters'=>json_encode($payinfo), 'redirect_url'=>$redirect_url]];
}
//微信小程序支付
static public function wxminipay(){
global $siteurl, $channel, $order, $ordername, $conf, $clientip;
$code = isset($_GET['code'])?trim($_GET['code']):exit('{"code":-1,"msg":"code不能为空"}');
//①、获取用户openid
$wxinfo = \lib\Channel::getWeixin($channel['appwxa']);
if(!$wxinfo)exit('{"code":-1,"msg":"支付通道绑定的微信小程序不存在"}');
try{
$tools = new \WeChatPay\JsApiTool($wxinfo['appid'], $wxinfo['appsecret']);
$openid = $tools->AppGetOpenid($code);
}catch(Exception $e){
exit('{"code":-1,"msg":"'.$e->getMessage().'"}');
}
$blocks = checkBlockUser($openid, TRADE_NO);
if($blocks)exit('{"code":-1,"msg":"'.$blocks['msg'].'"}');
//②、统一下单
try{
$payinfo = self::addOrder('WXPAY','MINI',$openid,$wxinfo['appid']);
}catch(Exception $ex){
exit('{"code":-1,"msg":"微信支付下单失败!'.$ex->getMessage().'"}');
}
exit(json_encode(['code'=>0, 'data'=>json_decode($payinfo, true)]));
}
//微信手机支付
static public function wxwappay(){
global $siteurl,$channel, $order, $ordername, $conf, $clientip;
$wxinfo = \lib\Channel::getWeixin($channel['appwxa']);
if(!$wxinfo) return ['type'=>'error','msg'=>'支付通道绑定的微信小程序不存在'];
try{
$code_url = wxminipay_jump_scheme($wxinfo['id'], TRADE_NO);
}catch(Exception $e){
return ['type'=>'error','msg'=>$e->getMessage()];
}
return ['type'=>'scheme','page'=>'wxpay_mini','url'=>$code_url];
}
//云闪付扫码支付
static public function bank(){
try{
$result = self::addOrder('CUPPAY','QR');
$code_url = $result['qrCode'];
}catch(Exception $ex){
return ['type'=>'error','msg'=>'云闪付下单失败!'.$ex->getMessage()];
}
return ['type'=>'qrcode','page'=>'bank_qrcode','url'=>$code_url];
}
//快捷支付
static public function fastpay(){
if(!empty($_COOKIE['sandpay_user_id'])){
$user_id = $_COOKIE['sandpay_user_id'];
}else{
$user_id = substr(getSid(), 0, 10);
setcookie('sandpay_user_id', $user_id, time()+3600*24*365, '/');
}
try{
$result = self::addOrder('FASTPAY','SANDH5',$user_id);
$jump_url = $result['cashierUrl'];
}catch(Exception $ex){
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
}
return ['type'=>'jump','url'=>$jump_url];
}
//异步回调
static public function notify(){
global $channel, $order;
$sign = $_POST['sign']; //签名
$data = $_POST['bizData']; //支付数据
require(PAY_ROOT."inc/SandpayClient.php");
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$verifyFlag = $client->verify($data, $sign);
if($verifyFlag){
$array = json_decode($data, true);
if($channel['appswitch']==1){
$params = [
'outReqTime' => date('YmdHis'),
'mid' => $channel['appid'],
'outOrderNo' => $array['outOrderNo'],
];
try{
$client->execute('/v4/sd-receipts/api/trans/trans.order.query', $params);
}catch(Exception $e){
//return ['type'=>'error','msg'=>'订单查询失败 '.$ex->getMessage()];
}
$log = "商户订单号:".TRADE_NO."\r\n异步通知:\r\n".$data."\r\n【订单查询接口】请求报文:\r\n".$client->request_body."\r\n【订单查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
if($array['orderStatus'] == 'success'){
$out_trade_no = $array['outOrderNo'];
$trade_no = $array['sandSerialNo'];
$money = $array['amount'];
$buyer = $array['payer']['payerAccNo'];
if($out_trade_no == TRADE_NO){
processNotify($order, $trade_no, $buyer);
}
return ['type'=>'html','data'=>'respCode=000000'];
}
}
return ['type'=>'html','data'=>'respCode=020002'];
}
//支付返回页面
static public function return(){
return ['type'=>'page','page'=>'return'];
}
//支付成功页面
static public function ok(){
return ['type'=>'page','page'=>'ok'];
}
//退款
static public function refund($order){
global $channel, $conf;
if(empty($order))exit();
require(PAY_ROOT."inc/SandpayClient.php");
$params = [
'marketProduct' => $channel['product'],
'outReqTime' => date('YmdHis'),
'mid' => $channel['appid'],
'outOrderNo' => $order['refund_no'],
'oriOutOrderNo' => $order['trade_no'],
'amount' => $order['refundmoney'],
'notifyUrl' => $conf['localurl'].'pay/refundnotify/'.TRADE_NO.'/',
];
try{
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$result = $client->execute('/v4/sd-receipts/api/trans/trans.order.refund', $params);
if($channel['appswitch']==1){
$log = "商户订单号:".$order['trade_no']."\r\n【退货接口】请求报文:\r\n".$client->request_body."\r\n【退货接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
return ['code'=>0, 'trade_no'=>$result['sandSerialNo'], 'refund_fee'=>$result['amount']];
}catch(Exception $ex){
return ['code'=>-1,'msg'=>$ex->getMessage()];
}
}
//退款回调
static public function refundnotify(){
global $channel, $order;
$sign = $_POST['sign']; //签名
$data = $_POST['bizData']; //支付数据
require(PAY_ROOT."inc/SandpayClient.php");
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$verifyFlag = $client->verify($data, $sign);
if($verifyFlag){
$array = json_decode($data, true);
if($array['orderStatus'] == 'success'){
$out_trade_no = $array['outOrderNo'];
$trade_no = $array['sandSerialNo'];
$money = $array['amount'];
return ['type'=>'html','data'=>'respCode=000000'];
}
}
return ['type'=>'html','data'=>'respCode=020002'];
}
//转账
static public function transfer($channel, $bizParam){
if(empty($channel) || empty($bizParam))exit();
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
$params = [
'mid' => $channel['appid'],
'outOrderNo' => $bizParam['out_biz_no'],
'amount' => $bizParam['money'],
'payeeInfo' => [
'accType' => 'cup',
'accNo' => $bizParam['payee_account'],
'accName' => $bizParam['payee_real_name'],
],
'payerInfo' => [
'sdaccSubId' => 'payment',
'remark' => $bizParam['transfer_desc'],
],
];
try{
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.order.create', $params);
$status = $result['paymentStatus'] == 'success' ? 1 : 0;
if($channel['appswitch']==1){
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【付款接口】请求报文:\r\n".$client->request_body."\r\n【付款接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
return ['code'=>0, 'status'=>$status, 'orderid'=>$result['sandSerialNo'], 'paydate'=>$result['finishedTime']];
}catch(Exception $ex){
return ['code'=>-1, 'msg'=>$ex->getMessage()];
}
}
//转账查询
static public function transfer_query($channel, $bizParam){
if(empty($channel) || empty($bizParam))exit();
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
$params = [
'mid' => $channel['appid'],
'outReqDate' => substr($bizParam['out_biz_no'], 0, 8),
'outOrderNo' => $bizParam['out_biz_no'],
];
try{
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.order.query', $params);
$status = $result['orderStatus'] == 'success' ? 1 : 0;
if($channel['appswitch']==1){
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【付款订单查询接口】请求报文:\r\n".$client->request_body."\r\n【付款订单查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
return ['code'=>0, 'status'=>$status];
}catch(Exception $ex){
return ['code'=>-1, 'msg'=>$ex->getMessage()];
}
}
//余额查询
static public function balance_query($channel, $bizParam){
if(empty($channel))exit();
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
$params = [
'mid' => $channel['appid'],
'sdaccSubId' => 'payment',
];
try{
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.balance.query', $params);
$account = $result['accountList'][0];
if($channel['appswitch']==1){
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【余额查询接口】请求报文:\r\n".$client->request_body."\r\n【余额查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
}
if(empty($account)) return ['code'=>-1, 'msg'=>'未查询到账户信息'];
return ['code'=>0, 'amount'=>$account['availableBal'], 'msg'=>'当前账户可用余额:'.$account['availableBal'].' 元,冻结金额:'.$account['frozenBal'].',在途余额:'.$account['transitBal']];
}catch(Exception $ex){
return ['code'=>-1, 'msg'=>$ex->getMessage()];
}
}
}