mirror of
https://github.com/ProxyPanel/ProxyPanel.git
synced 2026-04-11 23:19:05 +00:00
1. 彻底放弃向闭源版转型,清理掉一批闭源/烂尾的代码;
2. 重新优化优惠券功能的显示与生成;
2.1 现在用户使用优惠券后,购买页面会有优惠券优惠价格等提示;
2.2 支持管理自己定义优惠券代码,可是用HappyNewYear 这样的码了;
3. 管理员页面各搜索功能的BUG修改;
3.1 部分页面的搜索条件添加,以及对之前代码的烂尾进行补全;
3.2 统一代码检查用isset而非 !isEmpty, 来减少误判;
3.3 对搜索在不同设备下的显示进行了优化;
4. 针对html 和 js 项目的代码规范以及简写;
5. 针对 debug方便;添加了debug工具;
提示:请各位在生产环境下关闭debug模式;
266 lines
7.2 KiB
PHP
266 lines
7.2 KiB
PHP
<?php
|
||
|
||
// 生成SS密码
|
||
if (!function_exists('makeRandStr')) {
|
||
function makeRandStr($length = 6, $isNumbers = false)
|
||
{
|
||
// 密码字符集,可任意添加你需要的字符
|
||
if (!$isNumbers) {
|
||
$chars = 'abcdefghijkmnpqrstuvwxyzABCDEFGHJKMNPQRSTUVWXYZ23456789';
|
||
} else {
|
||
$chars = '0123456789';
|
||
}
|
||
|
||
$char = '';
|
||
for ($i = 0; $i < $length; $i++) {
|
||
$char .= $chars[mt_rand(0, strlen($chars) - 1)];
|
||
}
|
||
|
||
return $char;
|
||
}
|
||
}
|
||
|
||
// base64加密(处理URL)
|
||
if (!function_exists('base64url_encode')) {
|
||
function base64url_encode($data)
|
||
{
|
||
return strtr(base64_encode($data), ['+' => '-', '/' => '_', '=' => '']);
|
||
}
|
||
}
|
||
|
||
// base64解密(处理URL)
|
||
if (!function_exists('base64url_decode')) {
|
||
function base64url_decode($data)
|
||
{
|
||
return base64_decode(strtr($data, '-_', '+/'));
|
||
}
|
||
}
|
||
|
||
// 根据流量值自动转换单位输出
|
||
if (!function_exists('flowAutoShow')) {
|
||
function flowAutoShow($value = 0)
|
||
{
|
||
$kb = 1024;
|
||
$mb = 1048576;
|
||
$gb = 1073741824;
|
||
$tb = $gb * 1024;
|
||
$pb = $tb * 1024;
|
||
if (abs($value) >= $pb) {
|
||
return round($value / $pb, 2) . "PB";
|
||
} elseif (abs($value) >= $tb) {
|
||
return round($value / $tb, 2) . "TB";
|
||
} elseif (abs($value) >= $gb) {
|
||
return round($value / $gb, 2) . "GB";
|
||
} elseif (abs($value) >= $mb) {
|
||
return round($value / $mb, 2) . "MB";
|
||
} elseif (abs($value) >= $kb) {
|
||
return round($value / $kb, 2) . "KB";
|
||
} else {
|
||
return round($value, 2) . "B";
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!function_exists('toMB')) {
|
||
function toMB($traffic)
|
||
{
|
||
$mb = 1048576;
|
||
|
||
return $traffic * $mb;
|
||
}
|
||
}
|
||
|
||
if (!function_exists('toGB')) {
|
||
function toGB($traffic)
|
||
{
|
||
$gb = 1048576 * 1024;
|
||
|
||
return $traffic * $gb;
|
||
}
|
||
}
|
||
|
||
if (!function_exists('flowToGB')) {
|
||
function flowToGB($traffic)
|
||
{
|
||
$gb = 1048576 * 1024;
|
||
|
||
return $traffic / $gb;
|
||
}
|
||
}
|
||
|
||
// 文件大小转换
|
||
if (!function_exists('formatBytes')) {
|
||
function formatBytes($bytes, $precision = 2)
|
||
{
|
||
$units = ['B', 'KB', 'MB', 'GB', 'TB'];
|
||
$bytes = max($bytes, 0);
|
||
$pow = floor(($bytes ? log($bytes) : 0) / log(1024));
|
||
$pow = min($pow, count($units) - 1);
|
||
$bytes /= pow(1024, $pow);
|
||
|
||
return round($bytes, $precision) . ' ' . $units[$pow];
|
||
}
|
||
}
|
||
|
||
// 秒转时间
|
||
if (!function_exists('seconds2time')) {
|
||
function seconds2time($seconds)
|
||
{
|
||
$day = floor($seconds / (3600 * 24));
|
||
$hour = floor(($seconds % (3600 * 24)) / 3600);
|
||
$minute = floor((($seconds % (3600 * 24)) % 3600) / 60);
|
||
if ($day > 0) {
|
||
return $day . '天' . $hour . '小时' . $minute . '分';
|
||
} else {
|
||
if ($hour != 0) {
|
||
return $hour . '小时' . $minute . '分';
|
||
} else {
|
||
return $minute . '分';
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// 获取访客真实IP
|
||
if (!function_exists('getClientIP')) {
|
||
function getClientIP()
|
||
{
|
||
/*
|
||
* 访问时用localhost访问的,读出来的是“::1”是正常情况
|
||
* ::1说明开启了IPv6支持,这是IPv6下的本地回环地址的表示
|
||
* 使用IPv4地址访问或者关闭IPv6支持都可以不显示这个
|
||
*/
|
||
if (isset($_SERVER)) {
|
||
if (isset($_SERVER['HTTP_CF_CONNECTING_IP'])) {
|
||
$_SERVER['REMOTE_ADDR'] = $_SERVER['HTTP_CF_CONNECTING_IP'];
|
||
$ip = $_SERVER['REMOTE_ADDR'];
|
||
} elseif (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
|
||
$ip = $_SERVER['HTTP_X_FORWARDED_FOR'];
|
||
} elseif (isset($_SERVER['HTTP_CLIENT_IP'])) {
|
||
$ip = $_SERVER['HTTP_CLIENT_IP'];
|
||
} elseif (isset($_SERVER['REMOTE_ADDR'])) {
|
||
$ip = $_SERVER['REMOTE_ADDR'];
|
||
} else {
|
||
$ip = 'unknown';
|
||
}
|
||
} else {
|
||
// 绕过CDN获取真实访客IP
|
||
if (getenv('HTTP_X_FORWARDED_FOR')) {
|
||
$ip = getenv('HTTP_X_FORWARDED_FOR');
|
||
} elseif (getenv('HTTP_CLIENT_IP')) {
|
||
$ip = getenv('HTTP_CLIENT_IP');
|
||
} else {
|
||
$ip = getenv('REMOTE_ADDR');
|
||
}
|
||
}
|
||
|
||
if (trim($ip) == '::1') {
|
||
$ip = '127.0.0.1';
|
||
}
|
||
|
||
return $ip;
|
||
}
|
||
}
|
||
|
||
// 获取IPv6信息
|
||
if (!function_exists('getIPv6')) {
|
||
/*
|
||
* {
|
||
* "longitude": 105,
|
||
* "latitude": 35,
|
||
* "area_code": "0",
|
||
* "dma_code": "0",
|
||
* "organization": "AS23910 China Next Generation Internet CERNET2",
|
||
* "country": "China",
|
||
* "ip": "2001:da8:202:10::36",
|
||
* "country_code3": "CHN",
|
||
* "continent_code": "AS",
|
||
* "country_code": "CN"
|
||
* }
|
||
*
|
||
* {
|
||
* "longitude": 105,
|
||
* "latitude": 35,
|
||
* "area_code": "0",
|
||
* "dma_code": "0",
|
||
* "organization": "AS9808 Guangdong Mobile Communication Co.Ltd.",
|
||
* "country": "China",
|
||
* "ip": "2409:8a74:487:1f30:5178:e5a5:1f36:3525",
|
||
* "country_code3": "CHN",
|
||
* "continent_code": "AS",
|
||
* "country_code": "CN"
|
||
* }
|
||
*/
|
||
function getIPv6($ip)
|
||
{
|
||
$url = 'https://api.ip.sb/geoip/' . $ip;
|
||
|
||
try {
|
||
$ch = curl_init();
|
||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||
curl_setopt($ch, CURLOPT_TIMEOUT, 10);
|
||
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
|
||
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
|
||
curl_setopt($ch, CURLOPT_URL, $url);
|
||
curl_setopt($ch, CURLOPT_POST, 0);
|
||
|
||
$result = curl_exec($ch);
|
||
curl_close($ch);
|
||
|
||
$result = json_decode($result, true);
|
||
if (!is_array($result) || isset($result['code'])) {
|
||
throw new Exception('解析IPv6异常:' . $ip);
|
||
}
|
||
|
||
return $result;
|
||
} catch (\Exception $e) {
|
||
\Log::error($e->getMessage());
|
||
|
||
return [];
|
||
}
|
||
}
|
||
}
|
||
|
||
// 随机UUID
|
||
if (!function_exists('createGuid')) {
|
||
function createGuid()
|
||
{
|
||
mt_srand((double)microtime() * 10000);
|
||
$charid = strtoupper(md5(uniqid(rand(), true)));
|
||
$hyphen = chr(45);
|
||
$uuid = substr($charid, 0, 8) . $hyphen
|
||
. substr($charid, 8, 4) . $hyphen
|
||
. substr($charid, 12, 4) . $hyphen
|
||
. substr($charid, 16, 4) . $hyphen
|
||
. substr($charid, 20, 12);
|
||
|
||
return strtolower($uuid);
|
||
}
|
||
}
|
||
|
||
// 过滤emoji表情
|
||
if (!function_exists('filterEmoji')) {
|
||
function filterEmoji($str)
|
||
{
|
||
$str = preg_replace_callback('/./u',
|
||
function (array $match) {
|
||
return strlen($match[0]) >= 4 ? '' : $match[0];
|
||
},
|
||
$str);
|
||
|
||
return $str;
|
||
}
|
||
}
|
||
|
||
// 验证手机号是否正确
|
||
if (!function_exists('isMobile')) {
|
||
function isMobile($mobile)
|
||
{
|
||
if (!is_numeric($mobile)) {
|
||
return false;
|
||
}
|
||
|
||
return preg_match('#^13[\d]{9}$|^14[5,7]{1}\d{8}$|^15[^4]{1}\d{8}$|^17[0,6,7,8]{1}\d{8}$|^18[\d]{9}$#', $mobile) ? true : false;
|
||
}
|
||
}
|