需求
自己開發的小程式,例如社群類、電商類、新聞、部落格等型別小程式,會需要儲存很多圖片,圖片資源會給個人開發者帶來很多成本,萬一訪問量大了,網站頻寬、流量也會不堪重負,所以圖片資源必然不合適儲存在自己的伺服器裡面,而很多廠商提供的彈性儲存也是按需付費的。
想法
微信公眾平臺提供了一個素材管理,裡面可以上傳圖片、視訊、音訊等型別的素材,那麼是否可以將圖片素材上傳上去,獲取到url作為自己小程式專案的圖片伺服器呢?當然沒問題!
根據開發文件的說法:永久圖片素材新增後,將帶有URL返回給開發者,開發者可以在騰訊系域名內使用(騰訊系域名外使用,圖片將被遮蔽)。簡單來說就是你只能在騰訊的平臺使用這個圖片素材URL,否則無法展示,因為有防盜鏈。
開發
根據微信公眾平臺開發文件提供的【新增永久素材】介面可知,分為以下幾個步驟實現上傳圖片素材到微信伺服器。
- 獲取access_token
- 呼叫新增永久素材介面進行上傳素材
- 獲取到素材的永久URL
程式碼如下:
<?php
// 返回JSON
header("Content-type:application/json");
// 允許上傳的圖片字尾
$allowedExts = array("jpeg", "jpg", "png");
// 字尾名
if ($allowedExts[0] == 'jpeg') {
$hzm = 'jpg';
}else{
$hzm = $allowedExts[0];
}
// 獲取選擇的檔案
$temp = explode(".", $_FILES["file"]["name"]);
// 獲取檔案字尾名
$extension = end($temp);
if ((($_FILES["file"]["type"] == "image/gif")
|| ($_FILES["file"]["type"] == "image/jpeg")
|| ($_FILES["file"]["type"] == "image/jpg")
|| ($_FILES["file"]["type"] == "image/pjpeg")
|| ($_FILES["file"]["type"] == "image/x-png")
|| ($_FILES["file"]["type"] == "image/png"))
&& ($_FILES["file"]["size"] < 10485760)
&& in_array($extension, $allowedExts)){
if ($_FILES["file"]["error"] > 0){
$result = array(
'code' => 201,
'msg' => '上傳失敗'.$_FILES["file"]["error"]
);
}else{
// 重新命名
$new_file = date("Y-m-d")."-".rand(10000,99999).".".$hzm;
// 上傳到自己的伺服器
move_uploaded_file($_FILES["file"]["tmp_name"], "upload/".$new_file);
// 獲取真實地址
$filepath = realpath(dirname(__FILE__));
$upload_filepath = $filepath."/upload/".$new_file;
// 上傳到微信伺服器
$imgurl = upload_img($upload_filepath);
// 驗證上傳結果
if(strpos($imgurl,'http') !== false){
// 上傳成功
$result = array(
'code' => 200,
'msg' => '上傳成功',
'url' => $imgurl
);
// 刪除本地素材
unlink($upload_filepath);
}else{
// 上傳失敗
$result = array(
'code' => 202,
'msg' => '上傳失敗'
);
}
}
}
// 獲取access_token
function getToken(){
// appid和secret
$appid='填寫你公眾號的appid';
$appsecret='填寫你公眾號的appsecret';
// 讀取access_token
include './access_token.php';
// 判斷是否過期
if (time() > $access_token['expires']){
// 如果已經過期就得重新獲取並快取
$access_token = array();
$access_token['access_token'] = getNewToken($appid,$appsecret);
$access_token['expires']=time()+7000;
// 將陣列寫入php檔案
$arr = '<?php'.PHP_EOL.'$access_token = '.var_export($access_token,true).';'.PHP_EOL.'?>';
$arrfile = fopen("./access_token.php","w");
fwrite($arrfile,$arr);
fclose($arrfile);
// 返回當前的access_token
return $access_token['access_token'];
}else{
// 如果沒有過期就直接讀取快取檔案
return $access_token['access_token'];
}
}
// 獲取新的access_token
function getNewToken($appid,$appsecret){
$url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={$appid}&secret={$appsecret}";
$access_token_Arr = https_request($url);
return $access_token_Arr['access_token'];
}
// curl請求函式
function https_request ($url){
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
$out = curl_exec($ch);
curl_close($ch);
return json_decode($out,true);
}
// 上傳圖片素材到微信伺服器
function upload_img($realpath){
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, 'https://api.weixin.qq.com/cgi-bin/material/add_material?access_token='.getToken().'&type=image');
$data = array(
'media' => new CURLFile($realpath)
);
curl_setopt($ch, CURLOPT_POSTFIELDS, $data);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 2);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$upimg = curl_exec($ch);
return json_decode($upimg)->url;
curl_close($ch);
}
// 返回JSON物件
echo json_encode($result,JSON_UNESCAPED_UNICODE);
?>
請求返回
{"media_id":"8IZhZUPXxsG_omeA5giO5By8VyHnjk7_oy0Co9jVWwxpgm-sqhx_Hkz_9rLVF9Ws","url":"http:\/\/mmbiz.qpic.cn\/mmbiz_png\/5zLSKyuEW2Kt5ZGZg7XUx05QyGOVFCpHqKic74qQP4gxzQJYXjwN4aGEiadtfUXax4fCXXV5ia1UnvSwdqxuqLCqA\/0?wx_fmt=png","item":[]}
程式碼解釋
- 首先要將圖片從客戶端(小程式端)上傳到你自己的伺服器的upload目錄
- 然後呼叫新增永久素材介面從upload目錄上傳到微信的伺服器
- 需要配置公眾號的appid和appsecret獲取access_token
- 新增永久素材介面需要傳入access_token才可以呼叫
- access_token有效期是2小時,每天最多獲取2000次access_token,超過次數就獲取不到,為了能夠保證一直都能獲取到新的token,不能每上傳一次就獲取一個新的access_token,量大的話一下子就把2000次用完了。access_token只要在2小時內都可以隨便使用,所以需要進行快取access_token。
getToken()
這個函式就是快取access_token的步驟(如果access_token已經超過2小時就獲取新的access_token並快取到本地,如果access_token還沒超過2小時直接讀取本地快取的access_token) - 獲取access_token介面需要在公眾號後臺的安全中心配置白名單ip地址
注意
圖片連結只能在騰訊域名的平臺使用,在自建網站無法顯示,本文主要將的是小程式呼叫永久圖片素材,是可以用的!
小妙招
如果需要在非騰訊域名下使用永久圖片素材的URL,需要搭配反防盜鏈圖片跳板,類似http://xxx.com/?imgurl=永久圖片素材URL
來繞過防盜鏈,簡單來說就是一個圖片伺服器代理吧。
作者
TANKING