1)兩種access_token,網頁受權access_token和普通access_tokenphp
一、微信網頁受權是經過OAuth2.0機制實現的,在用戶受權給公衆號後,公衆號能夠獲取到一個網頁受權特有的接口調用憑證(網頁受權access_token),經過網頁受權access_token能夠進行受權後接口調用,如獲取用戶基本信息。html
二、其餘微信接口,須要經過基礎支持中的「獲取access_token」接口來獲取到的普通access_token調用。access_token是公衆號的全局惟一票據,access_token的有效期目前爲2個小時,需定時刷新,重複獲取將致使上次獲取的access_token失效。git
2)分別獲取access_tokengithub
一、網頁受權的:點擊查看網頁受權獲取用戶基本信息文檔,經過查看這個文檔,mongodb
能夠看到經過code換取網頁受權access_token,而這個code是經過微信的一個受權連接獲取到的,而後再根據文檔中的請求獲取到的,具體的連接地址和參數能夠參考文檔。數據庫
/** * 建立一個須要經過微信的OAuth2.0認證的服務url * @param $url 服務號須要認證訪問的url * @param $scope string snsapi_userinfo | snsapi_base * snsapi_userinfo 能夠用來獲取用戶信息 * snsapi_base 能夠用來獲取openid * @param string $state 自定義狀態值 * 此處約定爲from_weixin表明是從微信認證過來,通常無需輕易變化 * @return string 返回認證url地址 */ public function createAuthUrl($url, $scope = 'snsapi_base', $state = 'from_weixin') { $url = strval($url); $authUrl = 'https://open.weixin.qq.com/connect/oauth2/authorize'; /** * 此處有大坑,請不要打亂param的順序 * 不然微信認證界面會出現白屏 */ $param = array( 'appid' => $this->appId, 'redirect_uri' => urlencode($url), 'response_type' => 'code', 'scope' => $scope, 'state' => $state ); $seg = array(); foreach ($param as $k => $v) { $seg[] = "{$k}={$v}"; } return $authUrl . '?' . join('&', $seg) . '#wechat_redirect'; }
二、普通的:點擊查看獲取access token文檔,經過三個參數獲取到。api
這裏須要注意的是,獲取到的token,是有時效性的,2 個小時,因此我會保存在MongoDB中,先從數據庫中比對超時了沒有,沒有的話就直接從數據庫中獲取,減小沒必要要的請求。服務器
在與微信的交互中,會產生不少日誌信息,而且開發的時候常常須要分析這些日誌,這裏我將日誌都保存在了MongoDB中。MongoDB方便的地方是任何結構的數據都能放在一個document中,不像MySQL要定義好字段名,我常常調試的時候將各類結構放在一個document中。微信
在微信的入口頁面中,也就是前面提到的URL(服務器地址),會在這裏面作保存日誌的邏輯。邏輯包括關注的時候推送一條消息,二維碼掃描關注,點擊某個菜單產生事件,點擊菜單的超連接等。app
日誌結構以下:
一、代碼中包括簽名驗證邏輯
二、經過file_get_contents('php://input')來獲取請求數據,就是下面的getRawMsg方法
三、將推送日誌直接塞入到MongoDB中
四、將接收到的請求信息SimpleXMLElement對象,就是下面的parseMsg方法
五、handleEventMsg就是在處理各類不一樣狀況了
/** * 微信公衆號入口 */ public function actionPortal() { $weixin = new Weixin(); //簽名驗證邏輯 // if($weixin->checkSignature()){ // echo $_GET['echostr']; // } // exit; //讀取原始請求數據 $msg = $weixin->getRawMsg(); //推送日誌 $pushlog = new WeixinPushLog(); $pushlog->logWeixinPush($msg); $msgObj = $weixin->parseMsg($msg); if ($msgObj === false || !is_object($msgObj)) { exit; } switch ($msgObj->MsgType) { case 'event' : //接收事件消息 $this->handleEventMsg($msgObj); break; default : //todo break; } }
public function getRawMsg() { return file_get_contents('php://input'); } /** * 解析接收到的消息 * @param string $msg 消息體 * @return bool|SimpleXMLElement */ public function parseMsg($msg = '') { if (!$msg || empty($msg)) { return false; } $msgObj = simplexml_load_string($msg, 'SimpleXMLElement', LIBXML_NOCDATA); if ($msgObj === false || !($msgObj instanceof \SimpleXMLElement)) { return false; } return $msgObj; }
六、若是要推送消息,die這個方法得要加上
七、下面的代碼只列舉了兩種事件狀況,一種是訂閱、一種是點擊事件
八、createRawTuWenMsg是在拼接XML,點擊查看模板消息接口。
private function handleEventMsg($msgObj) { $weixin = new Weixin(); $openId = $msgObj->FromUserName; $fromUserName = $msgObj->ToUserName; //未關注,關注後推送 if ($msgObj->Event == 'subscribe') { $pushData['PicUrl'] = 'http://mmbiz.qpic.cn/'; $pushData['Title'] = '基因檢測,帶你一塊兒探索生命的奧妙 '; $pushData['Description'] = '爲何不一樣人在身高、體重、膚色和形狀上長得不同?可是每每又和本身的父母類似?'; $pushData['Url'] = 'http://mp.weixin.qq.com'; $msg = $weixin->createRawTuWenMsg($fromUserName, $openId, array($pushData)); die($msg); }elseif($msgObj->Event == 'CLICK'){ //die($msg); } }
public function createRawTuWenMsg($fromUserName, $toUserName, $items = array()) { if (!is_array($items)) { return ''; } $count = count($items); $its = ''; foreach ($items as $item) { $its .= <<<ITEMTPL <item> <Title><![CDATA[{$item['Title']}]]></Title> <Description><![CDATA[{$item['Description']}]]></Description> <PicUrl><![CDATA[{$item['PicUrl']}]]></PicUrl> <Url><![CDATA[{$item['Url']}]]></Url> </item> ITEMTPL; } $msg = <<<MSG <xml> <ToUserName><![CDATA[{$toUserName}]]></ToUserName> <FromUserName><![CDATA[{$fromUserName}]]></FromUserName> <CreateTime>12345678</CreateTime> <MsgType><![CDATA[news]]></MsgType> <ArticleCount>{$count}</ArticleCount> <Articles> {$its} </Articles> </xml> MSG; return $msg; }
demo下載: