git wechat.class.php,wechat-php-sdk/qywechat.class.php at master · gitye/wechat-php-sdk · GitHub

/**

*微信公众平台企业号PHP-SDK, 官方API类库

* @author binsee

* @link https://github.com/binsee/wechat-php-sdk

* @version 1.0

* usage:

* $options = array(

*'token'=>'tokenaccesskey', //填写应用接口的Token

*'encodingaeskey'=>'encodingaeskey', //填写加密用的EncodingAESKey

*'appid'=>'wxdk1234567890', //填写高级调用功能的app id

*'appsecret'=>'xxxxxxxxxxxxxxxxxxx', //填写高级调用功能的密钥

*'agentid'=>'1', //应用的id

*'debug'=>false, //调试开关

*'_logcallback'=>'logg', //调试输出方法,需要有一个string类型的参数

*);

*

*/

class Wechat

{

const MSGTYPE_TEXT = 'text';

const MSGTYPE_IMAGE = 'image';

const MSGTYPE_LOCATION = 'location';

const MSGTYPE_LINK = 'link'; //暂不支持

const MSGTYPE_EVENT = 'event';

const MSGTYPE_MUSIC = 'music'; //暂不支持

const MSGTYPE_NEWS = 'news';

const MSGTYPE_VOICE = 'voice';

const MSGTYPE_VIDEO = 'video';

const API_URL_PREFIX = 'https://qyapi.weixin.qq.com/cgi-bin';

const USER_CREATE_URL = '/user/create?';

const USER_UPDATE_URL = '/user/update?';

const USER_DELETE_URL = '/user/delete?';

const USER_GET_URL = '/user/get?';

const USER_LIST_URL = '/user/simplelist?';

const USER_GETINFO_URL = '/user/getuserinfo?';

const DEPARTMENT_CREATE_URL = '/department/create?';

const DEPARTMENT_UPDATE_URL = '/department/update?';

const DEPARTMENT_DELETE_URL = '/department/delete?';

const DEPARTMENT_MOVE_URL = '/department/move?';

const DEPARTMENT_LIST_URL = '/department/list?';

const TAG_CREATE_URL = '/tag/create?';

const TAG_UPDATE_URL = '/tag/update?';

const TAG_DELETE_URL = '/tag/delete?';

const TAG_GET_URL = '/tag/get?';

const TAG_ADDUSER_URL = '/tag/addtagusers?';

const TAG_DELUSER_URL = '/tag/deltagusers?';

const TAG_LIST_URL = '/tag/list?';

const MEDIA_UPLOAD_URL = '/media/upload?';

const MEDIA_GET_URL = '/media/get?';

const AUTHSUCC_URL = '/user/authsucc?';

const MASS_SEND_URL = '/message/send?';

const MENU_CREATE_URL = '/menu/create?';

const MENU_GET_URL = '/menu/get?';

const MENU_DELETE_URL = '/menu/delete?';

const TOKEN_GET_URL = '/gettoken?';

const OAUTH_PREFIX = 'https://open.weixin.qq.com/connect/oauth2';

const OAUTH_AUTHORIZE_URL = '/authorize?';

private $token;

private $encodingAesKey;

private $appid; //也就是企业号的CorpID

private $appsecret;

private $access_token;

private $agentid; //应用id AgentID

private $postxml;

private $agentidxml; //接收的应用id AgentID

private $_msg;

private $_receive;

private $_sendmsg; //主动发送消息的内容

private $_text_filter = true;

public $debug = false;

public $errCode = 40001;

public $errMsg = "no access";

private $_logcallback;

public function __construct($options)

{

$this->token = isset($options['token'])?$options['token']:'';

$this->encodingAesKey = isset($options['encodingaeskey'])?$options['encodingaeskey']:'';

$this->appid = isset($options['appid'])?$options['appid']:'';

$this->appsecret = isset($options['appsecret'])?$options['appsecret']:'';

$this->agentid = isset($options['agentid'])?$options['agentid']:'';

$this->debug = isset($options['debug'])?$options['debug']:false;

$this->_logcallback = isset($options['logcallback'])?$options['logcallback']:false;

}

private function log($log){

if ($this->debug && function_exists($this->_logcallback)) {

if (is_array($log)) $log = print_r($log,true);

return call_user_func($this->_logcallback,$log);

}

}

/**

* 数据XML编码

* @param mixed $data 数据

* @return string

*/

public static function data_to_xml($data) {

$xml = '';

foreach ($data as $key => $val) {

is_numeric($key) && $key = "item id=\"$key\"";

$xml .= "";

$xml .= ( is_array($val) || is_object($val)) ? self::data_to_xml($val) : self::xmlSafeStr($val);

list($key, ) = explode(' ', $key);

$xml .= "$key>";

}

return $xml;

}

public static function xmlSafeStr($str)

{

return '';

}

/**

* XML编码

* @param mixed $data 数据

* @param string $root 根节点名

* @param string $item 数字索引的子节点名

* @param string $attr 根节点属性

* @param string $id 数字索引子节点key转换的属性名

* @param string $encoding 数据编码

* @return string

*/

public function xml_encode($data, $root='xml', $item='item', $attr='', $id='id', $encoding='utf-8') {

if(is_array($attr)){

$_attr = array();

foreach ($attr as $key => $value) {

$_attr[] = "{$key}=\"{$value}\"";

}

$attr = implode(' ', $_attr);

}

$attr = trim($attr);

$attr = empty($attr) ? '' : " {$attr}";

$xml = "";

$xml .= self::data_to_xml($data, $item, $id);

$xml .= "{$root}>";

return $xml;

}

/**

* 微信api不支持中文转义的json结构

* @param array $arr

*/

static function json_encode($arr) {

$parts = array ();

$is_list = false;

//Find out if the given array is a numerical array

$keys = array_keys ( $arr );

$max_length = count ( $arr ) - 1;

if (($keys [0] === 0) && ($keys [$max_length] === $max_length )) { //See if the first key is 0 and last key is length - 1

$is_list = true;

for($i = 0; $i < count ( $keys ); $i ++) { //See if each key correspondes to its position

if ($i != $keys [$i]) { //A key fails at position check.

$is_list = false; //It is an associative array.

break;

}

}

}

foreach ( $arr as $key => $value ) {

if (is_array ( $value )) { //Custom handling for arrays

if ($is_list)

$parts [] = self::json_encode ( $value ); /* :RECURSION: */

else

$parts [] = '"' . $key . '":' . self::json_encode ( $value ); /* :RECURSION: */

} else {

$str = '';

if (! $is_list)

$str = '"' . $key . '":';

//Custom handling for multiple data types

if (!is_string ( $value ) && is_numeric ( $value ) && $value<2000000000)

$str .= $value; //Numbers

elseif ($value === false)

$str .= 'false'; //The booleans

elseif ($value === true)

$str .= 'true';

else

$str .= '"' . addslashes ( $value ) . '"'; //All other things

// :TODO: Is there any more datatype we should be in the lookout for? (Object?)

$parts [] = $str;

}

}

$json = implode ( ',', $parts );

if ($is_list)

return '[' . $json . ']'; //Return numerical JSON

return '{' . $json . '}'; //Return associative JSON

}

/**

* 过滤文字回复\r\n换行符

* @param string $text

* @return string|mixed

*/

private function _auto_text_filter($text) {

if (!$this->_text_filter) return $text;

return str_replace("\r\n", "\n", $text);

}

/**

* GET 请求

* @param string $url

*/

private function http_get($url){

$oCurl = curl_init();

if(stripos($url,"https://")!==FALSE){

curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE);

curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, FALSE);

curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); //CURL_SSLVERSION_TLSv1

}

