<?php
namespace app\api\controller;
use think\Controller;
use think\Db;
class Payment extends Controller {
static public $treeList = array();
public function initialize(){
parent::initialize();
$this->config['appid'] ='小程序的APPID';//$this->config['appid'];
$this->config['mch_id'] = '商户号';
$this->config['api_key'] = '小程序密钥';
}
public function notify()
{
$postXml = $GLOBALS["HTTP_RAW_POST_DATA"];
//禁止引用外部xml实体
libxml_disable_entity_loader(true);
$xmlstring = simplexml_load_string($postXml, 'SimpleXMLElement', LIBXML_NOCDATA);
$attr = json_decode(json_encode($xmlstring), true);
file_put_contents('jiesi.txt', var_export($attr,true));
if ($attr['result_code'] == 'SUCCESS' && $attr['return_code'] == 'SUCCESS') {
$order_sn = $attr['out_trade_no'];
db("costlog")->where(['order'=>$order_sn])->update(['status'=>1]);
//db("zls")->insert()
if ($res) {
$return = ['return_code' => 'SUCCESS', 'return_msg' => 'OK'];
$xml = '<xml>';
foreach ($return as $k => $v) {
$xml .= '<' . $k . '><![CDATA[' . $v . ']]></' . $k . '>';
}
$xml .= '</xml>';
return '<xml><return_code>SUCCESS</return_code><return_msg>OK</return_msg></xml>';
}
}
// public function pay(){
// $rrs= $this->getWXPayInfo('ddd',time(),1,'oLYxK5LaSbJWxwqtzhLLY6KJDHuw');
// exit(json_encode($rrs));
// }
/*
$body 内容
$out_trade_no 订单号
$total_fee 价格
$open_id openid
*/
public function getWXPayInfo($body, $out_trade_no, $total_fee, $open_id){
$response = $this->getPrePayOrder($body, $out_trade_no, $total_fee, $open_id);
// return $response;
if ($response['return_code'] == 'FAIL') {
return ajaxreturn($response['return_msg'],2);
// return $response;
}
$x = $this->getOrder($response['prepay_id']);
$data1['wdata'] = $x;
// $data1['pay_money'] = $order['order_amount'];
return $data1;
}
public function getPrePayOrder($body, $out_trade_no, $total_fee, $open_id)
{
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
//$notify_url = $this->config["notify_url"];
$onoce_str = $this->getRandChar(32);
$data["appid"] = $this->config['appid'];
$data["body"] = $body;
$data["mch_id"] = $this->config['mch_id'];//商户ID
$data["nonce_str"] = $onoce_str;//32位随机数
$data["notify_url"] = 'https://'.$_SERVER['HTTP_HOST'].'/api/payment/notify';//回调地址
$data["out_trade_no"] = $out_trade_no;//订单编号
$data["spbill_create_ip"] = $this->get_client_ip();//IP地址
$data["total_fee"] = $total_fee;//订单价格
$data["trade_type"] = "JSAPI";
$data["openid"] = $open_id;
$s = $this->getSign($data, false);
$data["sign"] = $s;
// return $data;
$xml = $this->arrayToXml($data);
$response = $this->postXmlCurl($xml, $url);
// 将微信返回的结果xml转成数组
return $this->xmlstr_to_array($response);
}
// 执行第二次签名,才能返回给客户端使用
public function getOrder($prepayId)
{
$data["appId"] = $this->config['appid'];
$data["nonceStr"] = $this->config['mch_id'];
$data["package"] = "prepay_id=" . $prepayId;
$data['signType'] = "MD5";
$data["timeStamp"] = time();
$s = $this->getSign1($data, false);
$data["sign"] = $s;
return $data;
}
// function payment()
// {
// $postXml = $GLOBALS["HTTP_RAW_POST_DATA"];
// //禁止引用外部xml实体
// libxml_disable_entity_loader(true);
// $xmlstring = simplexml_load_string($postXml, 'SimpleXMLElement', LIBXML_NOCDATA);
// $attr = json_decode(json_encode($xmlstring), true);
// if ($attr['result_code'] == 'SUCCESS' && $attr['return_code'] == 'SUCCESS') {
//
// //回调
//
// if ($res) {
// $return = ['return_code' => 'SUCCESS', 'return_msg' => 'OK'];
// $xml = '<xml>';
// foreach ($return as $k => $v) {
// $xml .= '<' . $k . '><![CDATA[' . $v . ']]></' . $k . '>';
// }
// $xml .= '</xml>';
// return '<xml><return_code>SUCCESS</return_code><return_msg>OK</return_msg></xml>';
// }
// }
//
// }
//
// 获取指定长度的随机字符串
function getRandChar($length)
{
$str = null;
$strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
$max = strlen($strPol) - 1;
for ($i = 0; $i < $length; $i++) {
$str .= $strPol[rand(0, $max)]; // rand($min,$max)生成介于min和max两个数之间的一个随机整数
}
return $str;
}
/*
* 生成签名
*/
function getSign($Obj)
{
foreach ($Obj as $k => $v) {
$Parameters[strtolower($k)] = $v;
}
// 签名步骤一:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
// echo "【string】 =".$String."</br>";
// 签名步骤二:在string后加入KEY
$String = $String . "&key=" . $this->config['api_key'];
// echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
// 签名步骤三:MD5加密
$result_ = strtoupper(md5($String));
return $result_;
}
/*
* 生成签名
*/
function getSign1($Obj)
{
foreach ($Obj as $k => $v) {
$Parameters[strtolower($k)] = $v;
}
// 签名步骤一:按字典序排序参数
ksort($Parameters);
//$String = $this->formatBizQueryParaMap($Parameters, false);
$String = "appId=" . $Obj['appId'] . "&nonceStr=" . $Obj['nonceStr'] . "&package=" . $Obj['package'] . "&signType=MD5&timeStamp=" . $Obj['timeStamp'];
// echo "【string】 =".$String."</br>";
// 签名步骤二:在string后加入KEY
$String = $String . "&key=" . $this->config['api_key'];
// echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
// 签名步骤三:MD5加密
//echo $String;
$result_ = strtoupper(md5($String));
return $result_;
}
/*
* 获取当前服务器的IP
*/
function get_client_ip()
{
if ($_SERVER['REMOTE_ADDR']) {
$cip = $_SERVER['REMOTE_ADDR'];
} elseif (getenv("REMOTE_ADDR")) {
$cip = getenv("REMOTE_ADDR");
} elseif (getenv("HTTP_CLIENT_IP")) {
$cip = getenv("HTTP_CLIENT_IP");
} else {
$cip = "unknown";
}
return $cip;
}
// 数组转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;
}
// post https请求,CURLOPT_POSTFIELDS xml格式
function postXmlCurl($xml, $url, $second = 30)
{
// 初始化curl
$ch = curl_init();
// 超时时间
curl_setopt($ch, CURLOPT_TIMEOUT, $second);
// 这里设置代理,如果有的话
// curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
// curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
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;
}
}
/**
* xml转成数组
*/
function xmlstr_to_array($xmlstr)
{
//$doc = new \DOMDocument();
//$doc->loadXML($xmlstr);
//return $this->domnode_to_array($doc->documentElement);
//禁止引用外部xml实体
libxml_disable_entity_loader(true);
$xmlstring = simplexml_load_string($xmlstr, 'SimpleXMLElement', LIBXML_NOCDATA);
$val = json_decode(json_encode($xmlstring), true);
return $val;
}
// 将数组转成uri字符串
function formatBizQueryParaMap($paraMap, $urlencode)
{
$buff = "";
ksort($paraMap);
foreach ($paraMap as $k => $v) {
if ($urlencode) {
$v = urlencode($v);
}
$buff .= strtolower($k) . "=" . $v . "&";
}
$reqPar;
if (strlen($buff) > 0) {
$reqPar = substr($buff, 0, strlen($buff) - 1);
}
return $reqPar;
}
function domnode_to_array($node)
{
$output = array();
switch ($node->nodeType) {
case XML_CDATA_SECTION_NODE:
case XML_TEXT_NODE:
$output = trim($node->textContent);
break;
case XML_ELEMENT_NODE:
for ($i = 0, $m = $node->childNodes->length; $i < $m; $i++) {
$child = $node->childNodes->item($i);
$v = $this->domnode_to_array($child);
if (isset($child->tagName)) {
$t = $child->tagName;
if (!isset($output[$t])) {
$output[$t] = array();
}
$output[$t][] = $v;
} elseif ($v) {
$output = (string)$v;
}
}
if (is_array($output)) {
if ($node->attributes->length) {
$a = array();
foreach ($node->attributes as $attrName => $attrNode) {
$a[$attrName] = (string)$attrNode->value;
}
$output['@attributes'] = $a;
}
foreach ($output as $t => $v) {
if (is_array($v) && count($v) == 1 && $t != '@attributes') {
$output[$t] = $v[0];
}
}
}
break;
}
return $output;
}
}
?>