简要说明

本次主要说的是微信授权获取用户基本信息,还有就是日志反馈系统。利用微信自带的消息及时通知开发人员程序的报错。可达到快速收到问题解决问题,有助于建立一个可靠的风控体系


注意这里代码只有隐性授权 显性授权自己加上即可  原理一样

系统告警通知是使用的微信模板消息进行一个告警的通知


基本model   不同的框架稍微改改就可以直接用(如:缓存方法哪里需要自己根据自己框架特点进行改动)


<?php
/**
 *微信推送消息
 *by sqc 
 * 
 */
defined('InShopNC') or exit('Access Invalid!');

class wx_centerModel extends Model{

    protected $appid;  
    protected $secrect;  
    protected $accessToken;
    protected $redirect_uri;// 微信授权回调地址
    protected $code; // 微信授权code
    protected $o_access_token; // 微信授权access_token 注意:此access_token与基础支持的access_token不同
    protected $openid; // 微信授权openid
    protected $unionid; // 微信授权unionid
    //protected $GobackUrl; // 授权结束后返回用户界面的地址 弃
    protected $errors;

    function  __construct()  
    {  
        $this->appid = '********';  
        $this->secrect = '**************';
        
        // 测试 121科技号
        // $this->appid = '************';  
        // $this->secrect = '****************';
        // 第三个参数可用于多个公众号分别存储access_token使得与主号分离
        $this->accessToken = $this->getToken($this->appid, $this->secrect,'wx_access_token');  
    }

    /**
     * 设置参数
     *
     * @param mixed $key
     * @param mixed $value
     */
    public function set($key,$value){
        $this->$key = $value;
    }
    /**
     * 读取参数
     */
    public function get($key){
        return $this->$key;
    }

    /**
     * 隐性授权第一步:获取code
     * @return [type] [description]
     */
    public function silent_outh_step1(){
        var_dump($this->redirect_uri);
        if($this->redirect_uri){
            $url="https://open.weixin.qq.com/connect/oauth2/authorize?appid={$this->appid}&redirect_uri=".urlencode($this->redirect_uri)."&response_type=code&scope=snsapi_base&state=state#wechat_redirect";
            header('Location:'.$url);
        }else{
            return $this->error('微信授权回调地址参数丢失');
        }
        
    }

    /**
     * 隐性授权第2步:获取openid等信息  注意,在未关注公众号时,用户访问公众号的网页,也会产生一个用户和公众号唯一的OpenID
     * @return 对象
     */
    public function silent_outh_step2(){
        $this->code = $this->code?$this->code:$_GET['code'];
        if($this->code){
            //第二步:通过code换取网页授权access_token
            $url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid={$this->appid}&secret={$this->secrect}&code={$this->code}&grant_type=authorization_code";
            $res_json = $this->request_get($url);
            $res_arrone = json_decode($res_json, true); //转换为数组
            if($res_arrone['openid']){
                $this->openid = $res_arrone['openid'];
                $this->o_access_token = $res_arrone['access_token'];
                $this->unionid = $res_arrone['unionid'];// 这一步会有unionid 这是手册中没有的(2017.4.6)
            }else{
                $this->error($res_arrone);
            }
        }else{
            $this->error('没有获取到微信授权code');
        }
        return $this;
        
    }


    /**
     * 授权第3步:获取用户基本信息的  注意:这个是基于(这2个字省略了很多哦 可以到我博客上看)显性授权的
     * 需scope为 snsapi_userinfo
     * 如果网页授权作用域为snsapi_userinfo,则此时开发者可以通过access_token和openid拉取用户信息了
     * @return [type] [description]
     */
    public function outh_step3(){
        if($this->openid && $this->o_access_token){
            $get_userinfo_url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token={$this->o_access_token}&openid={$this->openid}&lang=zh_CN";
            $userinfo_json = $this->request_get($get_userinfo_url);
            $userinfo = json_decode($userinfo_json, true); //转换为数组
            //判断授权是否成功
           if(empty($userinfo) || isset($userinfo['errcode'])){
                if(!$this->unionid){
                    return $this->error($userinfo);
                }
                $userinfo['unionid']=$this->unionid;
                $userinfo['openid']=$this->openid;
            }
            return $userinfo;
        }else{
            return $this->error('获取用户信息参数不全');
        }
        
    }