curl_setopt($oCurl, CURLOPT_URL, $url);

curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );

$sContent = curl_exec($oCurl);

$aStatus = curl_getinfo($oCurl);

curl_close($oCurl);

if(intval($aStatus["http_code"])==200){

return $sContent;

}else{

return false;

}

}

/**

* POST 请求

* @param string $url

* @param array $param

* @param boolean $post_file 是否文件上传

* @return string content

*/

private function http_post($url,$param,$post_file=false){

$oCurl = curl_init();

if(stripos($url,"https://")!==FALSE){

curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE);

curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, false);

curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); //CURL_SSLVERSION_TLSv1

}

if (is_string($param) || $post_file) {

$strPOST = $param;

} else {

$aPOST = array();

foreach($param as $key=>$val){

$aPOST[] = $key."=".urlencode($val);

}

$strPOST = join("&", $aPOST);

}

curl_setopt($oCurl, CURLOPT_URL, $url);

curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );

curl_setopt($oCurl, CURLOPT_POST,true);

curl_setopt($oCurl, CURLOPT_POSTFIELDS,$strPOST);

$sContent = curl_exec($oCurl);

$aStatus = curl_getinfo($oCurl);

curl_close($oCurl);

if(intval($aStatus["http_code"])==200){

return $sContent;

}else{

return false;

}

}

/**

* For weixin server validation

*/

private function checkSignature($str)

{

$signature = isset($_GET["msg_signature"])?$_GET["msg_signature"]:'';

$timestamp = isset($_GET["timestamp"])?$_GET["timestamp"]:'';

$nonce = isset($_GET["nonce"])?$_GET["nonce"]:'';

$tmpArr = array($str,$this->token, $timestamp, $nonce);//比普通公众平台多了一个加密的密文

sort($tmpArr, SORT_STRING);

$tmpStr = implode($tmpArr);

$shaStr = sha1($tmpStr);

if( $shaStr == $signature ){

return true;

}else{

return false;

}

}

/**

* 微信验证,包括post来的xml解密

* @param bool $return 是否返回

*/

public function valid($return=false)

{

$encryptStr="";

if ($_SERVER['REQUEST_METHOD'] == "POST") {

$postStr = file_get_contents("php://input");

$array = (array)simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);

$this->log($postStr);

if (isset($array['Encrypt'])) {

$encryptStr = $array['Encrypt'];

$this->agentidxml = isset($array['AgentID']) ? $array['AgentID']: '';

}

} else {

$encryptStr = isset($_GET["echostr"]) ? $_GET["echostr"]: '';

}

if ($encryptStr) {

$ret=$this->checkSignature($encryptStr);

}

if (!isset($ret) || !$ret) {

if (!$return) {

die('no access');

} else {

return false;

}

}

$pc = new Prpcrypt($this->encodingAesKey);

$array = $pc->decrypt($encryptStr,$this->appid);

if (!isset($array[0]) || ($array[0] != 0)) {

if (!$return) {

die('解密失败!');

} else {

return false;

}

}

if ($_SERVER['REQUEST_METHOD'] == "POST") {

$this->postxml = $array[1];

//$this->log($array[1]);

return ($this->postxml!="");

} else {

$echoStr = $array[1];

if ($return) {

return $echoStr;

} else {

die($echoStr);

}

}

return false;

}

/**

* 获取微信服务器发来的信息

*/

public function getRev()

{

if ($this->_receive) return $this;

$postStr = $this->postxml;

$this->log($postStr);

if (!empty($postStr)) {

$this->_receive = (array)simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);

if (!isset($this->_receive['AgentID'])) {

$this->_receive['AgentID']=$this->agentidxml; //当前接收消息的应用id

}

}

return $this;

}

/**

* 获取微信服务器发来的信息

*/

public function getRevData()

{

return $this->_receive;

}

/**

* 获取微信服务器发来的原始加密信息

*/

public function getRevPostXml()

{

return $this->postxml;

}

/**

* 获取消息发送者

*/

public function getRevFrom() {

if (isset($this->_receive['FromUserName']))

return $this->_receive['FromUserName'];

else

return false;

}

/**

* 获取消息接受者

*/

public function getRevTo() {

if (isset($this->_receive['ToUserName']))

return $this->_receive['ToUserName'];

else

return false;

}

/**

* 获取接收消息的应用id

*/

public function getRevAgentID() {

if (isset($this->_receive['AgentID']))

return $this->_receive['AgentID'];

else

return false;

}

/**

* 获取接收消息的类型

*/

public function getRevType() {

if (isset($this->_receive['MsgType']))

return $this->_receive['MsgType'];

else

return false;

}

/**

* 获取消息ID

*/

public function getRevID() {

if (isset($this->_receive['MsgId']))

return $this->_receive['MsgId'];

else

return false;

}

/**

* 获取消息发送时间

*/

public function getRevCtime() {

if (isset($this->_receive['CreateTime']))

return $this->_receive['CreateTime'];

else

return false;

}

/**

* 获取接收消息内容正文

*/

public function getRevContent(){

if (isset($this->_receive['Content']))

return $this->_receive['Content'];

else

return false;

}

/**

* 获取接收消息图片

*/

public function getRevPic(){

if (isset($this->_receive['PicUrl']))

return array(

'mediaid'=>$this->_receive['MediaId'],

'picurl'=>(string)$this->_receive['PicUrl'], //防止picurl为空导致解析出错

);

else

return false;

}

/**

* 获取接收地理位置

*/

public function getRevGeo(){

if (isset($this->_receive['Location_X'])){

return array(

'x'=>$this->_receive['Location_X'],

'y'=>$this->_receive['Location_Y'],

'scale'=>(string)$this->_receive['Scale'],

'label'=>(string)$this->_receive['Label']

);

} else

return false;

}

