-
Notifications
You must be signed in to change notification settings - Fork 31
/
Action.php
239 lines (226 loc) · 8.8 KB
/
Action.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
<?php
/**
* ShortLinks Plugin
*
* @copyright Copyright (c) 2011 DEFE (http://defe.me)
* @license GNU General Public License 2.0
*
*/
class ShortLinks_Action extends Typecho_Widget implements Widget_Interface_Do
{
private $db;
private $options;
private $notice;
public function __construct($request, $response, $params = null)
{
parent::__construct($request, $response, $params);
$this->db = Typecho_Db::get();
$this->options = self::getOptions();
$this->options->setDefault('logoUrl=' . Helper::options()->pluginUrl . '/ShortLinks/logo.png&siteCreatedYear=' . Helper::options()->plugin('ShortLinks')->siteCreatedYear . '¤tYear=' . date('Y'));
$this->notice = $this->widget('Widget_Notice');
}
/**
* 添加新的链接转换
*
*/
public function add()
{
$key = $this->request->key;
$key = $key ? $key : Typecho_Common::randString(8);
$target = $this->request->target;
if ($target === "" || $target === "http://") {
$this->notice->set(_t('请输入目标链接。'), 'error');
} //判断key是否被占用
elseif ($this->getTarget($key)) {
$this->notice->set(_t('该key已被使用,请更换key值!'), 'error');
} else {
$links = array(
'key' => $key,
'target' => $this->request->target,
'count' => 0,
);
$insertId = $this->db->query($this->db->insert('table.shortlinks')->rows($links));
if($insertId) {
$this->notice->set(_t('链接创建成功!'), 'success');
} else {
$this->notice->set(_t('链接创建失败!'), 'error');
}
}
}
/**
* 修改链接
*
*/
public function edit()
{
$target = $this->request->url;
$target = base64_decode($target); // base64解码url
$id = $this->request->id;
if (trim($target) == "" || $target == "http://") {
$this->response->throwJson('error');
} else {
if ($id) {
$this->db->query($this->db->update('table.shortlinks')->rows(array('target' => $target))
->where('id = ?', $id));
$this->response->throwJson('success');
}
}
}
/**
* 删除链接转换
*
* @param int $id
*/
public function del($id)
{
$affect = $this->db->query($this->db->delete('table.shortlinks')
->where('id = ?', $id));
if ($affect) {
$this->notice->set(_t('删除链接成功。'), 'success');
} else {
$this->notice->set(_t('链接删除失败,可能是该链接不存在。'), 'error');
}
}
/**
* 链接重定向
*
*/
public function shortlink()
{
$requestString = $this->request->key;
$siteUrl = preg_replace("/https?:\/\//", "", Typecho_Widget::widget('Widget_Options')->siteUrl);
$pOption = Typecho_Widget::widget('Widget_Options')->Plugin('ShortLinks'); // 插件选项
$referer = $this->request->getReferer();
$template = $pOption->goTemplate == 'NULL' ? null : $pOption->goTemplate;
// 允许空 referer
if (empty($referer) && $pOption->nullReferer === "1") {
$referer = $siteUrl;
}
$refererList = ShortLinks_Plugin::textareaToArr($pOption->refererList); // 允许的referer列表
$target = $this->getTarget($requestString);
// 设置nofollow属性
$this->response->setHeader('X-Robots-Tag', 'noindex, nofollow');
if ($target) {
// 自定义短链
// 增加统计
$count = $this->db->fetchObject($this->db->select('count')
->from('table.shortlinks')
->where('key = ?', $requestString))->count;
$count = $count + 1;
$this->db->query($this->db->update('table.shortlinks')
->rows(array('count' => $count))
->where('key = ?', $requestString));
} else if ($requestString === ShortLinks_Plugin::urlSafeB64Encode(ShortLinks_Plugin::urlSafeB64Decode($requestString))) {
// 自动转换链接处理
$target = ShortLinks_Plugin::urlSafeB64Decode($requestString);
$allow_redirect = false; // 默认不允许跳转
// 检查 referer
$allow_redirect = ShortLinks_Plugin::checkDomain($referer, $refererList);
if (strpos($referer, $siteUrl) !== false) {
$allow_redirect = true;
}
if (!$allow_redirect) {
// referer 非法跳转到首页
$this->response->redirect(Typecho_Widget::widget('Widget_Options')->siteUrl, 301);
exit();
}
} else {
throw new Typecho_Widget_Exception(_t('您访问的网页不存在'), 404);
}
if ($template === 'NULL' || $template === null) {
// 无跳转页面
$this->response->redirect(htmlspecialchars_decode($target), 301);
} else {
$filename = __DIR__ . DIRECTORY_SEPARATOR . 'templates' . DIRECTORY_SEPARATOR . $template . '.html';
if (PATH_SEPARATOR !== ':') {
$filename = mb_convert_encoding($filename, 'GBK', "auto");
}
$contents = file_get_contents($filename);
$html = str_replace(array('{{url}}', '{{delay}}'), array($target, $pOption->goDelay), $contents);
echo preg_replace_callback("/\{\{([_a-z0-9]+)\}\}/i",
array(__CLASS__, '__relpaceCallback'), $html);
exit();
}
}
/**
* 获取目标链接
*
* @param string $key
* @return void
*/
public function getTarget($key)
{
$target = $this->db->fetchRow($this->db->select('target')
->from('table.shortlinks')
->where(' key = ?', $key));
if (isset($target['target'])) {
return $target['target'];
} else {
return false;
}
}
/**
* 重设自定义链接
*/
public function resetLink()
{
$link = $this->request->link;
Helper::removeRoute('go');
Helper::addRoute('go', $link, 'ShortLinks_Action', 'shortlink');
$this->response->throwJson('success');
}
public function action()
{
$this->widget('Widget_User')->pass('administrator');
$this->on($this->request->is('add'))->add();
$this->on($this->request->is('edit'))->edit();
$this->on($this->request->is('del'))->del($this->request->del);
$this->on($this->request->is('resetLink'))->resetLink();
$this->response->goBack();
}
public function getOptions()
{
$values = $this->db->fetchAll($this->db->select('name', 'value')->from('table.options')->where('user = 0'));
$options = array();
foreach ($values as $value) {
if (strpos($value['name'], "plugin:") === 0) {
continue;
}
$options[$value['name']] = $value['value'];
}
/** 主题变量重载 */
if (!empty($options['theme:' . $options['theme']])) {
$themeOptions = null;
/** 解析变量 */
if ($themeOptions = unserialize($options['theme:' . $options['theme']])) {
/** 覆盖变量 */
$options = array_merge($options, $themeOptions);
}
}
$options['rootUrl'] = defined('__TYPECHO_ROOT_URL__') ? __TYPECHO_ROOT_URL__ : $this->request->getRequestRoot();
if (defined('__TYPECHO_ADMIN__')) {
/** 识别在admin目录中的情况 */
$adminDir = '/' . trim(defined('__TYPECHO_ADMIN_DIR__') ? __TYPECHO_ADMIN_DIR__ : '/admin/', '/');
$options['rootUrl'] = substr($options['rootUrl'], 0, -strlen($adminDir));
}
if (defined('__TYPECHO_SITE_URL__')) {
$options['siteUrl'] = __TYPECHO_SITE_URL__;
} else if (defined('__TYPECHO_DYNAMIC_SITE_URL__') && __TYPECHO_DYNAMIC_SITE_URL__) {
$options['siteUrl'] = $options['rootUrl'];
}
$options['originalSiteUrl'] = $options['siteUrl'];
$options['siteUrl'] = Typecho_Common::url(null, $options['siteUrl']);
return Typecho_Config::factory($options);
}
/**
* 替换回调
*
* @param Array $matches
* @return String
* @date 2020-05-01
*/
private function __relpaceCallback($matches)
{
return $this->options->{$matches[1]};
}
}