    /**
     * 获取用户基本信息(UnionID机制)  基于openid
     */
    public function get_user_info(){
        if($this->accessToken && $this->openid){
            $url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token={$this->accessToken}&openid={$this->openid}&lang=zh_CN";
            $userinfo_json = $this->request_get($url);
            $userinfo = json_decode($userinfo_json, true); //转换为数组
            if(empty($userinfo) || isset($userinfo['errcode'])){
                if(!$this->unionid){
                    $userinfo['get_user_info'] =1;// 与outh_step3做区分
                    return $this->error($userinfo);
                }
                $userinfo['unionid']=$this->unionid;
                $userinfo['openid']=$this->openid;
            }
            return $userinfo;
        }else{
            return $this->error('accessToken或openid无法获取到');
        }
    }

    /** 
     * 发送post请求 
     * @param string $url 
     * @param string $param 
     * @return bool|mixed 
     */  
    protected function request_post($url = '', $param = '')  
    {  
        if (empty($url) || empty($param)) {  
            return false;  
        }  
        $postUrl = $url;  
        $curlPost = $param;  
        $ch = curl_init(); //初始化curl  
        curl_setopt($ch, CURLOPT_URL, $postUrl); //抓取指定网页  
        curl_setopt($ch, CURLOPT_HEADER, 0); //设置header  
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); //要求结果为字符串且输出到屏幕上  
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);  
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE);  
        curl_setopt($ch, CURLOPT_POST, 1); //post提交方式  
        curl_setopt($ch, CURLOPT_POSTFIELDS, $curlPost);  
        $data = curl_exec($ch); //运行curl  
        curl_close($ch);  
        return $data;  
    }  
    /** 
     * 发送get请求 
     * @param string $url 
     * @return bool|mixed 
     */  
    protected function request_get($url = '')  
    {  
        if (empty($url)) {  
            return false;  
        }  
        $ch = curl_init();  
        curl_setopt($ch, CURLOPT_URL, $url);  
        curl_setopt($ch, CURLOPT_TIMEOUT, 500);  
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);  
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);  
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);  
        $data = curl_exec($ch);  
        curl_close($ch);  
        return $data;  
    }  
    /** 
     * @param $appid 
     * @param $appsecret 
     * @return mixed 
     * 获取token 
     */  
    public function getToken($appid, $appsecret,$name)
    {   
        $key =  $name?$name:$appid;
        $access_token_info = rkcache($key);
        $access_token_info = unserialize($access_token_info);
        if ($access_token_info['time']>TIMESTAMP) {  
            $access_token = $access_token_info['token'];  
        } else {  
            $url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=" . $appid . "&secret=" . $appsecret;  
            $token = $this->request_get($url);  
            $token = json_decode(stripslashes($token));  
            $arr = json_decode(json_encode($token), true);
            $access_token = $arr['access_token'];
            $cache_arr['token'] = $arr['access_token'];
            $cache_arr['time'] = TIMESTAMP + 7200;
            wkcache($key, serialize($cache_arr));  
        }  
        return $access_token;  
    }

    /**
     * 制作一个日志
     * @param  string $name 错误标题
     * @param  mid  $msg  错误描述 可以字符串 或 数组 数组只支持到2维
     * @param  string $type 错误级别
     *
     */
    public function make_log($name,$msg,$type='notice',$openid=''){
        $template_id ='DjyGuUbytYLKvEyQUxRRffsHIWYuErqrijovg6LZVd8';
        
        if(is_array($msg)){
            $tmp_msg = '';
            foreach ($msg as $k => $v) {
                $tmp_msg .=$k.':'.$v;
                if(is_array($v)){
                    $tmp_msg .=' ( ';
                    foreach ($v as $k2 => $v2) {
                        $tmp_msg .=$k2.'->'.$v2;
                    }
                    $tmp_msg .=' ) ';
                }
                $tmp_msg .='|||';
            }
            $msg = $tmp_msg;
        }
        $data=array(  
           'first'=>array('value'=>urlencode("商城组错误通知"),'color'=>"#00CD00"),  
           'keyword1'=>array('value'=>urlencode($name),'color'=>'#EE5C42'),  
           'keyword2'=>array('value'=>urlencode(date('Y-m-d H:i:s',TIMESTAMP)),'color'=>'#030303'),  
           'keyword3'=>array('value'=>urlencode($type),'color'=>'#EE9A00'),  
           'keyword4'=>array('value'=>urlencode($this->getClientIP()),'color'=>'#030303'),
           'keyword5'=>array('value'=>urlencode($msg),'color'=>'#EE4000'),
           'remark'=>array('value'=>urlencode('121商城组'),'color'=>'#030303'),  
        );
        if(!$openid){
            $log_m = Model('wx_log');
            $openids = $log_m->select();
            foreach ($openids as $ky => $ve) {
                if($ve['openid']){
                    $this->doSend_logInfo($ve['openid'],$template_id,'',$data);
                }
            }
            return true;
        }
       $wxmsgs = $this->doSend_logInfo($openid,$template_id,'',$data); 
    }
    /** 
     * 发送自定义的模板消息 
     * @param $touser 
     * @param $template_id 
     * @param $url 
     * @param $data 
     * @param string $topcolor 
     * @return bool 
     */  
    public function doSend_logInfo($touser, $template_id, $url, $data, $topcolor = '#7B68EE')  
    {  
        $template = array(  
            'touser' => $touser,  
            'template_id' => $template_id,  
            'url' => $url,  
            'topcolor' => $topcolor,  
            'data' => $data  
        );  
        $json_template = json_encode($template);  
        $url = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=" . $this->accessToken;  
        $dataRes = $this->request_post($url, urldecode($json_template));
        $dataRes = json_decode(stripslashes($dataRes));  
        $dataRes = json_decode(json_encode($dataRes), true);
        if ($dataRes['errcode'] == 0) {  
            return true;  
        } else {  
            return false;  
        }  
    }

    private function getClientIP(){   
        if (getenv("HTTP_CLIENT_IP"))  
            $ip = getenv("HTTP_CLIENT_IP");  
        else if(getenv("HTTP_X_FORWARDED_FOR"))  
            $ip = getenv("HTTP_X_FORWARDED_FOR");  
        else if(getenv("REMOTE_ADDR"))  
            $ip = getenv("REMOTE_ADDR");  
        else $ip = "Unknow";  
        return $ip;  
    }
    /**
     * 返回用户错误信息
     * @return 返回值可以是josn或者是array
     */
    private function error($msg,$type = 'array'){
        switch ($type) {
            case 'array':
                $this->errors['error'][] = $msg;
                $this->make_log('微信授权系统',$msg,'notice');
                return $this->errors;
                break;
            
            default:
                # code...
                break;
        }
    }     
}