/**

* 获取上报地理位置事件

*/

public function getRevEventGeo(){

if (isset($this->_receive['Latitude'])){

return array(

'x'=>$this->_receive['Latitude'],

'y'=>$this->_receive['Longitude'],

'precision'=>$this->_receive['Precision'],

);

} else

return false;

}

/**

* 获取接收事件推送

*/

public function getRevEvent(){

if (isset($this->_receive['Event'])){

$array['event'] = $this->_receive['Event'];

}

if (isset($this->_receive['EventKey'])){

$array['key'] = $this->_receive['EventKey'];

}

if (isset($array) && count($array) > 0) {

return $array;

} else {

return false;

}

}

/**

* 获取自定义菜单的扫码推事件信息

*

* 事件类型为以下两种时则调用此方法有效

* Event 事件类型,scancode_push

* Event 事件类型,scancode_waitmsg

*

* @return: array | false

* array (

* 'ScanType'=>'qrcode',

* 'ScanResult'=>'123123'

* )

*/

public function getRevScanInfo(){

if (isset($this->_receive['ScanCodeInfo'])){

if (!is_array($this->_receive['SendPicsInfo'])) {

$array=(array)$this->_receive['ScanCodeInfo'];

$this->_receive['ScanCodeInfo']=$array;

}else {

$array=$this->_receive['ScanCodeInfo'];

}

}

if (isset($array) && count($array) > 0) {

return $array;

} else {

return false;

}

}

/**

* 获取自定义菜单的图片发送事件信息

*

* 事件类型为以下三种时则调用此方法有效

* Event 事件类型,pic_sysphoto 弹出系统拍照发图的事件推送

* Event 事件类型,pic_photo_or_album 弹出拍照或者相册发图的事件推送

* Event 事件类型,pic_weixin 弹出微信相册发图器的事件推送

*

* @return: array | false

* array (

* 'Count' => '2',

* 'PicList' =>array (

* 'item' =>array (

* 0 =>array ('PicMd5Sum' => 'aaae42617cf2a14342d96005af53624c'),

* 1 =>array ('PicMd5Sum' => '149bd39e296860a2adc2f1bb81616ff8'),

* ),

* ),

* )

*

*/

public function getRevSendPicsInfo(){

if (isset($this->_receive['SendPicsInfo'])){

if (!is_array($this->_receive['SendPicsInfo'])) {

$array=(array)$this->_receive['SendPicsInfo'];

if (isset($array['PicList'])){

$array['PicList']=(array)$array['PicList'];

$item=$array['PicList']['item'];

$array['PicList']['item']=array();

foreach ( $item as $key => $value ){

$array['PicList']['item'][$key]=(array)$value;

}

}

$this->_receive['SendPicsInfo']=$array;

} else {

$array=$this->_receive['SendPicsInfo'];

}

}

if (isset($array) && count($array) > 0) {

return $array;

} else {

return false;

}

}

/**

* 获取自定义菜单的地理位置选择器事件推送

*

* 事件类型为以下时则可以调用此方法有效

* Event 事件类型,location_select 弹出系统拍照发图的事件推送

*

* @return: array | false

* array (

* 'Location_X' => '33.731655000061',

* 'Location_Y' => '113.29955200008047',

* 'Scale' => '16',

* 'Label' => '某某市某某区某某路',

* 'Poiname' => '',

* )

*

*/

public function getRevSendGeoInfo(){

if (isset($this->_receive['SendLocationInfo'])){

if (!is_array($this->_receive['SendLocationInfo'])) {

$array=(array)$this->_receive['SendLocationInfo'];

if (empty($array['Poiname'])) {

$array['Poiname']="";

}

if (empty($array['Label'])) {

$array['Label']="";

}

$this->_receive['SendLocationInfo']=$array;

} else {

$array=$this->_receive['SendLocationInfo'];

}

}

if (isset($array) && count($array) > 0) {

return $array;

} else {

return false;

}

}

/**

* 获取接收语音推送

*/

public function getRevVoice(){

if (isset($this->_receive['MediaId'])){

return array(

'mediaid'=>$this->_receive['MediaId'],

'format'=>$this->_receive['Format'],

);

} else

return false;

}

/**

* 获取接收视频推送

*/

public function getRevVideo(){

if (isset($this->_receive['MediaId'])){

return array(

'mediaid'=>$this->_receive['MediaId'],

'thumbmediaid'=>$this->_receive['ThumbMediaId']

);

} else

return false;

}

/**

* 设置回复消息

* Example: $obj->text('hello')->reply();

* @param string $text

*/

public function text($text='')

{

$msg = array(

'ToUserName' => $this->getRevFrom(),

'FromUserName'=>$this->getRevTo(),

'MsgType'=>self::MSGTYPE_TEXT,

'Content'=>$this->_auto_text_filter($text),

'CreateTime'=>time(),

);

$this->Message($msg);

return $this;

}

/**

* 设置回复消息

* Example: $obj->image('media_id')->reply();

* @param string $mediaid

*/

public function image($mediaid='')

{

$msg = array(

'ToUserName' => $this->getRevFrom(),

'FromUserName'=>$this->getRevTo(),

'MsgType'=>self::MSGTYPE_IMAGE,

'Image'=>array('MediaId'=>$mediaid),

'CreateTime'=>time(),

);

$this->Message($msg);

return $this;

}

/**

* 设置回复消息

* Example: $obj->voice('media_id')->reply();

* @param string $mediaid

*/

public function voice($mediaid='')

{

$msg = array(

'ToUserName' => $this->getRevFrom(),

'FromUserName'=>$this->getRevTo(),

'MsgType'=>self::MSGTYPE_IMAGE,

'Voice'=>array('MediaId'=>$mediaid),

'CreateTime'=>time(),

);

$this->Message($msg);

return $this;

}

/**

* 设置回复消息

* Example: $obj->video('media_id','title','description')->reply();

* @param string $mediaid

*/

public function video($mediaid='',$title='',$description='')

{

$msg = array(

'ToUserName' => $this->getRevFrom(),

'FromUserName'=>$this->getRevTo(),

'MsgType'=>self::MSGTYPE_IMAGE,

'Video'=>array(

'MediaId'=>$mediaid,

'Title'=>$title,

'Description'=>$description

),

'CreateTime'=>time(),

);

$this->Message($msg);

return $this;

}

/**

* 设置回复图文

* @param array $newsData

* 数组结构:

* array(

* "0"=>array(

* 'Title'=>'msg title',

* 'Description'=>'summary text',

* 'PicUrl'=>'http://www.domain.com/1.jpg',

* 'Url'=>'http://www.domain.com/1.html'

* ),

* "1"=>....

* )

*/

