利用微博当图床-php语言实现

什么是微博图床?

微博图床,指把图片上传到新浪微博的图库,然后用这个图片外链达到节约本地服务器空间及加快图片加载速度的目的。

相比于其它的图床,利用微博当图床具有加载速度快、不限流量、不限容量等优点。网上有许多浏览器插件或在线工具能够实现。直接利用第三方的工具如果只是上传一两张图片倒也方便,如果要大批量上传并获取外链就得自己写一个了。

微博图床实现方法探究

目前已知有两个方法能实现,其一是利用微博官方的开放平台的发微博 API 实现。这种方法也是我最先考虑的方法,因为毕竟是官方渠道,应该靠谱些。

然而经过一番折腾后,我发现这一方法根本不可行。因为微博限制了这一 API 的调用频次,在连续上传时立即就会被封堵。 :sweat:

方法二是利用微博的 MINI 发布框(http://weibo.com/minipublish)的图片上传功能。

正是这个发布框暴露了微博的一个图片上传接口

http://picupload.service.weibo.com/interface/pic_upload.php

网上的这些微博图床大都也是利用这个接口实现的。

利用这一接口的优点是可以实现快速连续上传图片而不被封。缺点是必须要带着微博登录的 Cookie 才能访问。

http://picupload.service.weibo.com/interface/pic_upload.php

网上的这些微博图床大都也是利用这个接口实现的。

利用这一接口的优点是可以实现快速连续上传图片而不被封。缺点是必须要带着微博登录的 Cookie 才能访问。

PHP代码

说了这么多,现在贴一下相关php代码:

/**
 * 上传图片到微博图床
 * @author mengkun  http://mkblog.cn
 * @param $file 图片文件/图片url
 * @param $multipart 是否采用multipart方式上传
 * @return 返回的json数据
 */
function upload($file, $multipart = true) {
    $cookie = '';    // 微博cookie
    $url = 'http://picupload.service.weibo.com/interface/pic_upload.php'
    .'?mime=image%2Fjpeg&data=base64&url=0&markpos=1&logo=&nick=0&marks=1&app=miniblog';
    if($multipart) {
        $url .= '&cb=http://weibo.com/aj/static/upimgback.html?_wv=5&callback=STK_ijax_'.time();
        if (class_exists('CURLFile')) {     // php 5.5
            $post['pic1'] = new CURLFile(realpath($file));
        } else {
            $post['pic1'] = '@'.realpath($file);
        }
    } else {
        $post['b64_data'] = base64_encode(file_get_contents($file));
    }
    // Curl提交
    $ch = curl_init($url);
    curl_setopt_array($ch, array(
        CURLOPT_POST => true,
        CURLOPT_VERBOSE => true,
        CURLOPT_RETURNTRANSFER => true,
        CURLOPT_HTTPHEADER => array("Cookie: $cookie"),
        CURLOPT_POSTFIELDS => $post,
    ));
    $output = curl_exec($ch);
    curl_close($ch);
    // 正则表达式提取返回结果中的json数据
    preg_match('/({.*)/i', $output, $match);
    if(!isset($match[1])) return '';
    return $match[1];
}

如你所见,代码中包含两种上传方式,一种是 multipart 方式上传,通过这种方式上传效率较高,局限性是只能上传服务器的本地图片。 另一种方式是base64方式,可以上传远程图片。(文字的表述可能不清楚,用代码就明白了 :eyes: )

upload('mypic.jpg', true);  // multipart方式上传
upload('http://www.mysite.cn/mypic.jpg', false);  // 非multipart方式(base64)上传

无论通过以上哪种方式,返回的都会是一段如下所示的json数据。

{"code":"A20001","data":{"count":2,"data":"eyJ1aWQiOjIyODczOTc1NzUsImFwcCI6Im1pbmlibG9nIiwiY291bnQiOjIsInRpbWUiOjE0OTM3ODE0NzQuMzQ5LCJwaWNzIjp7InBpY18yIjp7InJldCI6LTIsIm5hbWUiOiJwaWNfMiJ9LCJwaWNfMSI6eyJ3aWR0aCI6MzAwLCJzaXplIjo4Njk5LCJyZXQiOjEsImhlaWdodCI6MzAwLCJuYW1lIjoicGljXzEiLCJwaWQiOiI4ODU2ZWFjN2d5MWZmODA4NGhtanpqMjA4YzA4YzN5ZSJ9fX0=","pics":{"pic_2":{"ret":-2,"name":"pic_2"},"pic_1":{"width":300,"size":8699,"ret":1,"height":300,"name":"pic_1","pid":"8856eac7gy1ff8084hmjzj208c08c3ye"}}}}

其它的不用管,我们只需要 pic_1 这个子节点中的数据

如图所示,pid码就是忙活了这么久之后最终要用到的。

http://ww2.sinaimg.cn/large/ + 获取到的pid 就能组合成图片的外链。

比如上面示范的那段数据中包含的图片外链地址就是

http://ww2.sinaimg.cn/large/8856eac7gy1ff8084hmjzj208c08c3ye

其实 pid 中还包含了更多“秘密”,这里提供一下 pid 的终极解码算法 坏笑 :

/** 
 * 获取图片链接(本函数修改自 https://github.com/consatan/weibo_image_uploader) 
 * 
 * @param string $pid 微博图床pid,或者微博图床链接。传递的是链接的话, 
 *     仅是将链接的尺寸更改为目标尺寸而已。 
 * @param string $size 图片尺寸 
 * @param bool $https (true) 是否使用 https 协议 
 * @return string 图片链接 
 * 当 $pid 既不是 pid 也不是合法的微博图床链接时返回空值 
 */  
function getImageUrl($pid, $size = 0, $https = true)  
{  
    $sizeArr = array('large', 'mw1024', 'mw690', 'bmiddle', 'small', 'thumb180', 'thumbnail', 'square');  
    $pid = trim($pid);  
    $size = $sizeArr[$size];  
    // 传递 pid  
    if (preg_match('/^[a-zA-Z0-9]{32}$/', $pid) === 1) {  
        return ($https ? 'https' : 'http') . '://' . ($https ? 'ws' : 'ww')  
            . ((crc32($pid) & 3) + 1) . ".sinaimg.cn/" . $size  
            . "/$pid." . ($pid[21] === 'g' ? 'gif' : 'jpg');  
    }  
    // 传递 url  
    $url = $pid;  
    $imgUrl = preg_replace_callback('/^(https?:\/\/[a-z]{2}\d\.sinaimg\.cn\/)'  
        . '(large|bmiddle|mw1024|mw690|small|square|thumb180|thumbnail)'  
        . '(\/[a-z0-9]{32}\.(jpg|gif))$/i', function ($match) use ($size) {  
            return $match[1] . $size . $match[3];  
        }, $url, -1, $count);  
    if ($count === 0) {  
        return '';  
    }  
    return $imgUrl;  
}

附:微博Cookie获取方法

首先登录微博,然后按F12打开控制台 Network,然后再刷新页面就可以在 Headers 中看到cookie。 利用微博当图床-php语言实现

注:微博 cookie 的有效期似乎只有一天……要想“永久”有效,还得研究研究微博的模拟登录算法……博主水平有限,暂未研究成功 大哭

转载:MKblog

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏拂晓风起

浅谈HTML5单页面架构(一)——requirejs + angular + angular-route

2472
来自专栏落影的专栏

iOS开发笔记(五)

前言 社会的模式很多是重复的,当你做一样事情很擅长时,与之类似的事情也能触类旁通。 正文 Code开发 1、delegate的trick 很多人习惯在调用de...

4994
来自专栏黑白安全

社会工程学之手机号伪造

任意显示手机号,使用者想令对方看到你的手机号是怎样的,对方就看到是怎样的,不会显示机主号。

2303
来自专栏农夫安全

信息安全面试题---(渗透测试工程师-1)

前言 本帖提供一些渗透测试工程师面试基础题目,有需要的小伙伴可以收藏 1.拿到一个待检测的站,你觉得应该先做什么? 0x01 面试题目 · 收集信息 whoi...

1.1K6
来自专栏极客慕白的成长之路

GifCam – 简单好用的 GIF 动画录制软件

GifCam 是一款小而美的 GIF 录制软件。她拥有一个迷人的特点 —— 通过将软件置顶在所有窗口之上,你可以像相机一样调整录制区域,移动或缩放窗口。

972
来自专栏北京马哥教育

centos7.0体验与之前版本的不同

今天下午,没事干,在一台机器上装了一个centos7玩一玩,发现与之前版本有很大不同,不知道rhel7是不是也是这样,毕竟现在centos属于redhat了。 ...

2317
来自专栏SAP梦心的SAP分享

【SAP业务模式】之ICS(七):IDOC配置

     这是ICS业务模式系列的最后一篇了,主要讲解IDOC的配置。      一、指定EDI传输的供应商逻辑地址      事务代码:WEL1 ?     ...

2147
来自专栏月色的自留地

在龙芯小本上安装Debain8.10

2406
来自专栏Seebug漏洞平台

TP-LINK 远程代码执行漏洞 CVE-2017-13772 趣谈

原文地址:《A CURIOUS TALE OF REMOTE CODE EXECUTION, THE TP-LINK STORY – CVE-2017-1377...

1.5K6
来自专栏菩提树下的杨过

再谈web开中几种经典的大文件上传组件

1.aspnetupload 这是国人开发的一款收费.net控件,官网http://www.aspnetupload.net/ (不过也有一个免费版本的,允许上...

2106

扫码关注云+社区

领取腾讯云代金券