控制器总的授权调用



第一步



public function indexOp() {
        if(empty($_COOKIE['unionid'])){
            // 授权获取用户的unionid
            $this->check_back_url();
            // 开始调用授权
            $wx_model = Model('wx_center');
            $wx_model->set('redirect_uri',BASE_SITE_URL."/mobile/index.php?act=goods_wx&op=get_unionid");
            $wx_model->silent_outh_step1();
        }else{

            // 直接返回
            header('Location: '.$this->check_back_url());
            
        }
	}

第二步  $re = $wx_model->silent_outh_step2()->get_user_info(); 就是返回用户的基本信息


/**
     * 微信获取用户的信息
     */
    public function get_unionidOp() {
        $wx_model = Model('wx_center');
        $wx_model->set('code',$_GET['code']);
        $re = $wx_model->silent_outh_step2()->get_user_info();
        if(empty($re)||!$re['unionid']){
            echo '<meta charset="utf-8">';
            die('<center><h1>抱歉,您的网络超时,请稍后重试</h1></center>');
        }
        $_SESSION['my_userinfo'] = $re;
        setcookie("unionid",$re['unionid'],time()+365*3600*12,'/');
        header('Location: '.htmlspecialchars_decode(urldecode($_COOKIE['GobackUrl'])));die;
        
    }



告警消息的发送demo

$join_ceshi['商城告警测试'] = '成功成功';
 $join_ceshi['登陆错误'] = '10分钟多个注册导致系统故障';
$join_ceshi['订单告警'] = '10分钟内10000个下单';
$join_ceshi['下单错误'] = '参数错误';
$wx_model->make_log('这是一条测试数据',$join_ceshi,'notice',$re['openid']);



这个是我自己框架中model的调用 (逻辑处理model 类似于ThinkPHP中的大D方法)


$wx_model = Model('wx_center');