public function news($newsData=array())

{

$count = count($newsData);

$msg = array(

'ToUserName' => $this->getRevFrom(),

'FromUserName'=>$this->getRevTo(),

'MsgType'=>self::MSGTYPE_NEWS,

'CreateTime'=>time(),

'ArticleCount'=>$count,

'Articles'=>$newsData,

);

$this->Message($msg);

return $this;

}

/**

* 设置发送消息

* @param array $msg 消息数组

* @param bool $append 是否在原消息数组追加

*/

public function Message($msg = '',$append = false){

if (is_null($msg)) {

$this->_msg =array();

}elseif (is_array($msg)) {

if ($append)

$this->_msg = array_merge($this->_msg,$msg);

else

$this->_msg = $msg;

return $this->_msg;

} else {

return $this->_msg;

}

}

/**

*

* 回复微信服务器, 此函数支持链式操作

* Example: $this->text('msg tips')->reply();

* @param string $msg 要发送的信息, 默认取$this->_msg

* @param bool $return 是否返回信息而不抛出到浏览器 默认:否

*/

public function reply($msg=array(),$return = false)

{

if (empty($msg))

$msg = $this->_msg;

$xmldata= $this->xml_encode($msg);

$this->log($xmldata);

$pc = new Prpcrypt($this->encodingAesKey);

$array = $pc->encrypt($xmldata, $this->appid);

$ret = $array[0];

if ($ret != 0) {

$this->log('encrypt err!');

return false;

}

$timestamp = time();

$nonce = rand(77,999)*rand(605,888)*rand(11,99);

$encrypt = $array[1];

$tmpArr = array($this->token, $timestamp, $nonce,$encrypt);//比普通公众平台多了一个加密的密文

sort($tmpArr, SORT_STRING);

$signature = implode($tmpArr);

$signature = sha1($signature);

$smsg = $this->generate($encrypt, $signature, $timestamp, $nonce);

$this->log($smsg);

if ($return)

return $smsg;

elseif ($smsg){

echo $smsg;

return true;

}else

return false;

}

private function generate($encrypt, $signature, $timestamp, $nonce)

{

//格式化加密信息

$format = "

%s

";

return sprintf($format, $encrypt, $signature, $timestamp, $nonce);

}

/**

* 通用auth验证方法

* @param string $appid

* @param string $appsecret

* @param string $token 手动指定access_token,非必要情况不建议用

*/

