mirror of
https://github.com/kokororin/typecho-plugin-Access.git
synced 2025-01-13 13:51:03 +08:00
247 lines
11 KiB
PHP
247 lines
11 KiB
PHP
<?php
|
||
require_once __DIR__ . '/Access_Bootstrap.php';
|
||
/**
|
||
* 获取访客信息,生成统计图表,由<a href="https://zhaiyiming.com/">@一名宅</a> 部分优化重构。
|
||
*
|
||
* @package Access
|
||
* @author Kokororin
|
||
* @version 2.1.0
|
||
* @link https://github.com/kokororin/typecho-plugin-Access
|
||
*/
|
||
class Access_Plugin implements Typecho_Plugin_Interface
|
||
{
|
||
public static $panel = 'Access/page/console.php';
|
||
|
||
/**
|
||
* 激活插件方法,如果激活失败,直接抛出异常
|
||
*
|
||
* @access public
|
||
* @return string
|
||
* @throws Typecho_Plugin_Exception
|
||
*/
|
||
public static function activate()
|
||
{
|
||
$msg = Access_Plugin::install();
|
||
Helper::addPanel(1, self::$panel, _t('Access控制台'), _t('Access插件控制台'), 'subscriber');
|
||
Helper::addRoute("access_track_gif", "/access/log/track.gif", "Access_Action", 'writeLogs');
|
||
Helper::addRoute("access_migration", "/access/migration", "Access_Action", 'migration');
|
||
Helper::addRoute("access_logs", "/access/logs", "Access_Action", 'logs');
|
||
Helper::addRoute('access_statistic', '/access/statistic', 'Access_Action', 'statistic');
|
||
Typecho_Plugin::factory('Widget_Archive')->beforeRender = array('Access_Plugin', 'backend');
|
||
Typecho_Plugin::factory('Widget_Archive')->footer = array('Access_Plugin', 'frontend');
|
||
Typecho_Plugin::factory('admin/footer.php')->end = array('Access_Plugin', 'adminFooter');
|
||
return $msg ? _t($msg) : '';
|
||
}
|
||
|
||
/**
|
||
* 禁用插件方法,如果禁用失败,直接抛出异常
|
||
*
|
||
* @static
|
||
* @access public
|
||
* @return void
|
||
* @throws Typecho_Plugin_Exception
|
||
*/
|
||
public static function deactivate()
|
||
{
|
||
$config = Typecho_Widget::widget('Widget_Options')->plugin('Access');
|
||
if ($config->isDrop == 0) {
|
||
$db = Typecho_Db::get();
|
||
$adapterName = $db->getAdapterName();
|
||
if (strpos($adapterName, 'Mysql') !== false || strpos($adapterName, 'SQLite') !== false) {
|
||
$db->query("DROP TABLE `{$db->getPrefix()}access_logs`", Typecho_Db::WRITE);
|
||
} else if (strpos($adapterName, 'Pgsql') !== false) {
|
||
$db->query("DROP TABLE \"{$db->getPrefix()}access_logs\"", Typecho_Db::WRITE);
|
||
}
|
||
}
|
||
Helper::removePanel(1, self::$panel);
|
||
Helper::removeRoute("access_track_gif");
|
||
Helper::removeRoute("access_delete_logs");
|
||
Helper::removeRoute("access_statistic_view");
|
||
}
|
||
|
||
/**
|
||
* 获取插件配置面板
|
||
*
|
||
* @access public
|
||
* @param Typecho_Widget_Helper_Form $form 配置面板
|
||
* @return void
|
||
*/
|
||
public static function config(Typecho_Widget_Helper_Form $form)
|
||
{
|
||
$pageSize = new Typecho_Widget_Helper_Form_Element_Text(
|
||
'pageSize', null, '10',
|
||
'分页数量', '每页显示的日志数量');
|
||
$isDrop = new Typecho_Widget_Helper_Form_Element_Radio(
|
||
'isDrop', array(
|
||
'0' => '删除',
|
||
'1' => '不删除',
|
||
), '1', '删除数据表:', '请选择是否在禁用插件时,删除日志数据表');
|
||
$writeType = new Typecho_Widget_Helper_Form_Element_Radio(
|
||
'writeType', array(
|
||
'0' => '后端',
|
||
'1' => '前端',
|
||
), '0', '日志写入类型:', '请选择日志写入类型,如果写入速度较慢可选择前端写入日志。<br/>如果您使用了pjax,请在pjax相关事件中调用 window.Access.track() 方法。');
|
||
$blockIps = new Typecho_Widget_Helper_Form_Element_Textarea(
|
||
'blockIps', null, '',
|
||
'IP 黑名单', '每行一个,不记录来自这些 IP 的访问记录,支持 CIDR 掩码配置,支持 # 行注释');
|
||
$form->addInput($pageSize);
|
||
$form->addInput($isDrop);
|
||
$form->addInput($writeType);
|
||
$form->addInput($blockIps);
|
||
}
|
||
|
||
/**
|
||
* 个人用户的配置面板
|
||
*
|
||
* @access public
|
||
* @param Typecho_Widget_Helper_Form $form
|
||
* @return void
|
||
*/
|
||
public static function personalConfig(Typecho_Widget_Helper_Form $form)
|
||
{}
|
||
|
||
/**
|
||
* 初始化以及升级插件数据库,如初始化失败,直接抛出异常
|
||
*
|
||
* @access public
|
||
* @return string
|
||
* @throws Typecho_Plugin_Exception
|
||
*/
|
||
public static function install()
|
||
{
|
||
if (substr(trim(dirname(__FILE__), '/'), -6) != 'Access') {
|
||
throw new Typecho_Plugin_Exception(_t('插件目录名必须为Access'));
|
||
}
|
||
$db = Typecho_Db::get();
|
||
$adapterName = $db->getAdapterName();
|
||
|
||
if (strpos($adapterName, 'Mysql') !== false) {
|
||
$prefix = $db->getPrefix();
|
||
$scripts = file_get_contents('usr/plugins/Access/sql/Mysql.sql');
|
||
$scripts = str_replace('typecho_', $prefix, $scripts);
|
||
$scripts = str_replace('%charset%', 'utf8', $scripts);
|
||
$scripts = explode(';', $scripts);
|
||
try {
|
||
$configLink = '<a href="' . Helper::options()->adminUrl . 'options-plugin.php?config=Access">' . _t('前往设置') . '</a>';
|
||
# 初始化数据库如果不存在
|
||
if (!$db->fetchRow($db->query("SHOW TABLES LIKE '{$prefix}access_logs';", Typecho_Db::READ))) {
|
||
foreach ($scripts as $script) {
|
||
$script = trim($script);
|
||
if ($script) {
|
||
$db->query($script, Typecho_Db::WRITE);
|
||
}
|
||
}
|
||
$msg = _t('成功创建数据表,插件启用成功,') . $configLink;
|
||
} else {
|
||
$msg = _t('数据表已存在,插件启用成功,') . $configLink;
|
||
}
|
||
return $msg;
|
||
} catch (Typecho_Db_Exception $e) {
|
||
throw new Typecho_Plugin_Exception(_t('数据表建立失败,插件启用失败,错误信息:%s。', $e->getMessage()));
|
||
} catch (Exception $e) {
|
||
throw new Typecho_Plugin_Exception($e->getMessage());
|
||
}
|
||
} else if (strpos($adapterName, 'SQLite') !== false) {
|
||
$prefix = $db->getPrefix();
|
||
$scripts = file_get_contents('usr/plugins/Access/sql/SQLite.sql');
|
||
$scripts = str_replace('typecho_', $prefix, $scripts);
|
||
$scripts = explode(';', $scripts);
|
||
try {
|
||
$configLink = '<a href="' . Helper::options()->adminUrl . 'options-plugin.php?config=Access">' . _t('前往设置') . '</a>';
|
||
# 初始化数据库如果不存在
|
||
if (!$db->fetchRow($db->query("SELECT name FROM sqlite_master WHERE TYPE='table' AND name='{$prefix}access_logs';", Typecho_Db::READ))) {
|
||
foreach ($scripts as $script) {
|
||
$script = trim($script);
|
||
if ($script) {
|
||
$db->query($script, Typecho_Db::WRITE);
|
||
}
|
||
}
|
||
$msg = _t('成功创建数据表,插件启用成功,') . $configLink;
|
||
} else {
|
||
$msg = _t('数据表已经存在,插件启用成功,') . $configLink;
|
||
}
|
||
return $msg;
|
||
} catch (Typecho_Db_Exception $e) {
|
||
throw new Typecho_Plugin_Exception(_t('数据表建立失败,插件启用失败,错误信息:%s。', $e->getMessage()));
|
||
} catch (Exception $e) {
|
||
throw new Typecho_Plugin_Exception($e->getMessage());
|
||
}
|
||
} else if (strpos($adapterName, 'Pgsql') !== false) {
|
||
$prefix = $db->getPrefix();
|
||
$scripts = file_get_contents('usr/plugins/Access/sql/Pgsql.sql');
|
||
$scripts = str_replace('typecho_', $prefix, $scripts);
|
||
$scripts = explode(';', $scripts);
|
||
try {
|
||
$configLink = '<a href="' . Helper::options()->adminUrl . 'options-plugin.php?config=Access">' . _t('前往设置') . '</a>';
|
||
# 初始化数据库如果不存在
|
||
if (!$db->fetchRow($db->query("SELECT tablename FROM pg_tables WHERE schemaname='public' AND tablename='{$prefix}access_logs';", Typecho_Db::READ))) {
|
||
foreach ($scripts as $script) {
|
||
$script = trim($script);
|
||
if ($script) {
|
||
$db->query($script, Typecho_Db::WRITE);
|
||
}
|
||
}
|
||
$msg = _t('成功创建数据表,插件启用成功,') . $configLink;
|
||
} else {
|
||
$msg = _t('数据表已存在,插件启用成功,') . $configLink;
|
||
}
|
||
return $msg;
|
||
} catch (Typecho_Db_Exception $e) {
|
||
throw new Typecho_Plugin_Exception(_t('数据表建立失败,插件启用失败,错误信息:%s。', $e->getMessage()));
|
||
} catch (Exception $e) {
|
||
throw new Typecho_Plugin_Exception($e->getMessage());
|
||
}
|
||
} else {
|
||
throw new Typecho_Plugin_Exception(_t('你的适配器为%s,目前只支持Mysql、SQLite和Pgsql', $adapterName));
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 获取后端统计,该统计方法可以统计到一切访问
|
||
*
|
||
* @access public
|
||
* @return void
|
||
*/
|
||
public static function backend($archive)
|
||
{
|
||
$access = new Access_Core();
|
||
$config = Typecho_Widget::widget('Widget_Options')->plugin('Access');
|
||
|
||
if ($config->writeType == 0) {
|
||
$access->writeLogs($archive);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 获取前端统计,该方法要求客户端必须渲染网页,所以不能统计RSS等直接抓取PHP页面的方式
|
||
*
|
||
* @access public
|
||
* @return void
|
||
*/
|
||
public static function frontend($archive)
|
||
{
|
||
$config = Typecho_Widget::widget('Widget_Options')->plugin('Access');
|
||
if ($config->writeType == 1) {
|
||
$index = rtrim(Helper::options()->index, '/');
|
||
$access = new Access_Core();
|
||
$parsedArchive = $access->parseArchive($archive);
|
||
echo "<script type=\"text/javascript\">(function(w){var t=function(){var i=new Image();i.src='{$index}/access/log/track.gif?u='+location.pathname+location.search+location.hash+'&cid={$parsedArchive['content_id']}&mid={$parsedArchive['meta_id']}&rand='+new Date().getTime()};t();var a={};a.track=t;w.Access=a})(this);</script>";
|
||
}
|
||
}
|
||
|
||
public static function adminFooter()
|
||
{
|
||
$url = $_SERVER['PHP_SELF'];
|
||
$filename = substr($url, strrpos($url, '/') + 1);
|
||
if ($filename == 'index.php') {
|
||
echo '<script>
|
||
$(document).ready(function() {
|
||
$("#start-link").append("<li><a href=\"';
|
||
Helper::options()->adminUrl('extending.php?panel=' . self::$panel);
|
||
echo '\">' . _t('Access控制台') . '</a></li>");
|
||
});
|
||
</script>';
|
||
}
|
||
}
|
||
}
|