TaoLer/app/admin/controller/Seo.php

505 lines
18 KiB
PHP
Raw Normal View History

2022-04-17 17:09:19 +08:00
<?php
/*
* @Author: TaoLer <alipey_tao@qq.com>
* @Date: 2022-04-13 09:54:31
2022-08-16 15:29:04 +08:00
* @LastEditTime: 2022-08-14 09:23:13
2022-04-17 17:09:19 +08:00
* @LastEditors: TaoLer
* @Description: 搜索引擎SEO优化设置
2022-08-16 15:29:04 +08:00
* @FilePath: \github\TaoLer\app\admin\controller\Seo.php
2022-08-02 21:13:36 +08:00
* Copyright (c) 2020~2022 https://www.aieok.com All rights reserved.
2022-04-17 17:09:19 +08:00
*/
declare(strict_types=1);
namespace app\admin\controller;
use app\common\controller\AdminController;
use think\facade\View;
use think\facade\Request;
use think\facade\Db;
use taoser\SetArr;
2022-08-16 15:29:04 +08:00
use app\common\model\PushJscode;
2022-04-17 17:09:19 +08:00
class Seo extends AdminController
{
2022-08-02 21:13:36 +08:00
// 写ID
protected $w_id = 0;
2022-04-17 17:09:19 +08:00
public function index()
{
2022-08-02 18:46:05 +08:00
// 站点地图
$xml = '';
$xmlArr = $this->getXmlFile(public_path());
foreach($xmlArr as $v) {
$map = $this->getIndexUrl().'/'.$v;
$xml .= $map."\n";
}
// robots
2022-04-17 17:09:19 +08:00
if(is_file($rob = public_path().'robots.txt')){
$robots = file_get_contents($rob);
} else {
$robots = '';
2022-08-02 18:46:05 +08:00
}
// push_js
$pushjs = new PushJscode();
2022-08-16 15:29:04 +08:00
$jscode = $pushjs->getAllCodes(1);
2022-08-02 18:46:05 +08:00
View::assign(['xml'=>$xml,'jscode'=>$jscode,'robots'=>$robots]);
2022-04-17 17:09:19 +08:00
return View::fetch();
}
2022-08-02 21:13:36 +08:00
/**
* 百度推送
*
* @return void
*/
2022-04-17 17:09:19 +08:00
public function push()
{
2022-08-02 18:47:24 +08:00
$data = Request::only(['start_id','end_id','time']);
2022-08-02 21:13:36 +08:00
// 动态路由配置
$article_as = config('taoler.url_rewrite.article_as');
$api = config('taoler.baidu.push_api');
2022-08-02 21:13:36 +08:00
if(empty($api)) return json(['code'=>-1,'msg'=>'请先配置接口push_api']);
2022-04-17 17:09:19 +08:00
$urls = [];
if(empty($data['start_id']) || empty($data['end_id'])) {
2022-08-02 21:13:36 +08:00
if($article_as == '<ename>/'){ //变量路由
$artAllId = Db::name('article')
->alias('a')
->join('cate c','a.cate_id = c.id')
->field('a.id as aid,ename')
->where(['a.delete_time'=>0,'a.status'=>1])
->whereTime('a.create_time', $data['time'])
->select()->toArray();
} else { //常量路由
$artAllId = Db::name('article')->where(['delete_time'=>0,'status'=>1])->whereTime('create_time', $data['time'])->column('id');
}
} else {
2022-08-02 21:13:36 +08:00
if($article_as == '<ename>/') {
$artAllId = Db::name('article')
->alias('a')
->join('cate c','a.cate_id = c.id')
->field('a.id as aid,ename')
->where(['a.delete_time'=>0,'a.status'=>1])
->where('a.id','between',[$data['start_id'],$data['end_id']])
->whereTime('a.create_time', $data['time'])
->select()->toArray();
} else {
$artAllId = Db::name('article')->where(['delete_time'=>0,'status'=>1])->where('id','between',[$data['start_id'],$data['end_id']])->whereTime('create_time', $data['time'])->column('id');
}
}
2022-08-02 21:13:36 +08:00
2022-08-02 18:47:24 +08:00
if(empty($artAllId)) return json(['code'=>-1,'msg'=>'没有查询到结果,无需推送']);
// 组装链接数组
2022-08-02 21:13:36 +08:00
if($article_as == '<ename>/') {
foreach($artAllId as $art) {
$urls[] = $this->getRouteUrl($art['aid'], $art['ename']);
}
} else {
foreach($artAllId as $aid) {
$urls[] = $this->getRouteUrl($aid);
}
2022-04-17 17:09:19 +08:00
}
2022-08-02 21:13:36 +08:00
2022-08-02 18:47:24 +08:00
// 百度接口单次最大提交200进行分组
2022-08-02 21:13:36 +08:00
$urls = array_chunk($urls,2000);
2022-04-17 17:09:19 +08:00
$ch = curl_init();
2022-08-02 18:47:24 +08:00
foreach($urls as $url) {
$options = array(
CURLOPT_URL => $api,
CURLOPT_POST => true,
CURLOPT_RETURNTRANSFER => true,
CURLOPT_POSTFIELDS => implode("\n", $url),
CURLOPT_HTTPHEADER => array('Content-Type: text/plain'),
);
curl_setopt_array($ch, $options);
$result = curl_exec($ch);
if($result == false) {
return json(['code'=>-1,'msg'=>'接口失败']);
}
2022-04-17 17:09:19 +08:00
}
2022-08-02 18:47:24 +08:00
curl_close($ch);
2022-04-17 17:09:19 +08:00
$res = stripos($result,'error');
$data = json_decode($result);
if($res !== false) {
return json(['code'=>-1,'msg'=>$data->message]);
};
return json(['code'=>0,'msg'=>'成功推送'.$data->success.'条,今天剩余'.$data->remain]);
}
/**
* 百度接口配置
*
* @return void
*/
2022-04-17 17:09:19 +08:00
public function config()
{
$baidu = [];
$data = Request::only(['client_id','client_secret','push_api']);
foreach($data as $k => $v) {
if($v !== ''){
if($k == 'push_api') {
$baidu[$k] = "'$v'";
} else {
$baidu[$k] = $v;
}
}
}
2022-08-02 20:46:01 +08:00
// 获取Access Token分词的必选参数设置时写入
$baidu['grant_type'] = 'client_credentials';
$res = SetArr::name('taoler')->edit([
2022-04-17 17:09:19 +08:00
'baidu' => $baidu,
]);
if($res == true){
return json(['code'=>0,'msg'=>'设置成功']);
}
}
/**
* 百度谷歌sitemap生成xml文件
*
* @return void
*/
public function map()
{
$data = Request::only(['map_num','map_time','map_level']);
// 写文件字符串
$str = '';
2022-08-02 21:13:36 +08:00
// 标记每次调用首次调用写ID
2022-04-17 17:09:19 +08:00
$flag= true;
// 写ID
$w_id = '';
2022-08-02 18:48:31 +08:00
// 生成新文件编号,防止重复写入,
2022-04-17 17:09:19 +08:00
$i = 1;
// 获取public下所有xml文件
$newFile = $this->getXmlFile(public_path());
2022-08-02 21:13:36 +08:00
// 路由配置
$article_as = config('taoler.url_rewrite.article_as');
2022-04-17 17:09:19 +08:00
// 没有xml文件时不存在重写
if(empty($newFile)) {
$rewrite = false;
} else {
$xmlFile = end($newFile);
$strFile = file_get_contents(public_path().$xmlFile);
$num = substr_count($strFile, '<url>');
// 是否有需要追加在文件,判断最新文件是否未写满
$rewrite = $num < $data['map_num'] ? true : false;
}
2022-08-02 21:13:36 +08:00
2022-04-17 17:09:19 +08:00
if($rewrite){
// 需要追加的数量
$limit = (int) $data['map_num'] - $num;
$name = pathinfo($xmlFile,PATHINFO_FILENAME);
$arr = explode('_', $name);
// 检验当天,避免重复生成
if($arr[0] == date('Y-d-m'))
{
$i = $arr[1] + 1;
}
} else {
$limit = (int) $data['map_num'];
}
2022-08-02 21:13:36 +08:00
2022-04-17 17:09:19 +08:00
// 最新ID
$maxId = Db::name('article')->where(['delete_time'=>0,'status'=>1])->max('id');
2022-08-02 21:13:36 +08:00
2022-04-17 17:09:19 +08:00
do
{
2022-08-02 21:13:36 +08:00
$this->wr_id = $flag ? config('taoler.sitemap.write_id') : $this->w_id;
if($article_as == '<ename>/') {
// 分类名
$artAllId = Db::name('article')
->alias('a')
->join('cate c','a.cate_id = c.id')
->field('a.id as aid,a.update_time as uptime,ename')
->where(['a.delete_time'=>0,'a.status'=>1])
->where('a.id', '>', (int) $this->wr_id)
->order('a.id','asc')->limit($limit)
->select()->toArray();
} else {
$artAllId = Db::name('article')
2022-04-17 17:09:19 +08:00
->where(['delete_time'=>0,'status'=>1])
2022-08-02 21:13:36 +08:00
->where('id', '>', (int) $this->wr_id)
2022-04-17 17:09:19 +08:00
->order('id','asc')->limit($limit)->column('update_time','id');
2022-08-02 21:13:36 +08:00
}
2022-04-17 17:09:19 +08:00
if(empty($artAllId)) {
return json(['code'=>-1,'msg'=>'本次无需生成']);
} else {
// 本次最新文件ID
2022-08-02 21:13:36 +08:00
if($article_as == '<ename>/') {
// 循环拼接文件字符串
foreach($artAllId as $art) {
//url生成
$url = $this->getRouteUrl($art['aid'], $art['ename']);
$time = date('Y-m-d', $art['uptime']);
// 组装字符串
$str .= <<<STR
<url>
<loc>$url</loc>
<lastmod>$time</lastmod>
<changefreq>daily</changefreq>
<priority>0.5</priority>
</url>\n
STR;
}
$last_arr = end($artAllId);
$last_id = $last_arr['aid'];
} else {
// 循环拼接文件字符串
foreach($artAllId as $aid => $uptime) {
// url生成
$url = $this->getRouteUrl($aid);
$time = date('Y-m-d', $uptime);
// 组装字符串
$str .= <<<STR
<url>
<loc>$url</loc>
<lastmod>$time</lastmod>
<changefreq>daily</changefreq>
<priority>0.5</priority>
</url>\n
STR;
}
$last_id = array_key_last($artAllId);
2022-04-17 17:09:19 +08:00
}
2022-08-02 21:13:36 +08:00
2022-04-17 17:09:19 +08:00
// 写文件
if($rewrite){
// 写入旧xml文件
$reps = $str.'</urlset>';
$xml = preg_replace('/<\/urlset>/', $reps, $strFile);
$res = file_put_contents(public_path().$xmlFile, $xml);
if($res == false){
return json(['code'=>-1,'msg'=>$xmlFile.'写入失败']);
}
$limit = $data['map_num'];
$rewrite = false;
} else {
// 生成新xml
$xml = <<<XML
<?xml version="1.0" encoding="utf-8"?> <!-- XML文件需以utf-8编码-->
<urlset>\n$str</urlset>
XML;
$res = file_put_contents(public_path().date('Y-m-d').'_'.$i.'.xml', $xml);
if($res == false){
return json(['code'=>-1,'msg'=>date('Y-m-d').'_'.$i.'.xml写入失败']);
}
}
// 重置标记内容
$str = '';
$i++;
2022-08-02 21:13:36 +08:00
$this->w_id = $last_id;
2022-04-17 17:09:19 +08:00
$flag = false;
}
}
while($last_id < (int) $maxId);
// 写配置标记最后写入ID
$res = SetArr::name('taoler')->edit([
2022-04-17 17:09:19 +08:00
'sitemap' => [
'map_num' => $data['map_num'],
'write_id' => $last_id,
],
]);
if($res == false){
return json(['code'=>-1,'msg'=>'写xml配置失败']);
}
2022-08-02 21:13:36 +08:00
return json(['code'=>0,'msg'=>'本次成功生成' . count($artAllId) . '条xml']);
2022-04-17 17:09:19 +08:00
}
/**
* 返回public目录下xml名称数组
*
* @param string $dir
* @return array
*/
public function getXmlFile(string $dir) : array
{
$arr = [];
$files = array_diff(scandir($dir), array('.', '..'));
foreach ($files as $file) {
if(is_file("$dir/$file") && !is_link($dir) && (pathinfo("$dir/$file", PATHINFO_EXTENSION)) == 'xml')
{
$arr[] = "$file";
}
}
return $arr;
}
/**
* 生成robots
*
* @return void
*/
public function robots()
{
if(Request::isPost()){
$txt = input('robots');
$xmlArr = $this->getXmlFile(public_path());
foreach($xmlArr as $v) {
$res = stripos($txt, $v);
if($res == false){
$map = $this->getIndexUrl().'/'.$v;
$txt .= "\nsitemap:".$map;
}
}
$res = file_put_contents(public_path().'robots.txt',$txt);
if($res == false){
return json(['code'=>-1,'msg'=>$v.'写入失败']);
}
return json(['code'=>0,'msg'=>'设置成功']);
}
}
/**
2022-08-02 18:46:05 +08:00
* 保存搜索平台js代码
2022-04-17 17:09:19 +08:00
*
2022-08-02 18:46:05 +08:00
* @return void
2022-04-17 17:09:19 +08:00
*/
2022-08-02 18:46:05 +08:00
public function savePushJs()
2022-04-17 17:09:19 +08:00
{
2022-08-16 15:29:04 +08:00
$data = Request::only(['name','jscode','type']);
2022-08-02 18:46:05 +08:00
if(empty($data['name'])) {
return json(['code'=>-1,'msg'=>'请术输入名称']);
}
if(empty($data['jscode'])){
return json(['code'=>-1,'msg'=>'请术输入代码']);
}
$push = new PushJscode();
$res = $push->saveCode($data);
if(!$res) {
return json(['code'=>-1,'msg'=>'保存失败']);
2022-04-17 17:09:19 +08:00
}
2022-08-02 18:46:05 +08:00
return json(['code'=>0,'msg'=>'保存成功']);
}
2022-08-02 18:46:05 +08:00
/**
* 删除平台js代码
*
* @return void
*/
public function delPushJs()
{
$id = (int) input('id');
$push = new PushJscode();
$res = $push->delCode($id);
if(!$res) {
return json(['code'=>-1,'msg'=>'删除失败']);
}
return json(['code'=>0,'msg'=>'删除成功']);
2022-04-17 17:09:19 +08:00
}
2022-08-02 21:13:36 +08:00
/**
* 搜索引擎日志分析
*
* @return void
*/
2022-08-02 18:48:31 +08:00
public function searchLog()
{
$time = input('search_time');
2022-08-02 18:49:40 +08:00
$name = input('spider_name');
2022-08-02 20:39:46 +08:00
$page = input('page') ? input('page') : 1;
$limit = input('limit') ? input('limit') : 20;
2022-08-02 18:48:31 +08:00
$logPath = app()->getRootPath().'runtime/log/browse/'.$time.'.log';
$logPath = str_replace('\\','/',$logPath);
2022-08-02 18:49:40 +08:00
if(!file_exists($logPath)) {
return json(['code'=>-1,'msg'=>'还没有要分析的日志哦!']);
}
2022-08-02 18:48:31 +08:00
$log = file_get_contents($logPath);
$log = preg_replace('/\[info\][^\n]*compatible;/', '', $log);
2022-08-02 20:39:46 +08:00
$log = preg_replace('/\[info\][^\n]*(?=YisouSpider)/', ' ', $log);
2022-08-02 18:49:40 +08:00
switch($name) {
case 'Baiduspider':
preg_match_all('/(.*?)(?:Baiduspider)+[^\n]*\r?\n/',$log,$arr);
break;
case 'Bytespider':
preg_match_all('/(.*?)(?:Bytespider)+[^\n]*\r?\n/',$log,$arr);
break;
case 'Googlebot':
preg_match_all('/(.*?)(?:Googlebot)+[^\n]*\r?\n/',$log,$arr);
break;
case 'bingbot':
preg_match_all('/(.*?)(?:bingbot)+[^\n]*\r?\n/',$log,$arr);
break;
default:
// 正则全部蜘蛛
2022-08-02 20:39:46 +08:00
preg_match_all('/(.*?)(?:bingbot|Googlebot|Baiduspider|Bytespider|360Spider|YisouSpider|Sosospider|Sogou News Spider|SemrushBot|AhrefsBot|MJ12bot)+[^\n]*\r?\n/',$log,$arr);
2022-08-02 18:49:40 +08:00
}
2022-08-02 18:48:31 +08:00
2022-08-02 18:49:40 +08:00
// $string = '';
// foreach($arr[0] as $str) {
// $str = preg_replace('/\[(.*?)T/', '', $str);
// $str = preg_replace('/\+08:00\]/', '', $str);
// $string .= preg_replace('/\/(.*?)\)/', '', $str);
// }
2022-08-02 18:48:31 +08:00
2022-08-02 18:49:40 +08:00
// if(strlen($string)) {
// return json(['code'=>0,'msg'=>'分析成功','data'=>$string]);
// } else {
// return json(['code'=>-1,'msg'=>'还没有蜘蛛来哦']);
// }
2022-08-02 20:39:46 +08:00
$data = [];
2022-08-02 18:49:40 +08:00
$list = [];
if(count($arr[0])) {
$list['code']= 0;
$list['msg'] = '分析成功';
$list['count'] = count($arr[0]);
foreach($arr[0] as $k =>$str) {
// $str = preg_replace('/\[(.*?)T/', '', $str);
// $str = preg_replace('/\+08:00\]/', '', $str);
$str = preg_replace('/\/(.*?)\)/', '', $str);
2022-08-02 20:39:46 +08:00
// 时间
2022-08-02 18:49:40 +08:00
$ptime = "/([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})/";
preg_match($ptime, $str,$at);
$time = str_replace('T',' ',$at[0]);
//$list[$k]['time'] = $time;
// ip
2022-08-02 20:39:46 +08:00
$pip = '/((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})(\.((2(5[0-5]|[0-4]\d))|[0-1]?\d{1,2})){3}/';
2022-08-02 18:49:40 +08:00
preg_match($pip, $str,$aip);
$ip = $aip[0];
// url
$pattern="/(http|https):\/\/.*$/i";
preg_match($pattern, $str,$url);
2022-08-02 20:39:46 +08:00
// name
preg_match('/(?:bingbot|Googlebot|Baiduspider|Bytespider|360Spider|YisouSpider|Sosospider|Sogou News Spider|SemrushBot|AhrefsBot|MJ12bot)/', $str, $n);
$name = $n[0];
//$list['data'][] = ['id'=>$k + 1, 'time'=>$time, 'name'=>$name, 'ip'=>$ip, 'url'=>$url[0]];
$data[] = ['id'=>$k + 1, 'time'=>$time, 'name'=>$name, 'ip'=>$ip, 'url'=>$url[0]];
}
$datas = array_chunk($data,(int)$limit);
//$pages = count($datas);
2022-08-02 18:48:31 +08:00
2022-08-02 20:39:46 +08:00
foreach($datas as $k=>$v) {
if($page-1 == $k) {
$list['data'] = $v;
}
2022-08-02 18:49:40 +08:00
}
2022-08-02 20:39:46 +08:00
2022-08-02 18:49:40 +08:00
return json($list);
2022-08-02 18:48:31 +08:00
} else {
2022-08-02 18:49:40 +08:00
return json(['code'=>-1,'msg'=> '没有需要分析的数据']);
2022-08-02 18:48:31 +08:00
}
2022-08-02 18:49:40 +08:00
2022-08-02 18:48:31 +08:00
}
2022-08-16 15:29:04 +08:00
public function tagLinkList()
{
$arr = [];
$pushjs = new PushJscode();
$tags = $pushjs->getAllCodes(2);
if(count($tags)) {
$arr = ['code'=>0, 'msg'=>'', 'count' => count($tags)];
foreach($tags as $k=>$v) {
$arr['data'][] = ['id'=>$v['id'],'tag'=>$v['name'], 'link'=>$v['jscode'],'time'=>$v['create_time']];
}
} else {
$arr = ['code'=>-1, 'msg'=>'没有数据'];
}
return json($arr);
}
2022-04-17 17:09:19 +08:00
}