public function checkAuth($appid='',$appsecret='',$token=''){

if (!$appid || !$appsecret) {

$appid = $this->appid;

$appsecret = $this->appsecret;

}

if ($token) { //手动指定token,优先使用

$this->access_token=$token;

return $this->access_token;

}

//TODO: get the cache access_token

$result = $this->http_get(self::API_URL_PREFIX.self::TOKEN_GET_URL.'corpid='.$appid.'&corpsecret='.$appsecret);

if ($result)

{

$json = json_decode($result,true);

if (!$json || isset($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

$this->access_token = $json['access_token'];

$expire = $json['expires_in'] ? intval($json['expires_in'])-100 : 3600;

//TODO: cache access_token

return $this->access_token;

}

return false;

}

/**

* 删除验证数据

* @param string $appid

*/

public function resetAuth($appid=''){

if (!$appid) $appid = $this->appid;

$this->access_token = '';

//TODO: remove cache

return true;

}

/**

* 创建菜单

* @param array $data 菜单数组数据

* example:

* array (

* 'button' => array (

* 0 => array (

* 'name' => '扫码',

* 'sub_button' => array (

* 0 => array (

* 'type' => 'scancode_waitmsg',

* 'name' => '扫码带提示',

* 'key' => 'rselfmenu_0_0',

* ),

* 1 => array (

* 'type' => 'scancode_push',

* 'name' => '扫码推事件',

* 'key' => 'rselfmenu_0_1',

* ),

* ),

* ),

* 1 => array (

* 'name' => '发图',

* 'sub_button' => array (

* 0 => array (

* 'type' => 'pic_sysphoto',

* 'name' => '系统拍照发图',

* 'key' => 'rselfmenu_1_0',

* ),

* 1 => array (

* 'type' => 'pic_photo_or_album',

* 'name' => '拍照或者相册发图',

* 'key' => 'rselfmenu_1_1',

* )

* ),

* ),

* 2 => array (

* 'type' => 'location_select',

* 'name' => '发送位置',

* 'key' => 'rselfmenu_2_0'

* ),

* ),

* )

* type可以选择为以下几种,会收到相应类型的事件推送。请注意,3到8的所有事件,仅支持微信iPhone5.4.1以上版本,

* 和Android5.4以上版本的微信用户,旧版本微信用户点击后将没有回应,开发者也不能正常接收到事件推送。

* 1、click:点击推事件

* 2、view:跳转URL

* 3、scancode_push:扫码推事件

* 4、scancode_waitmsg:扫码推事件且弹出“消息接收中”提示框

* 5、pic_sysphoto:弹出系统拍照发图

* 6、pic_photo_or_album:弹出拍照或者相册发图

* 7、pic_weixin:弹出微信相册发图器

* 8、location_select:弹出地理位置选择器

*/

public function createMenu($data,$agentid=''){

if ($agentid=='') {

$agentid=$this->agentid;

}

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::MENU_CREATE_URL.'access_token='.$this->access_token.'&agentid='.$agentid,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return true;

}

return false;

}

/**

* 获取菜单

* @return array('menu'=>array(....s))

*/

public function getMenu($agentid=''){

if ($agentid=='') {

$agentid=$this->agentid;

}

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::MENU_GET_URL.'access_token='.$this->access_token.'&agentid='.$agentid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || isset($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 删除菜单

* @return boolean

*/

public function deleteMenu($agentid=''){

if ($agentid=='') {

$agentid=$this->agentid;

}

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::MENU_DELETE_URL.'access_token='.$this->access_token.'&agentid='.$agentid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return true;

}

return false;

}

/**

* 上传多媒体文件 (只有三天的有效期,过期自动被删除)

* 注意:数组的键值任意,但文件名前必须加@,使用单引号以避免本地路径斜杠被转义

* @param array $data {"media":'@Path\filename.jpg'}

* @param type 媒体文件类型:图片(image)、语音(voice)、视频(video),普通文件(file)

* @return boolean|array

* {

* "type": "image",

* "media_id": "0000001",

* "created_at": "1380000000"

* }

*/

public function uploadMedia($data, $type){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::MEDIA_UPLOAD.'access_token='.$this->access_token.'&type='.$type,$data,true);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 根据媒体文件ID获取媒体文件

* @param string $media_id 媒体文件id

* @return raw data

*/

public function getMedia($media_id){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::MEDIA_GET_URL.'access_token='.$this->access_token.'&media_id='.$media_id);

if ($result)

{

$json = json_decode($result,true);

if (isset($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $result;

}

return false;

}

/**

* 创建部门

* @param array $data 结构体为:

* array (

* "name" => "邮箱产品组", //部门名称

* "parentid" => "1" //父部门id

* "order" => "1", //(非必须)在父部门中的次序。从1开始,数字越大排序越靠后

* )

* @return boolean|array

* 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "created", //对返回码的文本描述内容

* "id": 2 //创建的部门id。

* }

*/

public function createDepartment($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::DEPARTMENT_CREATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 更新部门

* @param array $data 结构体为:

* array(

* "id" => "1" //(必须)部门id

* "name" => "邮箱产品组", //(非必须)部门名称

* "parentid" => "1", //(非必须)父亲部门id。根部门id为1

* "order" => "1", //(非必须)在父部门中的次序。从1开始,数字越大排序越靠后

* )

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "updated" //对返回码的文本描述内容

* }

*/

public function updateDepartment($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::DEPARTMENT_UPDATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 删除部门

* @param $id

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "deleted" //对返回码的文本描述内容

* }

*/

public function deleteDepartment($id){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::DEPARTMENT_DELETE_URL.'access_token='.$this->access_token.'&id='.$id);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 移动部门

* @param $data

* array(

* "department_id" => "5",//所要移动的部门

* "to_parentid" => "2",//想移动到的父部门节点,根部门为1

* "to_position" => "1"//(非必须)想移动到的父部门下的位置,1表示最上方,往后位置为2,3,4,以此类推,默认为1

* )

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "ok" //对返回码的文本描述内容

* }

*/

public function moveDepartment($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::DEPARTMENT_MOVE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 获取部门列表

* @return boolean|array 成功返回结果

* {

* "errcode": 0,

* "errmsg": "ok",

* "department": [ //部门列表数据。以部门的order字段从小到大排列

* {

* "id": 1,

* "name": "广州研发中心",

* "parentid": 0

* },

* {

* "id": 2

* "name": "邮箱产品部",

* "parentid": 1

* }

* ]

* }

*/

public function getDepartment(){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::DEPARTMENT_LIST_URL.'access_token='.$this->access_token);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 创建成员

* @param array $data 结构体为:

* array(

* "userid" => "zhangsan",

* "name" => "张三",

* "department" => [1, 2],

* "position" => "产品经理",

* "mobile" => "15913215421",

* "gender" => 1, //性别。gender=0表示男,=1表示女

* "tel" => "62394",

* "email" => "zhangsan@gzdev.com",

* "weixinid" => "zhangsan4dev"

* )

* @return boolean|array

* 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "created", //对返回码的文本描述内容

* }

*/

public function createUser($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::USER_CREATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 更新成员

* @param array $data 结构体为:

* array(

* "userid" => "zhangsan",

* "name" => "张三",

* "department" => [1, 2],

* "position" => "产品经理",

* "mobile" => "15913215421",

* "gender" => 1, //性别。gender=0表示男,=1表示女

* "tel" => "62394",

* "email" => "zhangsan@gzdev.com",

* "weixinid" => "zhangsan4dev"

* )

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "updated" //对返回码的文本描述内容

* }

*/

public function updateUser($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::USER_UPDATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 删除成员

* @param $userid 员工UserID。对应管理端的帐号

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "deleted" //对返回码的文本描述内容

* }

*/

public function deleteUser($userid){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::USER_DELETE_URL.'access_token='.$this->access_token.'&userid='.$userid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 获取成员信息

* @param $userid 员工UserID。对应管理端的帐号

* @return boolean|array 成功返回结果

* {

* "errcode": 0,

* "errmsg": "ok",

* "userid": "zhangsan",

* "name": "李四",

* "department": [1, 2],

* "position": "后台工程师",

* "mobile": "15913215421",

* "gender": 1, //性别。gender=0表示男,=1表示女

* "tel": "62394",

* "email": "zhangsan@gzdev.com",

* "weixinid": "lisifordev", //微信号

* "avatar": "http://wx.qlogo.cn/mmopen/ajNVdqHZLLA3W..../0", //头像url。注:如果要获取小图将url最后的"/0"改成"/64"即可

* "status": 1 //关注状态: 1=已关注,2=已冻结,4=未关注

* }

*/

public function getUserInfo($userid){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::USER_GET_URL.'access_token='.$this->access_token.'&userid='.$userid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 获取部门成员

* @param $department_id 部门id

* @param $fetch_child 1/0:是否递归获取子部门下面的成员

* @param $status 0获取全部员工,1获取已关注成员列表,2获取禁用成员列表,4获取未关注成员列表。status可叠加

* @return boolean|array 成功返回结果

* {

* "errcode": 0,

* "errmsg": "ok",

* "userlist": [

* {

* "userid": "zhangsan",

* "name": "李四"

* }

* ]

* }

*/

public function getUserList($department_id,$fetch_child=0,$status=0){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::USER_LIST_URL.'access_token='.$this->access_token

.'&department_id='.$department_id.'&fetch_child='.$fetch_child.'&status='.$status);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 根据code获取成员信息

* 通过Oauth2.0或者设置了二次验证时获取的code,用于换取成员的UserId和DeviceId

*

* @param $code Oauth2.0或者二次验证时返回的code值

* @param $agentid 跳转链接时所在的企业应用ID,未填则默认为当前配置的应用id

* @return boolean|array 成功返回数组

* array(

* 'UserId' => 'USERID', //员工UserID

* 'DeviceId' => 'DEVICEID' //手机设备号(由微信在安装时随机生成)

* )

*/

public function getUserId($code,$agentid=0){

if (!$agentid) $agentid=$this->agentid;

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::USER_GETINFO_URL.'access_token='.$this->access_token.'&code='.$code.'&agentid'.$agentid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 创建标签

* @param array $data 结构体为:

* array(

* "tagname" => "UI"

* )

* @return boolean|array

* 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "created", //对返回码的文本描述内容

* "tagid": "1"

* }

*/

public function createTag($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::TAG_CREATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 更新标签

* @param array $data 结构体为:

* array(

* "tagid" => "1",

* "tagname" => "UI design"

* )

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "updated" //对返回码的文本描述内容

* }

*/

public function updateTag($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::TAG_UPDATE_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 删除标签

* @param $tagid 标签TagID

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "deleted" //对返回码的文本描述内容

* }

*/

public function deleteTag($tagid){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::TAG_DELETE_URL.'access_token='.$this->access_token.'&tagid='.$tagid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 获取标签成员

* @param $tagid 标签TagID

* @return boolean|array 成功返回结果

* {

* "errcode": 0,

* "errmsg": "ok",

* "userlist": [

* {

* "userid": "zhangsan",

* "name": "李四"

* }

* ]

* }

*/

public function getTag($tagid){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::TAG_GET_URL.'access_token='.$this->access_token.'&tagid='.$tagid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 增加标签成员

* @param array $data 结构体为:

* array (

* "tagid" => "1",

* "userlist" => array( //企业员工ID列表

* "user1",

* "user2"

* )

* )

* @return boolean|array

* 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "ok", //对返回码的文本描述内容

* "invalidlist":"usr1|usr2|usr" //若部分userid非法,则会有此段。不在权限内的员工ID列表,以“|”分隔

* }

*/

public function addTagUser($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::TAG_ADDUSER_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 删除标签成员

* @param array $data 结构体为:

* array (

* "tagid" => "1",

* "userlist" => array( //企业员工ID列表

* "user1",

* "user2"

* )

* )

* @return boolean|array

* 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "deleted", //对返回码的文本描述内容

* "invalidlist":"usr1|usr2|usr" //若部分userid非法,则会有此段。不在权限内的员工ID列表,以“|”分隔

* }

*/

public function delTagUser($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::TAG_DELUSER_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 获取标签列表

* @return boolean|array 成功返回数组结果,这里附上json样例

* {

* "errcode": 0,

* "errmsg": "ok",

* "taglist":[

* {"tagid":1,"tagname":"a"},

* {"tagid":2,"tagname":"b"}

* ]

* }

*/

public function getTagList(){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::TAG_LIST_URL.'access_token='.$this->access_token);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode'])) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 主动发送信息接口

* @param array $data 结构体为:

* array(

* "touser" => "UserID1|UserID2|UserID3",

* "toparty" => "PartyID1|PartyID2 ",

* "totag" => "TagID1|TagID2 ",

* "safe":"0"//是否为保密消息,对于news无效

* "agentid" => "001",//应用id

* "msgtype" => "text", //根据信息类型,选择下面对应的信息结构体

*

* "text" => array(

* "content" => "Holiday Request For Pony(http://xxxxx)"

* ),

*

* "image" => array(

* "media_id" => "MEDIA_ID"

* ),

*

* "voice" => array(

* "media_id" => "MEDIA_ID"

* ),

*

* " video" => array(

* "media_id" => "MEDIA_ID",

* "title" => "Title",

* "description" => "Description"

* ),

*

* "file" => array(

* "media_id" => "MEDIA_ID"

* ),

*

* "news" => array(//不支持保密

* "articles" => array( //articles 图文消息,一个图文消息支持1到10个图文

* array(

* "title" => "Title", //标题

* "description" => "Description", //描述

* "url" => "URL", //点击后跳转的链接。可根据url里面带的code参数校验员工的真实身份。

* "picurl" => "PIC_URL", //图文消息的图片链接,支持JPG、PNG格式,较好的效果为大图640*320,

* //小图80*80。如不填,在客户端不显示图片

* ),

* )

* ),

*

* "mpnews" => array(

* "articles" => array( //articles 图文消息,一个图文消息支持1到10个图文

* array(

* "title" => "Title", //图文消息的标题

* "thumb_media_id" => "id", //图文消息缩略图的media_id

* "author" => "Author", //图文消息的作者(可空)

* "content_source_url" => "URL", //图文消息点击“阅读原文”之后的页面链接(可空)

* "content" => "Content" //图文消息的内容,支持html标签

* "digest" => "Digest description", //图文消息的描述

* "show_cover_pic" => "0" //是否显示封面,1为显示,0为不显示(可空)

* ),

* )

* )

* )

* 请查看官方开发文档中的 发送消息 -> 消息类型及数据格式

*

* @return boolean|array

* 如果对应用或收件人、部门、标签任何一个无权限,则本次发送失败;

* 如果收件人、部门或标签不存在,发送仍然执行,但返回无效的部分。

* {

* "errcode": 0,

* "errmsg": "ok",

* "invaliduser": "UserID1",

* "invalidparty":"PartyID1",

* "invalidtag":"TagID1"

* }

*/

public function sendMessage($data){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_post(self::API_URL_PREFIX.self::MASS_SEND_URL.'access_token='.$this->access_token,self::json_encode($data));

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* 二次验证

* 企业在开启二次验证时,必须填写企业二次验证页面的url。

* 当员工绑定通讯录中的帐号后,会收到一条图文消息,

* 引导员工到企业的验证页面验证身份,企业在员工验证成功后,

* 调用如下接口即可让员工关注成功。

*

* @param $userid

* @return boolean|array 成功返回结果

* {

* "errcode": 0, //返回码

* "errmsg": "ok" //对返回码的文本描述内容

* }

*/

public function authSucc($userid){

if (!$this->access_token && !$this->checkAuth()) return false;

$result = $this->http_get(self::API_URL_PREFIX.self::AUTHSUCC_URL.'access_token='.$this->access_token.'&userid='.$userid);

if ($result)

{

$json = json_decode($result,true);

if (!$json || !empty($json['errcode']) || $json['errcode']!=0) {

$this->errCode = $json['errcode'];

$this->errMsg = $json['errmsg'];

return false;

}

return $json;

}

return false;

}

/**

* oauth 授权跳转接口

* @param string $callback 回调URI

* @param string $state 重定向后会带上state参数,企业可以填写a-zA-Z0-9的参数值

* @return string

*/

public function getOauthRedirect($callback,$state='STATE',$scope='snsapi_base'){

return self::OAUTH_PREFIX.self::OAUTH_AUTHORIZE_URL.'appid='.$this->appid.'&redirect_uri='.urlencode($callback).'&response_type=code&scope='.$scope.'&state='.$state.'#wechat_redirect';

}

}

/**

* PKCS7Encoder class

*

* 提供基于PKCS7算法的加解密接口.

*/

class PKCS7Encoder

{

public static $block_size = 32;

/**

* 对需要加密的明文进行填充补位

* @param $text 需要进行填充补位操作的明文

* @return 补齐明文字符串

*/

function encode($text)

{

$block_size = PKCS7Encoder::$block_size;

$text_length = strlen($text);

//计算需要填充的位数

$amount_to_pad = PKCS7Encoder::$block_size - ($text_length % PKCS7Encoder::$block_size);

if ($amount_to_pad == 0) {

$amount_to_pad = PKCS7Encoder::block_size;

}

//获得补位所用的字符

$pad_chr = chr($amount_to_pad);

$tmp = "";

for ($index = 0; $index < $amount_to_pad; $index++) {

$tmp .= $pad_chr;

}

return $text . $tmp;

}

/**

* 对解密后的明文进行补位删除

* @param decrypted 解密后的明文

* @return 删除填充补位后的明文

*/

function decode($text)

{

$pad = ord(substr($text, -1));

if ($pad < 1 || $pad > PKCS7Encoder::$block_size) {

$pad = 0;

}

return substr($text, 0, (strlen($text) - $pad));

}

}

/**

* Prpcrypt class

*

* 提供接收和推送给公众平台消息的加解密接口.

*/

class Prpcrypt

{

public $key;

function Prpcrypt($k)

{

$this->key = base64_decode($k . "=");

}

/**

* 对明文进行加密

* @param string $text 需要加密的明文

* @return string 加密后的密文

*/

public function encrypt($text, $appid)

{

try {

//获得16位随机字符串,填充到明文之前

$random = $this->getRandomStr();//"aaaabbbbccccdddd";

$text = $random . pack("N", strlen($text)) . $text . $appid;

// 网络字节序

$size = mcrypt_get_block_size(MCRYPT_RIJNDAEL_128, MCRYPT_MODE_CBC);

$module = mcrypt_module_open(MCRYPT_RIJNDAEL_128, '', MCRYPT_MODE_CBC, '');

$iv = substr($this->key, 0, 16);

//使用自定义的填充方式对明文进行补位填充

$pkc_encoder = new PKCS7Encoder;

$text = $pkc_encoder->encode($text);

mcrypt_generic_init($module, $this->key, $iv);

//加密

$encrypted = mcrypt_generic($module, $text);

mcrypt_generic_deinit($module);

mcrypt_module_close($module);

//print(base64_encode($encrypted));

//使用BASE64对加密后的字符串进行编码

return array(ErrorCode::$OK, base64_encode($encrypted));

} catch (Exception $e) {

//print $e;

return array(ErrorCode::$EncryptAESError, null);

}

}

/**

* 对密文进行解密

* @param string $encrypted 需要解密的密文

* @return string 解密得到的明文

*/

public function decrypt($encrypted, $appid)

{

try {

//使用BASE64对需要解密的字符串进行解码

$ciphertext_dec = base64_decode($encrypted);

$module = mcrypt_module_open(MCRYPT_RIJNDAEL_128, '', MCRYPT_MODE_CBC, '');

$iv = substr($this->key, 0, 16);

mcrypt_generic_init($module, $this->key, $iv);

//解密

$decrypted = mdecrypt_generic($module, $ciphertext_dec);

mcrypt_generic_deinit($module);

mcrypt_module_close($module);

} catch (Exception $e) {

return array(ErrorCode::$DecryptAESError, null);

}

try {

//去除补位字符

$pkc_encoder = new PKCS7Encoder;

$result = $pkc_encoder->decode($decrypted);

//去除16位随机字符串,网络字节序和AppId

if (strlen($result) < 16)

return "";

$content = substr($result, 16, strlen($result));

$len_list = unpack("N", substr($content, 0, 4));

$xml_len = $len_list[1];

$xml_content = substr($content, 4, $xml_len);

$from_appid = substr($content, $xml_len + 4);

} catch (Exception $e) {

//print $e;

return array(ErrorCode::$IllegalBuffer, null);

}

if ($from_appid != $appid)

return array(ErrorCode::$ValidateAppidError, null);

return array(0, $xml_content);

}

/**

* 随机生成16位字符串

* @return string 生成的字符串

*/

function getRandomStr()

{

$str = "";

$str_pol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";

$max = strlen($str_pol) - 1;

for ($i = 0; $i < 16; $i++) {

$str .= $str_pol[mt_rand(0, $max)];

}

return $str;

}

}

/**

* error code

* 仅用作类内部使用,不用于官方API接口的errCode码

*/

class ErrorCode

{

public static $OK = 0;

public static $ValidateSignatureError = 40001;

public static $ParseXmlError = 40002;

public static $ComputeSignatureError = 40003;

public static $IllegalAesKey = 40004;

public static $ValidateAppidError = 40005;

public static $EncryptAESError = 40006;

public static $DecryptAESError = 40007;

public static $IllegalBuffer = 40008;

public static $EncodeBase64Error = 40009;

public static $DecodeBase64Error = 40010;

public static $GenReturnXmlError = 40011;

public static $errCode=array(

'0'=>'无问题',

'40001'=>'签名验证错误',

'40002'=>'xml解析失败',

'40003'=>'sha加密生成签名失败',

'40004'=>'encodingAesKey 非法',

'40005'=>'appid 校验错误',

'40006'=>'aes 加密失败',

'40007'=>'aes 解密失败',

'40008'=>'解密后得到的buffer非法',

'40009'=>'base64加密失败',

'40010'=>'base64解密失败',

'40011'=>'生成xml失败',

);

public static function getErrText($err) {

if (isset(self::$errCode[$err])) {

return self::$errCode[$err];

}else {

return false;

};

}

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/512098.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

函数式编程的Java编码实践:利用惰性写出高性能且抽象的代码

简介&#xff1a; 本文会以惰性加载为例一步步介绍函数式编程中各种概念&#xff0c;所以读者不需要任何函数式编程的基础&#xff0c;只需要对 Java 8 有些许了解即可。 作者 | 悬衡 来源 | 阿里技术公众号 本文会以惰性加载为例一步步介绍函数式编程中各种概念&#xff0c;所…

WorkManager从入门到实践,有这一篇就够了

作者 | Eason来源 | 程序员巴士前言一般情况下&#xff0c;我们大部分的操作都是在app打开的时候进行的&#xff0c;但是在某些情况下&#xff0c;即使app关闭了&#xff0c;我们也可能需要执行必要的动作&#xff0c;或者会采取一个动作&#xff0c;而不是让用户等待加载&…

终端卡顿优化的全记录

简介&#xff1a; 目前手机SOC的性能越来越少&#xff0c;很多程序员在终端程序的开发过程中也不太注意性能方面的优化&#xff0c;尤其是不注意对齐和分支优化&#xff0c;但是这两种问题一旦出现所引发的问题&#xff0c;是非常非常隐蔽难查的&#xff0c;不过好在项目中用到…

brew安装指定版本mysql,Mac 系统为 Valet 开发环境安装指定版本 MySQL

Mac 系统为 Valet 开发环境安装指定版本 MySQL由 学院君 创建于1年前, 最后更新于 5个月前版本号 #31547 views1 likes0 collects在 Mac 系统下使用 Valet 作为 Laravel 本地开发环境的话&#xff0c;需要自行安装 MySQL 数据库&#xff0c;我们通过 Homebrew 来安装。如果之前…

系统架构面临的三大挑战,看 Kubernetes 监控如何解决?

简介&#xff1a; 随着 Kubernetes 的不断实践落地&#xff0c;我们经常会遇到负载均衡、集群调度、水平扩展等问题。归根到底&#xff0c;这些问题背后都暴露出流量分布不均的问题。那么&#xff0c;我们该如何发现资源使用&#xff0c;解决流量分布不均问题呢&#xff1f;今天…

JavaScript 数组你都掰扯不明白,还敢说精通 JavaScript ?| 赠书

作者 | 哪吒来源 | CSDN博客最近小编在看文章的时候&#xff0c;总有很多刚刚入门的小白说精通这个&#xff0c;精通那个技术&#xff0c;更有意思的是&#xff0c;最近看到一则简历上说精通 JavaScript &#xff0c;聊一聊发现数组还不明白&#xff0c;就对外说精通~所以今天小…

php同时删除两个列表数据库,PHP 处理 数据库多表,既能高效又能思路清晰如何处理的?...

【情况如下】在表1中 记录着自行车编号 租户商家1 a2 a3 b4 c[这里的意思是&#xff0c;不同的商家。拥有的自行车&#xff0c;商家拥有自行车表]表2自行车编号 租出时间 租出金额1 2 41 3 51 6 102 3 53 4 7[这里 意味着。不同的商家 对自己的自行车 租出时间不同设置的价格&a…

基于消息队列 RocketMQ 的大型分布式应用上云实践

简介&#xff1a; Apache RocketMQ 作为阿里巴巴开源的支撑万亿级数据洪峰的分布式消息中间件&#xff0c;在众多行业广泛应用。在选型过程中&#xff0c;开发者一定会关注开源版与商业版的业务价值对比。 那么&#xff0c;今天就围绕着商业版本的消息队列 RocketMQ和开源版本 …

Gartner发布2022年政府行业主要技术趋势:XaaS、数字化、超自动化等

作者 | Gartner研究副总裁 Bettina Tratz-Ryan Gartner杰出研究副总裁John Kost Gartner高级研究总监 相斌斌 供稿 | Gartner 政府领导人和民选官员在2022年不仅要面对巨大的挑战&#xff0c;还要把握疫情与经济复苏应对措施、不断变化的政治需求和持续数字化变革所带来的机遇…

php include无效,php 两次include后,第一个include里的变量无效了

php 目录结构 rootindex.phpconfig.phpcindex.phproot/config.php 里的内容回复讨论(解决方案)首先,解析顺序如下1.文件路径是绝对路径?是, 包含, 解析结束。不是, 进入下一步。2.文件路径是相对路径(就像你的"../index.php")?是, 跳过include_path, 解析相对路径。…

RedShift到MaxCompute迁移实践指导

简介&#xff1a; 本文主要介绍Amazon Redshift如何迁移到MaxCompute&#xff0c;主要从语法对比和数据迁移两方面介绍&#xff0c;由于Amazon Redshift和MaxCompute存在语法差异&#xff0c;这篇文章讲解了一下语法差异 1.概要 本文档详细介绍了Redshift和MaxCompute之间SQL…

冲击中国超融合第一,浪潮的底气从何而来?

2016年被业内很多人称为“超融合元年”&#xff0c;一众存储、服务器、网络、安全厂商如同“摔杯为号”一般&#xff0c;齐齐迈入了超融合市场。于是&#xff0c;风助火势&#xff0c;火借风威&#xff0c;那年的“超融合”烧得正热。六年后的今天&#xff0c;当我们再度将目光…

数字农业WMS库存操作重构及思考

简介&#xff1a; 数字农业库存管理系统在2020年时&#xff0c;部门对产地仓生鲜水果生产加工数字化的背景下应运而生。项目一期的数农WMS中的各类库存操作均为单独编写。而伴随着后续的不断迭代&#xff0c;这些库存操作间慢慢积累了大量的共性逻辑&#xff1a;如参数校验、幂…

three.js和php,threejs--初创项目

一.环境介绍倘若你只是使用Three.js库中所提供的几何体&#xff0c;且不载入任何纹理贴图&#xff0c;则网页是可以从本地的文件系统中打开&#xff0c;并且是能够直接运行的&#xff0c;只需在文件管理器中双击HTML文件&#xff0c;它就可以在浏览器中进行显示。 (此时你将在地…

数字营销行业大数据平台云原生升级实战

简介&#xff1a; 加和科技CTO 王可攀&#xff1a;技术是为业务价值而服务 王可攀 加和科技CTO 本文将基于加和科技大数据平台升级过程中面临的问题和挑战、如何调整数据平台架构以及调整后的变化&#xff0c;为大家介绍数字营销行业大数据平台云原生升级实战经验。主要分为以…

云上更安全?亚马逊云科技宣布将持续加大在中国区域安全合规领域投入

编辑 | 宋慧 出品 | CSDN云计算 新冠疫情对我们工作产生了深远的影响&#xff0c;远程在线的工作与交流愈加普及&#xff0c;国内更多公司在推出居家办公的“混合办公”模式。不过&#xff0c;这也给了网络攻击更多的机会&#xff0c;CSDN看到&#xff0c;有多个安全领域的报告…

场景模型驱动自动化测试在盒马的探索及实践

简介&#xff1a; 盒马业务有如下几个特点&#xff1a;线上线下一体化、仓储配送一体化、超市餐饮一体化、经营作业一体化、多业态与平台化。在以上的种种原因&#xff0c;生鲜及物流体验是盒马的特点&#xff0c;但仓储配送一体化作业中&#xff0c;如何能更高效的提升测试效率…

基于 KubeVela 的 GitOps 交付

简介&#xff1a; KubeVela 是一个简单、易用、且高可扩展的云原生应用管理和交付平台&#xff0c;KubeVela 背后的 OAM 模型天然解决了应用构建过程中对复杂资源的组合、编排等管理问题&#xff0c;同时也将后期的运维策略模型化&#xff0c;这意味着 KubeVela 可以结合 GitOp…

php curlopt_postfields,PHP的CURLOPT_POSTFIELDS参数使用数组和字符串的区别 - CSDN博客

PHP的CURL组件是非常常用的HTTP请求模拟器。通常要发送post数据时&#xff0c;我已经习惯于这样写&#xff1a;curl_setopt( $ch, CURLOPT_POSTFIELDS,$post_data);但是在向某一个服务器发送请求的时候&#xff0c;服务器返回500。而使用socket方式连接上去发送请求&#xff0c…

BCS2022大会将提前至5月 网络安全产业空间扩容将成热门话题

年度网络安全的盛会即将开启。 2022年3月30日&#xff0c;2022年北京网络安全大会&#xff08;BCS2022&#xff09;新闻发布会在北京奇安信安全中心召开&#xff0c;宣布2022年北京网络安全大会“提档”至5月24日至26日&#xff0c;并与北辰集团国家会议中心达成战略合作&#…