diff options
author | mayx | 2022-01-04 20:42:55 +0800 |
---|---|---|
committer | mayx | 2022-01-04 20:42:55 +0800 |
commit | f4aa957c53cda659d026ffd23856f65a72fee739 (patch) | |
tree | afc51b78e1ff241c955ca30910e895e02e0a1d22 /_posts/2021-02-23-picbotpro.md |
Restore deleted repositories
Diffstat (limited to '_posts/2021-02-23-picbotpro.md')
-rw-r--r-- | _posts/2021-02-23-picbotpro.md | 118 |
1 files changed, 118 insertions, 0 deletions
diff --git a/_posts/2021-02-23-picbotpro.md b/_posts/2021-02-23-picbotpro.md new file mode 100644 index 0000000..2247a12 --- /dev/null +++ b/_posts/2021-02-23-picbotpro.md @@ -0,0 +1,118 @@ +--- +layout: post +title: 快速自制微信图片机器人 +tags: [微信, 图片, Pixiv, 机器人, PHP] +--- + + 优化真的是很复杂啊……<!--more--> + +# 起因 + 前段时间,我做出来了[能发图片的机器人](/2021/02/19/picbot.html),做出来之后我拿给群友们体验,但是很遗憾的是那个代码实在是不太行,首先有2个问题,第一是微信获取`access_token`的次数是有限的,我的第一版代码在每一次调用都去获取`access_token`,这样很快次数就会消耗光,后来我稍微改进了一下,设置了个缓存,结果呢,我检测的时候用了次数更少的接口……简直是太蠢了……之后呢?结果今天发现代码里有两个获取`access_token`的地方,缓存完全没起到作用…… + 总之上面的问题各种波折总算是解决好了,然后还有一个问题是我的图片来源是[Lolicon API](https://api.lolicon.app/setu),然后调用限制是300次/天,说实话,对于一个人来说这个数量是够了,但是如果有很多人,像测试号最多能容纳100人,那每天每人也就只有3次调用的机会。 + 那要怎么解决调用次数的问题呢?我首先想的就是缓存结果。 + +# 解决API调用次数过少的问题 + 因为对于图片来说,基本上没有什么变化的信息,所以如果能将每一次的结果缓存的话其实也没有问题。所以说干就干,我单独开了一个仓库[pixiv-index](https://github.com/Mabbs/pixiv-index)用来存储缓存的结果,具体代码的话都在这个仓库里面,每天会调用那个API直到用完次数。 + 考虑到大多数情况下也不需要原图,所以这个API里的图片都只是长或宽最大为1200px的缩略图。 + 使用方法也很简单,像PHP的话就可以这样写: +```php +<?php +$raw=json_decode(file_get_contents("https://mabbs.github.io/pixiv-index/index.json"),true); +echo file_get_contents('https://mabbs.github.io/pixiv-index/data/'.$raw[rand(0,count($raw)-1)]); +``` + 虽然问题解决了,但是我发现了一个巨大的缺陷,我设计这个脚本的初心是想着它有非常多的数据供我调用,结果我发现我错了,之前没有仔细看他们的文档,现在看了才发现,我想要的图片他们也只有仅仅3361张而已,实在是太少了,而总共的图片数量也只有17285张而已(即使那个站的数据也在以非常缓慢的速度增加)…… + 我只是懒得去别的地方找,而且因为这个API作者说那些图片都是Ta精心挑选的我才特意写了那个仓库的那些脚本,还特地学了一下Github Action…… ~~(虽然实际上是抄的那个[给开发者账号续命的](https://github.com/wangziyingwen/AutoApiSecret)那个仓库lol)~~ + +# 新的代码 + 解决了那些问题之后我又稍微优化了一下,把聊天机器人的功能剥离掉了,免得那个图灵机器人的API让人混乱。 +```php +<?php +$appid='微信appID'; +$secret='微信appsecret'; +$token='和配置的Token配置一致即可'; +$source='https://i.pximg.net'; + +ini_set('session.gc_maxlifetime', 7200); +ignore_user_abort(true); +set_time_limit(0); +session_id('Storage'); +session_start(); +if(!json_decode(file_get_contents('https://api.weixin.qq.com/cgi-bin/get_api_domain_ip?access_token='.$_SESSION['access_token']),true)['ip_list']){ +$_SESSION['access_token']=json_decode(file_get_contents('https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid='.$appid.'&secret='.$secret),true)['access_token']; +} +if($_GET["upap"]){ +define('MULTIPART_BOUNDARY', '--------------------------'.microtime(true)); + +$retry=3; +while(!$picdata||$retry<=0){ +$raw=json_decode(file_get_contents("https://mabbs.github.io/pixiv-index/index.json"),true); +$picdata=file_get_contents($source.json_decode(file_get_contents('https://mabbs.github.io/pixiv-index/data/'.$raw[rand(0,count($raw)-1)]),true)['url'], false, stream_context_create(array('http' => array('method' => 'GET','header' => "referer: https://www.pixiv.net/")))); +$retry-=1; +} + +$context = stream_context_create(array( + 'http' => array( + 'method' => 'POST', + 'header' => 'Content-Type: multipart/form-data; boundary='.MULTIPART_BOUNDARY, + 'content' => "--".MULTIPART_BOUNDARY."\r\n". + "Content-Disposition: filename=\"image.jpg\"\r\n". + "Content-Type: image/jpg\r\n\r\n". + $picdata."\r\n". + "--".MULTIPART_BOUNDARY."--\r\n" + ) +)); + +file_get_contents('https://api.weixin.qq.com/cgi-bin/message/custom/send?access_token='.$_SESSION['access_token'] , false, stream_context_create(array('http' => array('method' => 'POST','header' => 'Content-type: application/json;charset=utf-8','content' => '{ + "touser":"'.$_GET["openid"].'", + "msgtype":"image", + "image": + { + "media_id":"'.json_decode(file_get_contents('https://api.weixin.qq.com/cgi-bin/media/upload?access_token='.$_SESSION['access_token'].'&type=image', false, $context),true)['media_id'].'" + } +}')))); +exit(); +} + +$timestamp=$_GET["timestamp"]; +$nonce=$_GET["nonce"]; +$tmpArr=array($token, $timestamp, $nonce); +sort($tmpArr, SORT_STRING); +if( sha1(implode($tmpArr)) == $_GET["signature"] ){ +if($_GET["echostr"]){ +echo $_GET["echostr"]; +}else{ + +// 加载XML内容 +$content = file_get_contents("php://input"); +$p = xml_parser_create(); +xml_parse_into_struct($p, $content, $vals, $index); +xml_parser_free($p); +if($vals[$index['MSGTYPE'][0]]['value'] == 'text'){ +if($vals[$index['CONTENT'][0]]['value'] == '来点色图'){ + +echo '<xml> + <ToUserName><![CDATA['.$vals[$index['FROMUSERNAME'][0]]['value'].']]></ToUserName> + <FromUserName><![CDATA['.$vals[$index['TOUSERNAME'][0]]['value'].']]></FromUserName> + <CreateTime>'.time().'</CreateTime> + <MsgType><![CDATA[text]]></MsgType> + <Content><![CDATA[开始发起请求,请耐心等待]]></Content> +</xml>'; + +file_get_contents('https://'.$_SERVER['HTTP_HOST'].$_SERVER['PHP_SELF'].'?upap=1&openid='.$vals[$index['FROMUSERNAME'][0]]['value'], false, stream_context_create(array('http' => array('timeout' => 0.5)))); + +}else{ +echo 'success'; +} +} +} +}else{ +echo 'error'; +} +``` + 2021.02.26更新:似乎在库中的图片有一些是被删掉了,所以为了提高回复的成功率,增加了3次重试。 + +# 如何使用? + 具体应该不需要我说了吧,看之前的几篇关于微信机器人的文章里面的这段就行了。这里我删掉了2个参数,又增加了2个,一个是Token,想填啥都行,只要和测试号里配置一样就行。另一个是source,那个是Pixiv的图片服务器,如果后端服务器在国外那这个就不用管了,如果在国内的话需要改成`https://i.pixiv.cat`来做反代,或者如果有其他反代服务也可以,自己用CloudFlare Worker建一个也没有问题。 + +# 结尾 + 那个Lolicon API实在是不好用,不过我也懒得解决了,所以就托学弟在做Pixiv日榜的收集,回头看看效果怎么样,实在不行就去研究一下各种各样的什么booru之类的图站吧,用那些图片也是个不错的选择。 |