TaoLer/app/BaseController.php

488 lines
15 KiB
PHP
Raw Normal View History

2020-01-01 13:17:19 +08:00
<?php
declare (strict_types = 1);
namespace app;
use think\App;
use think\exception\ValidateException;
2022-11-18 10:31:44 +08:00
use think\response\Json;
2020-01-01 13:17:19 +08:00
use think\Validate;
2022-08-02 20:48:54 +08:00
use think\Response;
use think\exception\HttpResponseException;
use think\facade\Db;
use think\facade\Request;
use think\facade\Lang;
use think\facade\View;
2022-08-16 15:19:21 +08:00
use taoser\SetArr;
use app\common\lib\Uploads;
2020-01-01 13:17:19 +08:00
/**
* 控制器基础类
*/
abstract class BaseController
{
/**
* Request实例
* @var \think\Request
*/
protected $request;
/**
* 应用实例
* @var \think\App
*/
protected $app;
/**
* 是否批量验证
* @var bool
*/
protected $batchValidate = false;
/**
* 控制器中间件
* @var array
*/
protected $middleware = [];
/**
* 构造方法
* @access public
* @param App $app 应用对象
*/
public function __construct(App $app)
{
$this->app = $app;
$this->request = $this->app->request;
// 控制器初始化
$this->initialize();
}
// 初始化
protected function initialize()
{}
/**
* 验证数据
* @access protected
* @param array $data 数据
* @param string|array $validate 验证器名或者验证规则数组
* @param array $message 提示信息
* @param bool $batch 是否批量验证
* @return array|string|true
* @throws ValidateException
*/
protected function validate(array $data, $validate, array $message = [], bool $batch = false)
{
if (is_array($validate)) {
$v = new Validate();
$v->rule($validate);
} else {
if (strpos($validate, '.')) {
// 支持场景
2020-01-17 14:57:05 +08:00
[$validate, $scene] = explode('.', $validate);
2020-01-01 13:17:19 +08:00
}
$class = false !== strpos($validate, '\\') ? $validate : $this->app->parseClass('validate', $validate);
$v = new $class();
if (!empty($scene)) {
$v->scene($scene);
}
}
$v->message($message);
// 是否批量验证
if ($batch || $this->batchValidate) {
$v->batch(true);
}
return $v->failException(true)->check($data);
}
2022-08-02 20:48:54 +08:00
//显示网站设置
protected function getSystem()
{
//1.系统配置信息
return Db::name('system')->cache('system',3600)->find(1);
}
//域名协议转换 把数据库中的带HTTP或不带协议的域名转换为当前协议的域名前缀
protected function getHttpUrl($url)
{
//域名转换为无http协议
$www = stripos($url,'://') ? substr(stristr($url,'://'),3) : $url;
$htpw = Request::scheme().'://'. $www;
return $htpw;
}
2023-05-05 11:58:42 +08:00
protected function getDomain()
{
return $this->getHttpUrl($this->getSystem()['domain']);
}
/**
* 运行时间计算
* @return string
*/
2022-08-02 20:48:54 +08:00
protected function getRunTime()
{
//运行时间
$now = time();
$sys = $this->getSystem();
$count = $now-$sys['create_time'];
$days = floor($count/86400);
$hos = floor(($count%86400)/3600);
$mins = floor(($count%3600)/60);
$years = floor($days/365);
if($years >= 1){
$days = floor($days%365);
}
$runTime = $years ? "{$years}{$days}{$hos}{$mins}" : "{$days}{$hos}{$mins}";
return $runTime;
}
/**
2023-05-05 11:58:42 +08:00
* 非admin应用的文章url路由地址
* @param int $aid
* @param $ename
2022-08-02 20:48:54 +08:00
* @return string
*/
2023-05-05 11:58:42 +08:00
protected function getRouteUrl(int $aid, $ename = '')
2022-08-02 20:48:54 +08:00
{
2023-05-05 11:58:42 +08:00
$domain = $this->getDomain();
$appName = app('http')->getName();
$articleUrl = (string) url('article_detail', ['id' => $aid]);
// 详情动态路由,$aid, $ename
2022-08-02 21:13:36 +08:00
if(config('taoler.url_rewrite.article_as') == '<ename>/'){
2023-05-05 11:58:42 +08:00
$articleUrl = (string) url('article_detail', ['id' => (int) $aid, 'ename'=> $ename]);
2023-03-16 22:30:36 +08:00
}
2023-05-05 11:58:42 +08:00
// // 判断应用是否绑定域名
// $app_bind = array_search($appName, config('app.domain_bind'));
// // 判断应用是否域名映射
// $app_map = array_search($appName, config('app.app_map'));
2022-08-02 20:48:54 +08:00
2023-05-05 11:58:42 +08:00
//a.appName不是admin
return $domain . $articleUrl;
2022-08-02 20:48:54 +08:00
}
2022-08-16 15:19:21 +08:00
/**
* 关键词
* 通过百度分词接口获取关键词或者标签
* flag 1.为word时获取分词2.为tag时获取标签
2022-08-16 15:19:21 +08:00
*
2023-02-11 21:24:02 +08:00
* @return array
2022-08-16 15:19:21 +08:00
*/
2022-08-16 15:29:04 +08:00
public function setKeywords($data)
2022-08-16 15:19:21 +08:00
{
2022-08-16 15:29:04 +08:00
$keywords = [];
// 百度分词自动生成关键词
if(!empty(config('taoler.baidu.client_id')) == true) {
//headers数组内的格式
$headers = array();
$headers[] = "Content-Type:application/json";
switch($data['flag']) {
//分词
case 'word':
$url = 'https://aip.baidubce.com/rpc/2.0/nlp/v1/lexer?charset=UTF-8&access_token='.config('taoler.baidu.access_token');
$body = ["text" => $data['keywords']];
break;
//标签
case 'tag':
$url = 'https://aip.baidubce.com/rpc/2.0/nlp/v1/keyword?charset=UTF-8&access_token='.config('taoler.baidu.access_token');
$body = ['title' => $data['keywords'], 'content'=>$data['content']];
break;
default:
$url = 'https://aip.baidubce.com/rpc/2.0/nlp/v1/lexer?charset=UTF-8&access_token='.config('taoler.baidu.access_token');
$body = ["text" => $data['keywords']];
}
$postBody = json_encode($body);
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $url);
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_HTTPHEADER, $headers);//设置请求头
curl_setopt($curl, CURLOPT_POSTFIELDS, $postBody);//设置请求体
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($curl, CURLOPT_CUSTOMREQUEST, 'POST');//使用一个自定义的请求信息来代替"GET"或"HEAD"作为HTTP请求。(这个加不加没啥影响)
$datas = curl_exec($curl);
if($datas == false) {
echo '接口无法链接';
} else {
$res = stripos($datas,'error_code');
// 接收返回的数据
$dataItem = json_decode($datas);
if($res == false) {
// 数据正常
$items = $dataItem->items;
foreach($items as $item) {
switch($data['flag']) {
case 'word':
if($item->pos == 'n' && !in_array($item->item,$keywords)){
$keywords[] = $item->item;
}
break;
case 'tag':
if(!in_array($item->tag,$keywords)){
$keywords[] = $item->tag;
}
break;
default:
if($item->pos == 'n' && !in_array($item->item,$keywords)){
$keywords[] = $item->item;
}
}
}
} else {
// 接口正常但获取数据失败可能参数错误重新获取token
$url = 'https://aip.baidubce.com/oauth/2.0/token';
$post_data['grant_type'] = config('taoler.baidu.grant_type');;
$post_data['client_id'] = config('taoler.baidu.client_id');
$post_data['client_secret'] = config('taoler.baidu.client_secret');
$o = "";
foreach ( $post_data as $k => $v )
{
$o.= "$k=" . urlencode( $v ). "&" ;
}
$post_data = substr($o,0,-1);
$res = $this->request_post($url, $post_data);
// 写入token
SetArr::name('taoler')->edit([
'baidu'=> [
'access_token' => json_decode($res)->access_token,
]
]);
echo 'api接口数据错误 - ';
echo $dataItem->error_msg;
}
}
}
2023-02-11 21:24:02 +08:00
return $keywords;
2022-08-16 15:19:21 +08:00
}
// api_post接口
function request_post($url = '', $param = '')
{
if (empty($url) || empty($param)) {
return false;
}
$postUrl = $url;
$curlPost = $param;
$curl = curl_init();//初始化curl
curl_setopt($curl, CURLOPT_URL,$postUrl);//抓取指定网页
curl_setopt($curl, CURLOPT_HEADER, 0);//设置header
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);//要求结果为字符串且输出到屏幕上
curl_setopt($curl, CURLOPT_POST, 1);//post提交方式
curl_setopt($curl, CURLOPT_POSTFIELDS, $curlPost);
$data = curl_exec($curl);//运行curl
curl_close($curl);
return $data;
}
/**
* 标题调用百度关键词词条
*
2022-11-18 10:31:44 +08:00
* @return Json
2022-08-16 15:19:21 +08:00
*/
public function getBdiduSearchWordList($words)
{
if(empty($words)) return json(['code'=>-1,'msg'=>'null']);
$url = 'https://www.baidu.com/sugrec?prod=pc&from=pc_web&wd='.$words;
//$result = Api::urlGet($url);
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $url);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
$datas = curl_exec($curl);
curl_close($curl);
$data = json_decode($datas,true);
if(isset($data['g'])) {
return json(['code'=>0,'msg'=>'success','data'=>$data['g']]);
} else {
return json(['code'=>-1,'msg'=>'null']);
}
}
/**
* 上传接口
*
* @return void
*/
public function uploadFiles($type)
{
2023-05-05 12:09:57 +08:00
$max_file_seze = $this->getSystem()['upsize'];
2022-08-16 15:19:21 +08:00
$uploads = new Uploads();
switch ($type){
case 'image':
2023-05-05 12:09:57 +08:00
$upRes = $uploads->put('file','article_pic',$max_file_seze,'image');
2022-08-16 15:19:21 +08:00
break;
case 'zip':
2023-05-05 12:09:57 +08:00
$upRes = $uploads->put('file','article_zip',$max_file_seze,'application|image');
2022-08-16 15:19:21 +08:00
break;
case 'video':
2023-05-05 12:09:57 +08:00
$upRes = $uploads->put('file','article_video',$max_file_seze,'video|audio');
2022-08-16 15:19:21 +08:00
break;
case 'audio':
2023-05-05 12:09:57 +08:00
$upRes = $uploads->put('file','article_audio',$max_file_seze,'audio');
2022-08-16 15:19:21 +08:00
break;
default:
2023-05-05 12:09:57 +08:00
$upRes = $uploads->put('file','article_file',$max_file_seze,'image');
2022-08-16 15:19:21 +08:00
break;
}
return $upRes;
}
2022-09-23 11:03:10 +08:00
//获取artcile内容所有图片返回数组
2022-08-16 15:29:04 +08:00
protected function getArticleAllpic($str)
{
2022-09-23 11:03:10 +08:00
//正则匹配<img src="http://img.com" />
2022-08-16 15:29:04 +08:00
$pattern = "/<[img|IMG].*?src=[\'|\"](.*?(?:[\.gif|\.jpg|\.png]))[\'|\"].*?[\/]?>/";
preg_match_all($pattern,$str,$matchContent);
if(isset($matchContent[1])){
$img = $matchContent[1];
}else{
$temp = "./images/no-image.jpg";//在相应位置放置一张命名为no-image的jpg图片
}
return $img;
}
2022-09-23 11:03:10 +08:00
//下载远程图片
2022-08-16 15:29:04 +08:00
private function downloadImage($url)
{
$ch = curl_init();
curl_setopt ($ch, CURLOPT_CUSTOMREQUEST, 'GET' );
curl_setopt ($ch, CURLOPT_SSL_VERIFYPEER, false );
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 30);
$file = curl_exec($ch);
curl_close($ch);
return $this->saveAsImage($url, $file);
}
2022-09-23 11:03:10 +08:00
//把图片保存到本地
2022-08-16 15:29:04 +08:00
private function saveAsImage($url, $file)
{
$filename = pathinfo($url, PATHINFO_BASENAME);
//$dirname = pathinfo(parse_url($url, PHP_URL_PATH), PATHINFO_DIRNAME);
$dirname = date('Ymd',time());
//路径
2023-05-05 11:58:42 +08:00
$path = 'storage/download/article_pic/' . $dirname . '/';
2022-08-16 15:29:04 +08:00
//绝对文件夹
$fileDir = public_path() . $path;
//文件绝对路径
$filePath = $fileDir . $filename;
//相对路径文件名
$realFile = '/' . $path . $filename;
// 如果目录不存在,则创建
if(!is_dir($fileDir)) {
mkdir($fileDir, 0777, true);
}
if(file_exists($filePath)) {
//$this->output->writeln("【已存在】输出路径" . $fullpath);
return $realFile;
} else {
$resource = fopen($filePath, 'a');
$res = fwrite($resource, $file);
fclose($resource);
if($res !== false) {
return $realFile;
}
}
}
//下载网络图片到本地并替换
public function downUrlPicsReaplace($content)
{
// 批量下载网络图片并替换
$images = $this->getArticleAllpic($content);
if(count($images)) {
foreach($images as $image){
2022-09-23 11:03:10 +08:00
//1.带http地址的图片2.非本站的网络图片 3.非带有?号等参数的图片
if((stripos($image,'http') !== false) && (stripos($image, Request::domain()) === false) && (stripos($image, '?') === false)) {
// 如果图片中没有带参数或者加密可下载
//下载远程图片(可下载)
$newImageUrl = $this->downloadImage($image);
//替换图片链接
$content = str_replace($image,Request::domain().$newImageUrl,$content);
2022-08-16 15:29:04 +08:00
}
}
2022-09-23 11:03:10 +08:00
//不可下载的图片如加密或者带有参数的图片如type=jpeg,直接返回content
2022-08-16 15:29:04 +08:00
}
2022-09-23 11:03:10 +08:00
2022-08-16 15:29:04 +08:00
return $content;
}
2023-05-05 11:58:42 +08:00
/**
* array_filter过滤返回函数
* @param $arr
* @return bool
*/
protected function filter($arr) :bool
{
if($arr === '' || $arr === null){
return false;
}
return true;
}
/**
* 过滤为空null等筛选参数
* @param array $array
* @return array
*/
public function getParamFilter(array $array) :array
{
2023-05-05 12:04:46 +08:00
return array_filter($array, function($arr){
if($arr === '' || $arr === null){
return false;
}
return true;
});
2023-05-05 11:58:42 +08:00
}
/**
* 数组根据sort字段数值进行排序
* @param array $array 数组
* @param string $sort 排序字段
* @return array 返回数组
*/
public function getArrSort(array $array, string $sort = 'sort') :array
{
// 排序
$cmf_arr = array_column($array, 'sort');
array_multisort($cmf_arr, SORT_ASC, $array);
return $array;
}
2023-05-05 12:07:11 +08:00
/**
* 过滤字符串中表情
* @param $str string 字符串内容
* @return string
*/
public function filterEmoji(string $str): string
{
$str = preg_replace_callback('/./u', function (array $match) {
return strlen($match[0]) >= 4 ? '' : $match[0];
}, $str);
return $str;
}
2020-01-01 13:17:19 +08:00
}