perf: 增加原生缓存
This commit is contained in:
33
app/common/bootstrap/RedisCache.php
Normal file
33
app/common/bootstrap/RedisCache.php
Normal file
@@ -0,0 +1,33 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\bootstrap;
|
||||
|
||||
use support\Cache;
|
||||
use support\Log;
|
||||
use think\Container;
|
||||
use think\DbManager;
|
||||
use Webman\Bootstrap;
|
||||
use Workerman\Timer;
|
||||
use Workerman\Worker;
|
||||
|
||||
class RedisCache implements Bootstrap
|
||||
{
|
||||
public static function start(?Worker $worker)
|
||||
{
|
||||
// TODO: Implement start() method.
|
||||
if ($worker) {
|
||||
try {
|
||||
Timer::add(55, function () {
|
||||
Cache::get('ping');
|
||||
});
|
||||
} catch (\Throwable $e) {
|
||||
Log::error('RedisCache error: ' . $e->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
if (class_exists(DbManager::class)) {
|
||||
$manager_instance = Container::getInstance()->make(DbManager::class);
|
||||
$manager_instance->setCache(Cache::instance());
|
||||
}
|
||||
}
|
||||
}
|
||||
226
app/common/driver/notice/EmailDriver.php
Normal file
226
app/common/driver/notice/EmailDriver.php
Normal file
@@ -0,0 +1,226 @@
|
||||
<?php
|
||||
declare(strict_types=1);
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin 极速开发框架 [基于WebMan开发]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2020-2030 http://www.swiftadmin.net
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin.net High Speed Development Framework
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: meystack <coolsec@foxmail.com> Apache 2.0 License
|
||||
// +----------------------------------------------------------------------
|
||||
namespace app\common\driver\notice;
|
||||
|
||||
use PHPMailer\PHPMailer\Exception;
|
||||
use PHPMailer\PHPMailer\PHPMailer;
|
||||
use Psr\SimpleCache\InvalidArgumentException;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
|
||||
class EmailDriver
|
||||
{
|
||||
/**
|
||||
* @PHPMailer 对象实例
|
||||
*/
|
||||
public object $mail;
|
||||
|
||||
//默认配置
|
||||
protected array $config = [
|
||||
'smtp_debug' => false, // 是否调试
|
||||
'smtp_host' => 'smtp.163.com', // 服务器地址
|
||||
'smtp_port' => 587, // 服务器端口
|
||||
'smtp_user' => 'yourname@163.com', // 邮件用户名
|
||||
'smtp_pass' => '****', // 邮件密码
|
||||
'smtp_name' => '管理员', // 发送邮件显示
|
||||
];
|
||||
|
||||
/**
|
||||
* 类构造函数
|
||||
* class constructor.
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws Exception
|
||||
* @throws ModelNotFoundException
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
// 此配置项为数组
|
||||
$email = saenv('email');
|
||||
$this->config = array_merge($this->config, $email);
|
||||
|
||||
// 创建PHPMailer对象实例
|
||||
$this->mail = new PHPMailer();
|
||||
$this->mail->CharSet = 'UTF-8';
|
||||
$this->mail->IsSMTP();
|
||||
|
||||
/**
|
||||
* 是否开启调试模式
|
||||
*/
|
||||
$this->mail->SMTPDebug = $this->config['smtp_debug'];
|
||||
$this->mail->SMTPAuth = true;
|
||||
$this->mail->SMTPSecure = 'ssl';
|
||||
$this->mail->SMTPOptions = array(
|
||||
'ssl' => array(
|
||||
'verify_peer' => false,
|
||||
'verify_peer_name' => false,
|
||||
'allow_self_signed' => true
|
||||
)
|
||||
);
|
||||
|
||||
$this->mail->Host = $this->config['smtp_host'];
|
||||
$this->mail->Port = $this->config['smtp_port'];
|
||||
$this->mail->Username = $this->config['smtp_user'];
|
||||
$this->mail->Password = trim($this->config['smtp_pass']);
|
||||
$this->mail->SetFrom($this->config['smtp_user'], $this->config['smtp_name']);
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置邮件主题
|
||||
* @param string $subject 邮件主题
|
||||
* @return $this
|
||||
*/
|
||||
public function Subject(string $subject): EmailDriver
|
||||
{
|
||||
$this->mail->Subject = $subject;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置发件人
|
||||
* @param string $email 发件人邮箱
|
||||
* @param string $name 发件人名称
|
||||
* @return $this
|
||||
* @throws Exception
|
||||
*/
|
||||
public function from(string $email, string $name = ''): EmailDriver
|
||||
{
|
||||
$this->mail->setFrom($email, $name);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置邮件内容
|
||||
* @param $MsgHtml
|
||||
* @param boolean $isHtml 是否HTML格式
|
||||
* @return $this
|
||||
* @throws Exception
|
||||
*/
|
||||
public function MsgHTML($MsgHtml, bool $isHtml = true): EmailDriver
|
||||
{
|
||||
if ($isHtml) {
|
||||
$this->mail->msgHTML($MsgHtml);
|
||||
} else {
|
||||
$this->mail->Body = $MsgHtml;
|
||||
}
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置收件人
|
||||
* @param $email
|
||||
* @return $this
|
||||
* @throws Exception
|
||||
*/
|
||||
public function address($email): EmailDriver
|
||||
{
|
||||
$list = $this->buildAddress($email);
|
||||
foreach ($list as $address => $name) {
|
||||
$this->mail->addAddress($address, $name);
|
||||
}
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 构建Email地址
|
||||
* @param $emails
|
||||
* @return array
|
||||
*/
|
||||
protected function buildAddress($emails): array
|
||||
{
|
||||
if (!is_array($emails)) {
|
||||
$emails = explode(',', str_replace(";", ",", $emails));
|
||||
}
|
||||
|
||||
foreach ($emails as $key => $value) {
|
||||
$list[is_numeric($key) ? $value : $key] = is_numeric($key) ? "" : $value;
|
||||
}
|
||||
return $list ?? [];
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置抄送
|
||||
* @param string $address
|
||||
* @param string $name
|
||||
* @return EmailDriver
|
||||
* @throws @\PHPMailer\PHPMailer\Exception
|
||||
*/
|
||||
public function addCC(string $address, string $name = ''): EmailDriver
|
||||
{
|
||||
$this->mail->addCC($address, $name);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置密送
|
||||
* @param string $address
|
||||
* @param string $name
|
||||
* @return EmailDriver
|
||||
* @throws @\PHPMailer\PHPMailer\Exception
|
||||
*/
|
||||
public function addBCC(string $address, string $name = ''): EmailDriver
|
||||
{
|
||||
$this->mail->addBCC($address, $name);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加附件
|
||||
* @param string $path 附件路径
|
||||
* @param string $name 附件名称
|
||||
* @throws Exception
|
||||
*/
|
||||
public function attachment(string $path, string $name = ''): EmailDriver
|
||||
{
|
||||
if (is_file($path)) {
|
||||
$this->mail->addAttachment($path, $name);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 发送邮件
|
||||
* @return boolean
|
||||
* @throws Exception
|
||||
*/
|
||||
public function send(): bool
|
||||
{
|
||||
if (!$this->mail->send()) {
|
||||
throw new Exception($this->mail->ErrorInfo);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 测试发送邮件
|
||||
* @param array $config
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
public function testEmail(array $config = []): bool
|
||||
{
|
||||
$this->config = array_merge($this->config, $config);
|
||||
$this->mail->Host = $this->config['smtp_host'];
|
||||
$this->mail->Port = $this->config['smtp_port'];
|
||||
$this->mail->Username = $this->config['smtp_user'];
|
||||
$this->mail->Password = trim($this->config['smtp_pass']);
|
||||
$this->mail->SetFrom($this->config['smtp_user'], $this->config['smtp_name']);
|
||||
if (!$this->address($config['smtp_test'])->Subject("测试邮件")->MsgHTML("如果您看到这封邮件,说明测试成功了!")->send()) {
|
||||
throw new Exception($this->mail->ErrorInfo);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
25
app/common/exception/DumpException.php
Normal file
25
app/common/exception/DumpException.php
Normal file
@@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\exception;
|
||||
|
||||
/**
|
||||
* 全局操作异常类
|
||||
* Class OperateException
|
||||
* @package app\common\exception
|
||||
*/
|
||||
class DumpException extends \Exception
|
||||
{
|
||||
/**
|
||||
* 附加数据
|
||||
* @var array
|
||||
*/
|
||||
public array $data = [];
|
||||
|
||||
public function __construct($message = '', $code = 0, array $data = [], \Throwable $previous = null)
|
||||
{
|
||||
$this->data = $data;
|
||||
$this->code = $code;
|
||||
$this->message = $message;
|
||||
parent::__construct($this->message, $this->code, $previous);
|
||||
}
|
||||
}
|
||||
27
app/common/exception/OperateException.php
Normal file
27
app/common/exception/OperateException.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\exception;
|
||||
|
||||
use app\common\library\ResultCode;
|
||||
|
||||
/**
|
||||
* 全局操作异常类
|
||||
* Class OperateException
|
||||
* @package app\common\exception
|
||||
*/
|
||||
class OperateException extends \Exception
|
||||
{
|
||||
/**
|
||||
* 附加数据
|
||||
* @var array
|
||||
*/
|
||||
public array $data = [];
|
||||
|
||||
public function __construct($message = '', $code = 0, array $data = [], \Throwable $previous = null)
|
||||
{
|
||||
$this->data = $data;
|
||||
$this->code = $code ?: ResultCode::UNKNOWN['code'];
|
||||
$this->message = $message ?: ResultCode::UNKNOWN['msg'];
|
||||
parent::__construct($this->message, $this->code, $previous);
|
||||
}
|
||||
}
|
||||
25
app/common/exception/user/UserException.php
Normal file
25
app/common/exception/user/UserException.php
Normal file
@@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\exception\user;
|
||||
use app\common\library\ResultCode;
|
||||
|
||||
/**
|
||||
* 用户异常类
|
||||
* Class UserException
|
||||
*/
|
||||
class UserException extends \Exception
|
||||
{
|
||||
/**
|
||||
* 附加数据
|
||||
* @var array
|
||||
*/
|
||||
public array $data = [];
|
||||
|
||||
public function __construct($message = '', $code = 0, array $data = [], \Throwable $previous = null)
|
||||
{
|
||||
$this->data = $data;
|
||||
$this->code = $code ?: ResultCode::UNKNOWN['code'];
|
||||
$this->message = $message ?: ResultCode::UNKNOWN['msg'];
|
||||
parent::__construct($this->message, $this->code, $previous);
|
||||
}
|
||||
}
|
||||
14
app/common/logic/user/UserLogic.php
Normal file
14
app/common/logic/user/UserLogic.php
Normal file
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\logic\user;
|
||||
|
||||
use app\common\service\BaseServiceLogic;
|
||||
|
||||
/**
|
||||
* 用户逻辑层
|
||||
* Class UserLogic
|
||||
*/
|
||||
class UserLogic extends BaseServiceLogic
|
||||
{
|
||||
|
||||
}
|
||||
89
app/common/service/BaseServiceLogic.php
Normal file
89
app/common/service/BaseServiceLogic.php
Normal file
@@ -0,0 +1,89 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\service;
|
||||
|
||||
/**
|
||||
* 基础服务类
|
||||
* @package app\common\service
|
||||
*/
|
||||
class BaseServiceLogic
|
||||
{
|
||||
/**
|
||||
* 返回状态码
|
||||
* @var int
|
||||
*/
|
||||
protected static int $code = 0;
|
||||
|
||||
/**
|
||||
* 错误信息
|
||||
* @var string
|
||||
*/
|
||||
protected static string $error;
|
||||
|
||||
/**
|
||||
* 返回数据
|
||||
* @var array
|
||||
*/
|
||||
protected static array $data;
|
||||
|
||||
/**
|
||||
* 设置返回状态码
|
||||
* @param int $code
|
||||
* @return void
|
||||
*/
|
||||
public static function setCode(int $code): void
|
||||
{
|
||||
self::$code = $code;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取返回状态码
|
||||
* @return int
|
||||
* @return void
|
||||
*/
|
||||
public static function getCode(): int
|
||||
{
|
||||
return self::$code;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置错误信息
|
||||
* @param string $error
|
||||
* @return void
|
||||
*/
|
||||
public static function setError(string $error): void
|
||||
{
|
||||
self::$error = $error;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取错误信息
|
||||
* @return string
|
||||
* @return void
|
||||
*/
|
||||
public static function getError(): string
|
||||
{
|
||||
return self::$error;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置返回数据
|
||||
* @param array $data
|
||||
* @return void
|
||||
*/
|
||||
public static function setData(array $data): void
|
||||
{
|
||||
self::$data = $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取返回数据
|
||||
* @return array
|
||||
* @return void
|
||||
*/
|
||||
public static function getData(): array
|
||||
{
|
||||
return self::$data;
|
||||
}
|
||||
|
||||
}
|
||||
171
app/common/service/notice/EmailService.php
Normal file
171
app/common/service/notice/EmailService.php
Normal file
@@ -0,0 +1,171 @@
|
||||
<?php
|
||||
declare(strict_types=1);
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin 极速开发框架 [基于WebMan开发]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2020-2030 http://www.swiftadmin.net
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin.net High Speed Development Framework
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: meystack <coolsec@foxmail.com> Apache 2.0 License
|
||||
// +----------------------------------------------------------------------
|
||||
namespace app\common\service\notice;
|
||||
|
||||
use app\common\driver\notice\EmailDriver;
|
||||
use app\common\exception\OperateException;
|
||||
use app\common\model\system\User;
|
||||
use app\common\model\system\UserValidate;
|
||||
use PHPMailer\PHPMailer\Exception;
|
||||
use Psr\SimpleCache\InvalidArgumentException;
|
||||
use support\App;
|
||||
use system\Random;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
|
||||
class EmailService
|
||||
{
|
||||
/**
|
||||
* 发送间隔时间
|
||||
* @var int
|
||||
*/
|
||||
const EMAIL_SEND_INTERVAL = 60;
|
||||
|
||||
/**
|
||||
* 验证码过期时间
|
||||
* @var int
|
||||
*/
|
||||
const EXPIRE_TIME = 5; //验证码过期时间(分钟)
|
||||
|
||||
/**
|
||||
* 发送邮件
|
||||
* @param $email
|
||||
* @param $title
|
||||
* @param $content
|
||||
* @return bool
|
||||
* @throws \Exception
|
||||
*/
|
||||
public static function send($email, $title, $content): bool
|
||||
{
|
||||
$eDriver = new EmailDriver();
|
||||
try {
|
||||
$eDriver->address($email)->Subject($title)->MsgHTML($content)->send();
|
||||
} catch (\PHPMailer\PHPMailer\Exception $e) {
|
||||
throw new \Exception($e->getMessage());
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 发送验证码
|
||||
* @param $email
|
||||
* @param $event
|
||||
* @return bool
|
||||
* @throws OperateException
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public static function captcha($email, $event): bool
|
||||
{
|
||||
$result = self::getLastMsg($email, $event);
|
||||
if (!empty($result) && time() - strtotime($result['create_time']) < self::EMAIL_SEND_INTERVAL) {
|
||||
throw new OperateException(__('发送频繁'));
|
||||
}
|
||||
|
||||
$userinfo = (new User())->where('email', $email)->findOrEmpty()->toArray();
|
||||
if (in_array($event, ['register', 'change']) && $userinfo) {
|
||||
throw new OperateException(__('当前邮箱已被注册'));
|
||||
} else if ($event == 'forgot' && !$userinfo) {
|
||||
throw new OperateException(__('当前邮箱不存在'));
|
||||
}
|
||||
|
||||
$captcha = Random::number();
|
||||
$filePath = root_path() . 'extend/conf/tpl/captcha.tpl';
|
||||
if (!is_file($filePath)) {
|
||||
throw new OperateException(__('验证码模板不存在'));
|
||||
}
|
||||
|
||||
$eDriver = new EmailDriver();
|
||||
$data = [$captcha, saenv('site_name'), date('Y-m-d H:i:s')];
|
||||
$content = str_replace(['{code}', '{site_name}', '{time}'], $data, read_file($filePath));
|
||||
try {
|
||||
|
||||
// 发送邮件
|
||||
$eDriver->address($email)->Subject("验证码")->MsgHTML($content)->send();
|
||||
// 保存验证码
|
||||
(new UserValidate())->create([
|
||||
'code' => $captcha,
|
||||
'event' => $event,
|
||||
'email' => $email,
|
||||
'status' => 1,
|
||||
]);
|
||||
} catch (\Throwable $th) {
|
||||
throw new OperateException(__('验证码发送失败'));
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 校验验证码
|
||||
* @param $email
|
||||
* @param $code
|
||||
* @param $event
|
||||
* @return bool
|
||||
* @throws \Exception
|
||||
*/
|
||||
public static function checkCaptcha($email, $code, $event): bool
|
||||
{
|
||||
$model = new UserValidate();
|
||||
$result = $model->where([
|
||||
['event', '=', $event],
|
||||
['email', '=', $email],
|
||||
['code', '=', $code],
|
||||
['status', '=', 1],
|
||||
])->order("id", "desc")->findOrEmpty()->toArray();
|
||||
|
||||
if (!empty($result)) {
|
||||
$model->where('id', $result['id'])->update(['status' => 0]);
|
||||
$expires = time() - strtotime($result['create_time']);
|
||||
if ($expires <= self::EXPIRE_TIME * 60) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取最后一条
|
||||
* @param string $email
|
||||
* @param string $event
|
||||
* @return array
|
||||
*/
|
||||
public static function getLastMsg(string $email, string $event): array
|
||||
{
|
||||
return (new UserValidate())->where([
|
||||
['email', '=', $email],
|
||||
['event', '=', $event],
|
||||
])->order('id', 'desc')->findOrEmpty()->toArray();
|
||||
}
|
||||
|
||||
/**
|
||||
* 过滤邮箱格式
|
||||
* @param string $email
|
||||
* @return string
|
||||
*/
|
||||
public static function filterEmail(string $email): string
|
||||
{
|
||||
return filter_var($email, FILTER_SANITIZE_EMAIL);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $params
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function testEmail(array $params = []): bool
|
||||
{
|
||||
$eDriver = new EmailDriver();
|
||||
return $eDriver->testEmail($params);
|
||||
}
|
||||
}
|
||||
159
app/common/service/notice/SmsService.php
Normal file
159
app/common/service/notice/SmsService.php
Normal file
@@ -0,0 +1,159 @@
|
||||
<?php
|
||||
declare (strict_types=1);
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin 极速开发框架 [基于WebMan开发]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2020-2030 http://www.swiftadmin.net
|
||||
// +----------------------------------------------------------------------
|
||||
// | swiftAdmin.net High Speed Development Framework
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: meystack <coolsec@foxmail.com> Apache 2.0 License
|
||||
// +----------------------------------------------------------------------
|
||||
namespace app\common\service\notice;
|
||||
|
||||
use app\common\exception\OperateException;
|
||||
use app\common\model\system\User;
|
||||
use app\common\model\system\UserValidate;
|
||||
use Psr\SimpleCache\InvalidArgumentException;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
use Webman\Event\Event;
|
||||
|
||||
class SmsService
|
||||
{
|
||||
/**
|
||||
* 发送间隔时间
|
||||
* @var int
|
||||
*/
|
||||
const EMAIL_SEND_INTERVAL = 60;
|
||||
|
||||
/**
|
||||
* 验证码过期时间
|
||||
* @var int
|
||||
*/
|
||||
const EXPIRE_TIME = 5; //验证码过期时间(分钟)
|
||||
|
||||
/**
|
||||
* 类构造函数
|
||||
* class constructor
|
||||
* @access public
|
||||
*/
|
||||
public function __construct()
|
||||
{}
|
||||
|
||||
/**
|
||||
* 发送短信
|
||||
* @param $mobile
|
||||
* @param $event
|
||||
* @return bool
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
* @throws \Exception|\Psr\SimpleCache\InvalidArgumentException
|
||||
*/
|
||||
public static function send($mobile, $event): bool
|
||||
{
|
||||
$result = self::getLastMsg($mobile, $event);
|
||||
if (!empty($result)
|
||||
&& time() - strtotime($result['create_time']) < self::EMAIL_SEND_INTERVAL) {
|
||||
throw new OperateException(__('发送频繁'));
|
||||
}
|
||||
|
||||
$userinfo = (new User())->where('mobile', $mobile)->findOrEmpty()->toArray();
|
||||
if (in_array($event, ['register', 'change']) && $userinfo) {
|
||||
throw new OperateException(__('当前手机号已注册'));
|
||||
} else if ($event == 'forgot' && !$userinfo) {
|
||||
throw new OperateException(__('当前手机号未注册'));
|
||||
}
|
||||
|
||||
if (!Event::hasListener('smsMsgSend')) {
|
||||
throw new OperateException(__('短信插件未安装'));
|
||||
}
|
||||
|
||||
list($smsType, $config) = self::getSmsConfig();
|
||||
$smsConf = include(root_path() . "extend/conf/sms/sms.php");
|
||||
if (!isset($smsConf[$smsType][$event]['template'])) {
|
||||
throw new OperateException(__('短信模板错误'));
|
||||
}
|
||||
|
||||
$response = Event::emit('smsMsgSend', [
|
||||
'mobile' => $mobile,
|
||||
'event' => $event,
|
||||
'template' => $smsConf[$smsType][$event]['template'],
|
||||
],true);
|
||||
|
||||
if (isset($response['error']) && $response['error']) {
|
||||
throw new \Exception($response['error']);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 校验验证码
|
||||
* @param $mobile
|
||||
* @param $captcha
|
||||
* @param $event
|
||||
* @return bool
|
||||
* @throws DbException
|
||||
*/
|
||||
public static function checkCaptcha($mobile, $captcha, $event): bool
|
||||
{
|
||||
$model = new UserValidate();
|
||||
$result = $model->where([
|
||||
['event', '=', $event],
|
||||
['mobile', '=', $mobile],
|
||||
['code', '=', $captcha],
|
||||
['status', '=', 1],
|
||||
])->order("id", "desc")->findOrEmpty()->toArray();
|
||||
|
||||
if (!empty($result)) {
|
||||
$model->where('id', $result['id'])->update(['status' => 0]);
|
||||
$expires = time() - strtotime($result['create_time']);
|
||||
if ($expires <= self::EXPIRE_TIME * 60) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取最后一条
|
||||
* @param string $mobile
|
||||
* @param string $event
|
||||
* @return array
|
||||
*/
|
||||
public static function getLastMsg(string $mobile, string $event): array
|
||||
{
|
||||
$mobile = str_replace(['+86', '-', ' ', '.'], '', $mobile);
|
||||
return (new UserValidate())->where([
|
||||
['mobile', '=', $mobile],
|
||||
['event', '=', $event],
|
||||
])->order('id', 'desc')->findOrEmpty()->toArray();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 校验手机号
|
||||
* @param $mobile
|
||||
* @return bool
|
||||
*/
|
||||
public static function filterMobile($mobile): bool
|
||||
{
|
||||
$pattern = '/^((13[0-9])|(14[5,7,9])|(15[^4])|(18[0-9])|(17[0,1,3,5,6,7,8]))\d{8}$/';
|
||||
return (bool)preg_match($pattern, $mobile);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取配置信息
|
||||
* @return array
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
protected static function getSmsConfig(): array
|
||||
{
|
||||
$smsType = saenv('smstype');
|
||||
$config = saenv($smsType) ?? [];
|
||||
return ['type' => $smsType, 'config' => $config];
|
||||
}
|
||||
}
|
||||
494
app/common/service/user/UserService.php
Normal file
494
app/common/service/user/UserService.php
Normal file
@@ -0,0 +1,494 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\service\user;
|
||||
|
||||
use app\common\exception\OperateException;
|
||||
use app\common\exception\user\UserException;
|
||||
use app\common\model\system\User as UserModel;
|
||||
use app\common\model\system\UserLog;
|
||||
use app\common\model\system\UserNotice;
|
||||
use app\common\service\notice\EmailService;
|
||||
use app\common\service\notice\SmsService;
|
||||
use PHPMailer\PHPMailer\Exception;
|
||||
use Psr\SimpleCache\InvalidArgumentException;
|
||||
use system\Random;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
use support\Cache;
|
||||
use Webman\Event\Event;
|
||||
|
||||
/**
|
||||
* 用户中心服务
|
||||
* @package app\common\service\system
|
||||
*/
|
||||
class UserService
|
||||
{
|
||||
/**
|
||||
* 保活时间
|
||||
* @var int
|
||||
*/
|
||||
protected static int $keepTime = 604800;
|
||||
|
||||
/**
|
||||
* 类构造函数
|
||||
* class constructor.
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
self::$keepTime = config('cookie.expire');
|
||||
}
|
||||
|
||||
/**
|
||||
* 注册服务
|
||||
* @param array $post
|
||||
* @return array
|
||||
* @throws DbException
|
||||
* @throws InvalidArgumentException
|
||||
* @throws OperateException
|
||||
*/
|
||||
public static function register(array $post): array
|
||||
{
|
||||
if (!saenv('user_status')) {
|
||||
throw new OperateException('暂未开放注册!');
|
||||
}
|
||||
|
||||
// 是否手机注册
|
||||
$regType = saenv('user_register_style');
|
||||
if ($regType == 'mobile') {
|
||||
$mobile = $post['mobile'] ?? '';
|
||||
$captcha = $post['captcha'] ?? '';
|
||||
if (!SmsService::checkCaptcha($mobile, $captcha, 'register')) {
|
||||
throw new OperateException('验证码错误');
|
||||
}
|
||||
}
|
||||
|
||||
// 禁止批量注册
|
||||
$where[] = ['create_ip', '=', request()->getRealIp()];
|
||||
$where[] = ['create_time', '>', linux_time(1)];
|
||||
$totalMax = UserModel::where($where)->count();
|
||||
if ($totalMax >= saenv('user_register_second')) {
|
||||
throw new OperateException('禁止批量注册');
|
||||
}
|
||||
|
||||
try {
|
||||
// 加密盐值
|
||||
$salt = Random::alpha();
|
||||
$userInfo = [
|
||||
'nickname' => $post['nickname'],
|
||||
'email' => $post['email'] ?? '',
|
||||
'mobile' => $post['mobile'] ?? '',
|
||||
'salt' => $salt,
|
||||
'pwd' => encryptPwd($post['pwd'], $salt),
|
||||
'invite_id' => input('inviter', request()->cookie('inviter')),
|
||||
];
|
||||
$userInfo = self::createUser($userInfo);
|
||||
} catch (\Throwable $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return self::createUserCookies($userInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
* 登录服务
|
||||
* @param string $nickname
|
||||
* @param string $pwd
|
||||
* @return array
|
||||
* @throws InvalidArgumentException|OperateException
|
||||
*/
|
||||
public static function accountLogin(string $nickname, string $pwd): array
|
||||
{
|
||||
// 普通登录支持邮箱和用户名登录
|
||||
if (filter_var($nickname, FILTER_VALIDATE_EMAIL)) {
|
||||
$where[] = ['email', '=', htmlspecialchars(trim($nickname))];
|
||||
} else {
|
||||
$where[] = ['nickname', '=', htmlspecialchars(trim($nickname))];
|
||||
}
|
||||
|
||||
$userInfo = UserModel::where($where)->findOrEmpty()->toArray();
|
||||
if (empty($userInfo)) {
|
||||
throw new OperateException('用户名或密码错误');
|
||||
}
|
||||
|
||||
$uPwd = encryptPwd($pwd, $userInfo['salt']);
|
||||
if ($userInfo['pwd'] != $uPwd) {
|
||||
$errorMsg = '用户名或密码错误';
|
||||
UserLog::write($errorMsg, $userInfo['nickname'], $userInfo['id']);
|
||||
throw new OperateException($errorMsg);
|
||||
}
|
||||
|
||||
if (!$userInfo['status']) {
|
||||
$errorMsg = '用户禁用或未审核,请联系管理员';
|
||||
UserLog::write($errorMsg, $userInfo['nickname'], $userInfo['id']);
|
||||
throw new OperateException($errorMsg);
|
||||
}
|
||||
|
||||
return self::createUserCookies($userInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
* 手机登录
|
||||
* @param string $mobile
|
||||
* @param string $captcha
|
||||
* @return array
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws InvalidArgumentException
|
||||
* @throws ModelNotFoundException|OperateException
|
||||
*/
|
||||
public static function mobileLogin(string $mobile, string $captcha): array
|
||||
{
|
||||
if (!SmsService::checkCaptcha($mobile, $captcha, 'login')) {
|
||||
throw new OperateException('验证码错误');
|
||||
}
|
||||
|
||||
$userInfo = UserModel::where(['mobile' => $mobile])->findOrEmpty()->toArray();
|
||||
if (empty($userInfo)) {
|
||||
$maxId = UserModel::max('id');
|
||||
$userInfo = [
|
||||
'nickname' => 'u' . ($maxId + 1),
|
||||
'mobile' => $mobile,
|
||||
];
|
||||
return self::createUser($userInfo);
|
||||
} else if (!$userInfo['status']) {
|
||||
$errorMsg = '用户禁用或未审核,请联系管理员';
|
||||
UserLog::write($errorMsg, $userInfo['nickname'], $userInfo['id']);
|
||||
throw new OperateException($errorMsg);
|
||||
}
|
||||
self::updateUser($userInfo);
|
||||
return self::createUserCookies($userInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建用户
|
||||
* @param array $userInfo
|
||||
* @return array
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException|OperateException
|
||||
*/
|
||||
public static function createUser(array $userInfo): array
|
||||
{
|
||||
if (isset($userInfo['nickname']) && UserModel::getByNickname($userInfo['nickname'])) {
|
||||
throw new OperateException('当前用户名已被占用!');
|
||||
}
|
||||
|
||||
if (isset($userInfo['email']) && UserModel::getByEmail($userInfo['email'])) {
|
||||
throw new OperateException('当前邮箱已被占用!');
|
||||
}
|
||||
|
||||
if (isset($userInfo['mobile']) && UserModel::getByMobile($userInfo['mobile'])) {
|
||||
throw new OperateException('当前手机号已被占用!');
|
||||
}
|
||||
|
||||
if (isset($userInfo['pwd']) && $userInfo['pwd']) {
|
||||
$userInfo['salt'] = Random::alpha();
|
||||
$userInfo['pwd'] = encryptPwd($userInfo['pwd'], $userInfo['salt']);
|
||||
}
|
||||
|
||||
$userInfo['login_time'] = time();
|
||||
$userInfo['login_ip'] = request()->getRealIp();
|
||||
$userInfo['create_ip'] = request()->getRealIp();
|
||||
|
||||
try {
|
||||
$userInfo = UserModel::create($userInfo)->toArray();
|
||||
} catch (\Throwable $th) {
|
||||
throw new OperateException($th->getMessage());
|
||||
}
|
||||
|
||||
return $userInfo;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* 返回前端令牌
|
||||
* @param array $userInfo
|
||||
* @return array
|
||||
*/
|
||||
public static function createUserCookies(array $userInfo = []): array
|
||||
{
|
||||
$userToken = UserTokenService::buildToken($userInfo['id']);
|
||||
$response = response()
|
||||
->cookie('uid', $userInfo['id'], self::$keepTime, '/')
|
||||
->cookie('token', $userToken, self::$keepTime, '/')
|
||||
->cookie('nickname', $userInfo['nickname'], self::$keepTime, '/');
|
||||
Cache::set($userToken, $userInfo['id'], self::$keepTime);
|
||||
Cache::set('user_info_' . $userInfo['id'], $userInfo, self::$keepTime);
|
||||
\Webman\Event\Event::emit("userLoginSuccess", $userInfo);
|
||||
return ['token' => $userToken, 'id' => $userInfo['id'], 'response' => $response];
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新用户信息
|
||||
* @param array $userInfo
|
||||
*/
|
||||
public static function updateUser(array $userInfo = []): void
|
||||
{
|
||||
$data['login_time'] = time();
|
||||
$data['login_ip'] = request()->getRealIp();
|
||||
$data['login_count'] = $userInfo['login_count'] + 1;
|
||||
$data['update_time'] = time();
|
||||
UserModel::update($data, ['id' => $userInfo['id']]);
|
||||
UserLog::write('登录成功', $userInfo['nickname'], $userInfo['id'], 1);
|
||||
Event::emit('userLoginSuccess', $userInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
* 退出登录
|
||||
* @return void
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public static function logout(): void
|
||||
{
|
||||
response()->cookie('uid', null);
|
||||
response()->cookie('token', null);
|
||||
response()->cookie('nickname', null);
|
||||
Cache::delete(UserTokenService::getToken());
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $params
|
||||
* @param int $userId
|
||||
* @return bool
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
* @throws OperateException
|
||||
*/
|
||||
public static function editProfile(array $params = [], int $userId = 0): bool
|
||||
{
|
||||
$userInfo = UserModel::where('id', $userId)->findOrEmpty()->toArray();
|
||||
if (empty($userInfo)) {
|
||||
throw new OperateException('用户不存在');
|
||||
}
|
||||
|
||||
$data = [];
|
||||
if (isset($params['nickname']) && $params['nickname']) {
|
||||
// 验证昵称是否存在
|
||||
$data['nickname'] = $params['nickname'];
|
||||
if ($data['nickname'] != $userInfo['nickname']
|
||||
&& UserModel::where('nickname', $data['nickname'])->find()) {
|
||||
throw new OperateException('昵称已存在');
|
||||
}
|
||||
}
|
||||
|
||||
$fields = ['avatar', 'name', 'wechat', 'qq', 'idCard', 'address', 'gender'];
|
||||
foreach ($fields as $field) {
|
||||
if (isset($params[$field])) {
|
||||
$data[$field] = $params[$field];
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
UserModel::update($data, ['id' => $userId]);
|
||||
} catch (\Exception $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 忘记密码
|
||||
* @param array $params
|
||||
* @return bool
|
||||
* @throws DataNotFoundException|DbException|ModelNotFoundException|OperateException
|
||||
*/
|
||||
public static function forgotPwd(array $params = []): bool
|
||||
{
|
||||
$value = $params['name'] ?? '';
|
||||
$pwd = $params['pwd'] ?? '';
|
||||
$captcha = $params['captcha'] ?? '';
|
||||
$filterVar = filter_var($value, FILTER_VALIDATE_EMAIL);
|
||||
|
||||
// 获取验证服务类
|
||||
$checkClass = $filterVar ? EmailService::class : SmsService::class;
|
||||
if (!$checkClass::checkCaptcha($value, $captcha, 'forgot')) {
|
||||
throw new OperateException('无效的验证码');
|
||||
}
|
||||
|
||||
$where = [$filterVar ? 'email' : 'mobile' => $value];
|
||||
$userInfo = UserModel::where($where)->findOrEmpty()->toArray();
|
||||
if (empty($userInfo)) {
|
||||
throw new OperateException('用户不存在');
|
||||
}
|
||||
|
||||
try {
|
||||
$salt = Random::alpha();
|
||||
$pwd = encryptPwd($pwd, $salt);
|
||||
UserModel::update(['id' => $userInfo['id'], 'pwd' => $pwd, 'salt' => $salt]);
|
||||
} catch (\Exception $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改密码
|
||||
* @param array $params
|
||||
* @param int $userId
|
||||
* @return bool
|
||||
* @throws OperateException
|
||||
*/
|
||||
public static function changePwd(array $params = [], int $userId = 0): bool
|
||||
{
|
||||
$pwd = $params['pwd'] ?? '';
|
||||
$oldPwd = $params['oldpwd'] ?? '';
|
||||
$userInfo = UserModel::where('id', $userId)->findOrEmpty()->toArray();
|
||||
if (empty($userInfo)) {
|
||||
throw new OperateException('用户不存在');
|
||||
}
|
||||
|
||||
$yPwd = encryptPwd($oldPwd, $userInfo['salt']);
|
||||
if (!empty($userInfo['pwd']) && $yPwd != $userInfo['pwd']) {
|
||||
throw new OperateException('原密码错误');
|
||||
}
|
||||
|
||||
$salt = Random::alpha();
|
||||
$pwd = encryptPwd($pwd, $salt);
|
||||
try {
|
||||
UserModel::update([
|
||||
'id' => $userId,
|
||||
'pwd' => $pwd,
|
||||
'salt' => $salt,
|
||||
]);
|
||||
} catch (\Exception $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改手机号
|
||||
* @param $email
|
||||
* @param $captcha
|
||||
* @param $event
|
||||
* @param $userId
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
* @throws UserException|OperateException
|
||||
*/
|
||||
public static function changeEmail($email, $captcha, $event, $userId): bool
|
||||
{
|
||||
if (!EmailService::filterEmail($email)) {
|
||||
throw new OperateException("邮箱格式不正确");
|
||||
}
|
||||
|
||||
if ($email && UserModel::getByEmail($email)) {
|
||||
throw new OperateException("您输入的邮箱已被占用");
|
||||
}
|
||||
|
||||
try {
|
||||
EmailService::checkCaptcha($email, $captcha, $event);
|
||||
UserModel::update(['id' => $userId, 'email' => $email]);
|
||||
} catch (\Throwable $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改手机号
|
||||
* @param $mobile
|
||||
* @param $captcha
|
||||
* @param $event
|
||||
* @param $userId
|
||||
* @return bool
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
* @throws UserException|OperateException
|
||||
*/
|
||||
public static function changeMobile($mobile, $captcha, $event, $userId): bool
|
||||
{
|
||||
if (!SmsService::filterMobile($mobile)) {
|
||||
throw new OperateException("手机号码格式不正确");
|
||||
}
|
||||
|
||||
if ($mobile && UserModel::getByMobile($mobile)) {
|
||||
throw new OperateException("您输入的手机号已被占用");
|
||||
}
|
||||
try {
|
||||
if (!SmsService::checkCaptcha($mobile, $captcha, $event)) {
|
||||
throw new UserException('无效的验证码');
|
||||
}
|
||||
UserModel::update(['id' => $userId, 'mobile' => (int)$mobile]);
|
||||
} catch (\Throwable $e) {
|
||||
throw new OperateException($e->getMessage());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取消息列表
|
||||
* @param int $limit
|
||||
* @param int $page
|
||||
* @param array $where
|
||||
* @return array
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
*/
|
||||
public static function listMessage(mixed $limit = 10, mixed $page = 1, array $where = []): array
|
||||
{
|
||||
$count = (new UserNotice)->where($where)->count();
|
||||
$page = ($count <= $limit) ? 1 : $page;
|
||||
$list = (new UserNotice)->where($where)->order('id', 'desc')->limit($limit)->page($page)->select()->toArray();
|
||||
return [$list, $count];
|
||||
}
|
||||
|
||||
/**
|
||||
* 读取消息
|
||||
* @param $id
|
||||
* @param int $userId
|
||||
* @return array
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException|OperateException
|
||||
*/
|
||||
public static function viewMessage($id, int $userId = 0): array
|
||||
{
|
||||
$where[] = ['id', '=', $id];
|
||||
$where[] = ['user_id', '=', $userId];
|
||||
$msgInfo = UserNotice::where($where)->findOrEmpty()->toArray();
|
||||
if (empty($msgInfo)) {
|
||||
throw new OperateException('消息不存在');
|
||||
}
|
||||
|
||||
if ($msgInfo['status'] == 0) {
|
||||
UserNotice::update(['id' => $id, 'status' => 1]);
|
||||
}
|
||||
|
||||
if ($msgInfo['send_id']) {
|
||||
$fromInfo = UserModel::where('id', $msgInfo['send_id'])->findOrEmpty()->toArray();
|
||||
$msgInfo['nickname'] = $fromInfo['nickname'] ?? 'Unknown';
|
||||
}
|
||||
|
||||
$unread = UserNotice::where(['user_id' => $userId, 'status' => 0])->count();
|
||||
return ['msgInfo' => $msgInfo, 'unread' => $unread];
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $ids
|
||||
* @param string $type
|
||||
* @param int $userId
|
||||
* @return bool
|
||||
* @throws UserException
|
||||
*/
|
||||
public static function batchMessage($ids, string $type = 'del', int $userId = 0): bool
|
||||
{
|
||||
$where[] = ['user_id', '=', $userId];
|
||||
$where[] = ['id', 'in', implode(',', $ids)];
|
||||
try {
|
||||
$type == 'del' ? UserNotice::where($where)->delete() : UserNotice::where($where)->update(['status' => 1]);
|
||||
} catch (\Exception $e) {
|
||||
throw new UserException($e->getMessage());
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
||||
82
app/common/service/user/UserTokenService.php
Normal file
82
app/common/service/user/UserTokenService.php
Normal file
@@ -0,0 +1,82 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\service\user;
|
||||
|
||||
use app\common\model\system\User as UserModel;
|
||||
use Psr\SimpleCache\InvalidArgumentException;
|
||||
use system\Random;
|
||||
use support\Cache;
|
||||
|
||||
/**
|
||||
* 用户token服务
|
||||
* Class UserTokenService
|
||||
*/
|
||||
class UserTokenService
|
||||
{
|
||||
/**
|
||||
* 保活时间
|
||||
* @var int
|
||||
*/
|
||||
protected static int $keepTime = 604800;
|
||||
|
||||
/**
|
||||
* 类构造函数
|
||||
* class constructor.
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
self::$keepTime = config('cookie.expire');
|
||||
}
|
||||
|
||||
/**
|
||||
* 校验登录
|
||||
* @return array
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public static function isLogin(): array
|
||||
{
|
||||
$token = self::getToken();
|
||||
$userId = self::checkToken($token);
|
||||
if (empty($userId)) {
|
||||
return [];
|
||||
}
|
||||
|
||||
$userInfo = UserModel::with('group')->where(['id' => $userId])->findOrEmpty()->toArray();
|
||||
$userInfo['has_password'] = empty($userInfo['password']) ? 0 : 1;
|
||||
unset($userInfo['password'], $userInfo['salt']);
|
||||
return $userInfo;
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成token
|
||||
* @access protected
|
||||
* @param int $id
|
||||
* @return string
|
||||
*/
|
||||
public static function buildToken(int $id = 0): string
|
||||
{
|
||||
return md5(Random::alpha(16) . $id . time());
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取token
|
||||
* return string
|
||||
*/
|
||||
public static function getToken(): string
|
||||
{
|
||||
$token = request()->header('Authorization') ?: request()->header('token');
|
||||
return $token ?? input('token', request()->cookie('token')) ?: 'undefined';
|
||||
}
|
||||
|
||||
/**
|
||||
* 校验token
|
||||
* @access protected
|
||||
* @param string $token
|
||||
* @return mixed
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public static function checkToken(string $token): mixed
|
||||
{
|
||||
return Cache::get($token) ?? 0;
|
||||
}
|
||||
}
|
||||
113
app/common/service/utils/FtpService.php
Normal file
113
app/common/service/utils/FtpService.php
Normal file
@@ -0,0 +1,113 @@
|
||||
<?php
|
||||
|
||||
namespace app\common\service\utils;
|
||||
|
||||
use system\Random;
|
||||
|
||||
/**
|
||||
* FTP上传类
|
||||
* Class FtpService
|
||||
* @package app\common\service\utils
|
||||
*/
|
||||
class FtpService
|
||||
{
|
||||
/**
|
||||
* 默认配置
|
||||
* @var array
|
||||
*/
|
||||
protected static array $config = [
|
||||
'upload_ftp_host' => '127.0.0.1', // 服务器地址
|
||||
'upload_ftp_port' => 21, // 服务器端口
|
||||
'upload_ftp_user' => 'username', // FTP用户名
|
||||
'upload_ftp_pass' => 'password', // FTP密码
|
||||
'upload_path' => 'upload', // 上传路径
|
||||
];
|
||||
|
||||
/**
|
||||
* 类构造函数
|
||||
* class constructor.
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
$config = saenv('upload', true);
|
||||
self::$config = array_merge(self::$config, $config);
|
||||
}
|
||||
|
||||
/**
|
||||
* FTP上传函数
|
||||
* @access public
|
||||
* @param string $source 源文件
|
||||
* @param string $filepath 文件路径
|
||||
* @param string $filename 文件名称
|
||||
* @return bool
|
||||
*/
|
||||
public static function ftpUpload(string $source, string $filepath, string $filename): bool
|
||||
{
|
||||
if (!empty($source) && !empty($filepath)) {
|
||||
// 链接FTP
|
||||
$connect = @ftp_connect(self::$config['upload_ftp_host'], self::$config['upload_ftp_port']) or die('Could not connect');
|
||||
if (!ftp_login($connect, self::$config['upload_ftp_user'], self::$config['upload_ftp_pass'])) {
|
||||
return false;
|
||||
}
|
||||
// 开启被动模式
|
||||
@ftp_pasv($connect, TRUE);
|
||||
$source = @fopen($source, "r");
|
||||
|
||||
// 循环创建文件夹
|
||||
$filepath = str_replace("\\", '/', $filepath);
|
||||
$dirs = explode('/', $filepath);
|
||||
foreach ($dirs as $val) {
|
||||
if (!@ftp_chdir($connect, $val)) {
|
||||
if (!ftp_mkdir($connect, $val)) {
|
||||
//创建失败
|
||||
return false;
|
||||
}
|
||||
// 切换目录
|
||||
@ftp_chdir($connect, $val);
|
||||
}
|
||||
}
|
||||
if (!@ftp_fput($connect, $filename, $source, FTP_BINARY)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@ftp_close($connect);
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* FTP测试函数
|
||||
* @access public
|
||||
* @param array $config 配置信息
|
||||
* @return bool true|false
|
||||
*/
|
||||
public static function ftpTest(array $config): bool
|
||||
{
|
||||
|
||||
try {
|
||||
$connect = @ftp_connect($config['host'], (int)$config['port']);
|
||||
@ftp_login($connect, $config['user'], $config['pass']);
|
||||
// 开启被动模式
|
||||
ftp_pasv($connect, TRUE);
|
||||
$folder = Random::alpha(16);
|
||||
if (ftp_mkdir($connect, $folder)) {
|
||||
// 读取测试文件
|
||||
$location = __DIR__;
|
||||
$source = fopen($location, "r"); // 上传测试文件
|
||||
$filename = $folder . "/target.txt";
|
||||
ftp_fput($connect, $filename, $source, FTP_BINARY);
|
||||
// 删除测试文件
|
||||
ftp_delete($connect, $filename);
|
||||
ftp_rmdir($connect, $folder);
|
||||
ftp_close($connect);
|
||||
return true;
|
||||
}
|
||||
} catch (\Throwable $th) {
|
||||
return false;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
31
app/index/view/user/change_email.html
Normal file
31
app/index/view/user/change_email.html
Normal file
@@ -0,0 +1,31 @@
|
||||
<layout name="layout:layout"/>
|
||||
<!-- 内容主体区域 -->
|
||||
<div id="content">
|
||||
<div class="layui-fluid">
|
||||
<form class="layui-form layui-form-fixed" action="/index/user/changeEmail">
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:__('邮箱地址')}</label>
|
||||
<div class="layui-input-inline" style="width: 265px">
|
||||
<input name="email" placeholder="{:__('请输入邮箱地址')}" type="text" class="layui-input email"
|
||||
autocomplete="off" lay-verify="required|email"/>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:__('验证码')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="captcha" placeholder="{:__('请输入验证码')}" type="text" class="layui-input"
|
||||
autocomplete="off" lay-verify="required" maxlength="10"/>
|
||||
</div>
|
||||
<button class="layui-btn layui-btn-normal" type="button" lay-ajax data-url="/api/ajax/emailSend"
|
||||
data-object="email:email,event:event">{:__('获取')}
|
||||
</button>
|
||||
</div>
|
||||
<input type="text" hidden name="event" class="event" value="change">
|
||||
<div class="layui-footer layui-form-item layui-center">
|
||||
<button class="layui-btn layui-btn-primary" type="button" sa-event="closeDialog">{:__('取消')}</button>
|
||||
<button class="layui-btn layui-btn-normal " data-reload="parent" lay-filter="submitIframe" lay-submit>{:__('提交')}</button>
|
||||
</div>
|
||||
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
30
app/index/view/user/change_mobile.html
Normal file
30
app/index/view/user/change_mobile.html
Normal file
@@ -0,0 +1,30 @@
|
||||
<layout name="layout:layout"/>
|
||||
<!-- 内容主体区域 -->
|
||||
<div id="content">
|
||||
<div class="layui-fluid">
|
||||
<form class="layui-form layui-form-fixed" action="/index/user/changeMobile">
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:__('手机号')}</label>
|
||||
<div class="layui-input-inline" style="width: 265px">
|
||||
<input name="mobile" placeholder="{:__('请输入号码')}" type="text" class="layui-input mobile"
|
||||
autocomplete="off" lay-verify="required|phone"/>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:__('验证码')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="captcha" placeholder="{:__('请输入验证码')}" type="text" class="layui-input"
|
||||
autocomplete="off" lay-verify="required" maxlength="10"/>
|
||||
</div>
|
||||
<button class="layui-btn layui-btn-normal" type="button" lay-ajax data-url="/api/ajax/smsSend"
|
||||
data-object="mobile:mobile,event:event">{:__('获取')}
|
||||
</button>
|
||||
</div>
|
||||
<input type="text" hidden name="event" class="event" value="change">
|
||||
<div class="layui-footer layui-form-item layui-center">
|
||||
<button class="layui-btn layui-btn-primary" type="button" sa-event="closeDialog">{:__('取消')}</button>
|
||||
<button class="layui-btn layui-btn-normal" data-reload="parent" lay-filter="submitIframe" lay-submit>{:__('提交')}</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
82
app/index/view/user/change_pwd.html
Normal file
82
app/index/view/user/change_pwd.html
Normal file
@@ -0,0 +1,82 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>修改密码 | {$site_name}</title>
|
||||
<include file="user:include"/>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<div class="layui-card-body">
|
||||
<form action="/index/user/changePwd" class="layui-form layui-form-fixed">
|
||||
<input type="text" name="nickname" value="{$user.nickname}" hidden="">
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">原密码</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" id="oldpwd" name="oldpwd" class="layui-input" >
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">新密码</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="password" id="pwd" name="pwd" class="layui-input" >
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">确认密码</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="password" id="repwd" name="repwd" class="layui-input" >
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item" style="margin-top: 22px;text-align: center;">
|
||||
<button type="button" class="layui-btn layui-btn-primary" sa-event="closeDialog" >关闭</button>
|
||||
<button type="submit" class="layui-btn layui-btn-normal" lay-submit="" lay-filter="submit">立即提交</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
<script>
|
||||
layui.use(['jquery','form'],function(){
|
||||
var $ = layui.jquery;
|
||||
var form = layui.form;
|
||||
|
||||
//监听提交
|
||||
form.on('submit(submit)', function(data){
|
||||
|
||||
if($("#pwd").val().length < 6){
|
||||
layer.tips('请至少输入6个字符作为密码',"#pwd",{'tips':3});
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($("#pwd").val() !== $("#repwd").val()) {
|
||||
layer.tips('两次输入的密码不一样!',"#repwd",{'tips':3});
|
||||
return false;
|
||||
}
|
||||
|
||||
$.ajax({
|
||||
url: $(this).attr('action'),
|
||||
type: 'POST',
|
||||
dataType: 'json',
|
||||
timeout: 6000,
|
||||
data: data.field,
|
||||
success: function(res){
|
||||
|
||||
if (res.code === 200) {
|
||||
top.layer.msg(res.msg);
|
||||
parent.location.reload();
|
||||
}
|
||||
else {
|
||||
top.layer.msg(res.msg,'error');
|
||||
}
|
||||
|
||||
}
|
||||
});
|
||||
|
||||
return false;
|
||||
});
|
||||
|
||||
})
|
||||
</script>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
234
app/index/view/user/login.html
Normal file
234
app/index/view/user/login.html
Normal file
@@ -0,0 +1,234 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>{$site_name} 用户登录</title>
|
||||
<meta http-equiv="Cache-Control" content="no-transform"/>
|
||||
<meta http-equiv="Cache-Control" content="no-siteapp"/>
|
||||
<meta name="applicable-device" content="pc,mobile">
|
||||
<meta name="viewport"
|
||||
content="width=device-width, initial-scale=1.0, user-scalable=0, minimum-scale=1.0, maximum-scale=1.0">
|
||||
<script src="/static/js/layui/layui.js"></script>
|
||||
<link rel="stylesheet" type="text/css" href="/static/js/layui/css/font-awesome.css"/>
|
||||
<link rel="stylesheet" type="text/css" href="/static/js/layui/css/layui.css"/>
|
||||
<link rel="stylesheet" type="text/css" href="/static/css/login.css"/>
|
||||
<script src="/static/js/center.js?v={:release()}"></script>
|
||||
<link rel="shortcut icon" href="/favicon.ico" type="image/x-icon"/>
|
||||
</head>
|
||||
<body>
|
||||
<div id="header"></div>
|
||||
<div id="content" class="userLogin layui-fluid">
|
||||
<div class="formLogin form-items active">
|
||||
<div class="layui-tab layui-tab-brief">
|
||||
<ul class="layui-tab-title">
|
||||
<li class="layui-this" data-action="login">账号登录</li>
|
||||
<li data-action="mobileLogin">免密登录</li>
|
||||
</ul>
|
||||
<div class="layui-tab-content pb0">
|
||||
<div class="layui-tab-item layui-show">
|
||||
<form class="layui-form login" action="/index/user/login" method="post">
|
||||
<div class="layui-form-item item-input">
|
||||
<input type="text" id="nickname" name="nickname" lay-verify="required" data-title="账号"
|
||||
placeholder="{:__('请输入账号或邮箱')}" class="inputStyle" value="">
|
||||
</div>
|
||||
<div class="layui-form-item item-input">
|
||||
<input type="password" id="pwd" name="pwd" lay-verify="required" data-title="密码"
|
||||
maxlength="32"
|
||||
placeholder="{:__('密码')}" class="inputStyle" value="">
|
||||
<span class="fr-icon visiblePwd"><i class="layui-icon fa-eye-slash"></i></span>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
<div class="layui-tab-item">
|
||||
<form class="layui-form mobileLogin" action="/index/user/mobileLogin" method="post">
|
||||
<div class="layui-form-item item-input">
|
||||
<input type="text" id="mobile" name="mobile" lay-verify="required" data-title="手机号"
|
||||
placeholder="{:__('请输入手机号码')}" class="inputStyle mobile" value="15100000000">
|
||||
</div>
|
||||
<div class="layui-form-item item-input">
|
||||
<input type="text" id="captcha" name="captcha" lay-verify="required" data-title="验证码"
|
||||
maxlength="6"
|
||||
placeholder="{:__('验证码')}" class="inputStyle" value="123456">
|
||||
<input type="text" hidden name="event" class="event" value="login">
|
||||
<span class="fr-icon getCaptcha" lay-ajax data-url="/api/ajax/smsSend"
|
||||
data-object="mobile:mobile,event:event">获取验证码</span>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form">
|
||||
<input type="hidden" id="token" name="__token__" value="{:token()}"/>
|
||||
<div class="layui-form-item">
|
||||
<div class="fl">
|
||||
<input type="checkbox" name="remember" lay-skin="primary"
|
||||
title="{:__('记住密码')}" checked>
|
||||
</div>
|
||||
<div class="fr">
|
||||
<a lay-open data-title="找回密码" data-area="500px,320px" data-object="top"
|
||||
data-url="/index/user/forgot"
|
||||
class="forgot-link" style="margin-top: 7px;">{:__('忘记密码?')}</a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<input id="iframeSubmit" type="submit" value="{:__('立即登录')}"
|
||||
class="layui-btn layui-btn-login layui-btn-fluid layui-btn-normal">
|
||||
</div>
|
||||
<div class="agreement">
|
||||
<span>未注册手机验证后自动登录,注册即代表同意<a href="#">《用户协议》</a>以及网站<a
|
||||
href="#">《隐私保护指引》</a></span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="scanLogin form-items">
|
||||
<div class="qrcode-title">扫码登录</div>
|
||||
<div class="qrcode-Box">
|
||||
<div class="qrcode-img" id="qrcode" title="">
|
||||
<canvas width="150" height="150" style="display: none;"></canvas>
|
||||
<img id="scanCover" src="/static/images/qrcode-qun.png" style="display: block;" width="150" height="150"
|
||||
alt="Scan me!">
|
||||
</div>
|
||||
<p>打开<a href="#" target="_blank" rel="noopener noreferrer">{$site_name} App</a></p>
|
||||
<p>在「我的」页面右上角打开扫一扫</p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="loginLine"></div>
|
||||
<div class="socialLogin">
|
||||
<span class="social-title">社交帐号登录</span>
|
||||
<div class="social-group">
|
||||
<a class="social-item" href="{:url('/index/third/login',['type'=>'weixin'])}&ref={$referer}"
|
||||
target="_top">
|
||||
<span>
|
||||
<i class="layui-icon layui-icon-login-wechat"></i>
|
||||
<span class="tit">微信</span>
|
||||
</span>
|
||||
</a>
|
||||
<a class="social-item" href="{:url('/index/third/login',['type'=>'qq'])}&ref={$referer}"
|
||||
target="_top">
|
||||
<span>
|
||||
<i class="layui-icon layui-icon-login-qq"></i>
|
||||
<span class="tit">QQ</span>
|
||||
</span>
|
||||
</a>
|
||||
<a class="social-item" href="{:url('/index/third/login',['type'=>'weibo'])}&ref={$referer}"
|
||||
target="_top">
|
||||
<span><i class="layui-icon layui-icon-login-weibo"></i>
|
||||
<span class="tit">微博</span></span>
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="appUserDown">
|
||||
<div class="fr">
|
||||
<a class="link" href="#" target="_top">
|
||||
<span>
|
||||
<i class="layui-icon fa-apple"></i>
|
||||
<i class="layui-icon fa-android"></i>
|
||||
<i class="layui-icon fa-windows"></i>
|
||||
<span class="tit">下载 APP</span>
|
||||
</span>
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="switchTabs" id="switchTabs">
|
||||
<img class="switch-item active" data-action="scanLogin" src="/static/images/qrcode-login.png" alt="qrcode">
|
||||
<img class="switch-item" data-action="login" src="/static/images/mobile-login.png" alt="qrcode">
|
||||
</div>
|
||||
|
||||
</div>
|
||||
<script>
|
||||
layui.use(['jquery', 'layer'], function (e) {
|
||||
|
||||
let $ = layui.jquery;
|
||||
let layer = layui.layer;
|
||||
let queryTicket = null;
|
||||
|
||||
/* 点击执行登录 */
|
||||
$('#iframeSubmit').click(function (e) {
|
||||
let data = {};
|
||||
let action = $('li.layui-this').data('action');
|
||||
let form = $('form.' + action);
|
||||
form.find('input').each(function (index, item) {
|
||||
let value = $(item).val();
|
||||
let name = $(item).attr('name');
|
||||
let title = $(item).data('title');
|
||||
if (!value) {
|
||||
layer.msg(title + '不能为空', 'info');
|
||||
$(item).focus();
|
||||
return false;
|
||||
}
|
||||
data[name] = value;
|
||||
});
|
||||
|
||||
/* 发送登录数据 */
|
||||
if (Object.keys(data).length >= 2) {
|
||||
$.post(form.attr('action'), data, function (res) {
|
||||
if (res.code === 200) {
|
||||
layer.msg(res.msg);
|
||||
setTimeout(function () {
|
||||
window.location.href = '/index/user/index';
|
||||
}, 1000);
|
||||
} else {
|
||||
layer.msg(res.msg, 'error')
|
||||
}
|
||||
}, 'json');
|
||||
}
|
||||
})
|
||||
|
||||
/* 显示隐藏密码 */
|
||||
$('.visiblePwd').click(function () {
|
||||
let type = $('#pwd').attr('type');
|
||||
if (type === 'password') {
|
||||
$('#pwd').attr('type', 'text');
|
||||
$(this).find('i.layui-icon').addClass('fa-eye').removeClass('fa-eye-slash');
|
||||
} else {
|
||||
$('#pwd').attr('type', 'password');
|
||||
$(this).find('i.layui-icon').addClass('fa-eye-slash').removeClass('fa-eye');
|
||||
}
|
||||
});
|
||||
|
||||
/* 切换登录方式 */
|
||||
$("#switchTabs .switch-item").click(function () {
|
||||
$(this).removeClass('active');
|
||||
let action = $(this).data('action');
|
||||
$("#switchTabs .switch-item").not($(this)).addClass('active');
|
||||
if (action === 'scanLogin') {
|
||||
$('.scanLogin').addClass('active');
|
||||
$('.formLogin').removeClass('active');
|
||||
$.get('/index/user/scanLogin', {}, function (res) {
|
||||
if (res.code === 200) {
|
||||
!queryTicket && queryRequest(res.data.ticket);
|
||||
$('#scanCover').attr('src', res.data.qrcode || '/static/images/qrcode.png');
|
||||
} else {
|
||||
layer.msg(res.msg, 'info');
|
||||
}
|
||||
})
|
||||
} else {
|
||||
clearInterval(queryTicket);
|
||||
queryTicket = null;
|
||||
$('.formLogin').addClass('active');
|
||||
$('.scanLogin').removeClass('active');
|
||||
}
|
||||
});
|
||||
|
||||
/* 定时器 */
|
||||
const queryRequest = function (ticket) {
|
||||
queryTicket = setInterval(function () {
|
||||
$.post('/index/user/scanTicket', {
|
||||
ticket: ticket,
|
||||
}, function (res) {
|
||||
if (res.code === 200) {
|
||||
clearInterval(queryTicket);
|
||||
top.layer.msg(res.msg);
|
||||
top.location.reload();
|
||||
}
|
||||
}, 'json');
|
||||
}, 1000);
|
||||
}
|
||||
});
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user