CRMEB/crmeb/app/api/controller/v1/LoginController.php
2023-09-19 17:39:39 +08:00

511 lines
16 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<?php
// +----------------------------------------------------------------------
// | CRMEB [ CRMEB赋能开发者助力企业发展 ]
// +----------------------------------------------------------------------
// | Copyright (c) 2016~2023 https://www.crmeb.com All rights reserved.
// +----------------------------------------------------------------------
// | Licensed CRMEB并不是自由软件未经许可不能去掉CRMEB相关版权
// +----------------------------------------------------------------------
// | Author: CRMEB Team <admin@crmeb.com>
// +----------------------------------------------------------------------
namespace app\api\controller\v1;
use app\Request;
use app\services\message\notice\SmsService;
use app\services\wechat\WechatServices;
use think\facade\Config;
use crmeb\services\CacheService;
use app\services\user\LoginServices;
use think\exception\ValidateException;
use app\api\validate\user\RegisterValidates;
/**
* 微信小程序授权类
* Class AuthController
* @package app\api\controller
*/
class LoginController
{
protected $services;
/**
* LoginController constructor.
* @param LoginServices $services
*/
public function __construct(LoginServices $services)
{
$this->services = $services;
}
/**
* H5账号登陆
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function login(Request $request)
{
[$account, $password, $spread] = $request->postMore([
'account', 'password', 'spread'
], true);
if (!$account || !$password) {
return app('json')->fail(410000);
}
if (strlen(trim($password)) < 6 || strlen(trim($password)) > 32) {
return app('json')->fail(400762);
}
return app('json')->success(410001, $this->services->login($account, $password, $spread));
}
/**
* 退出登录
* @param Request $request
* @return mixed
*/
public function logout(Request $request)
{
$key = trim(ltrim($request->header(Config::get('cookie.token_name')), 'Bearer'));
CacheService::delete(md5($key));
return app('json')->success(410002);
}
/**
* 获取发送验证码key
* @return mixed
*/
public function verifyCode()
{
$unique = password_hash(uniqid(true), PASSWORD_BCRYPT);
CacheService::set('sms.key.' . $unique, 0, 300);
$time = sys_config('verify_expire_time', 1);
return app('json')->success(['key' => $unique, 'expire_time' => $time]);
}
/**
* 获取图片验证码
* @param Request $request
* @return \think\Response
*/
public function captcha(Request $request)
{
ob_clean();
$rep = captcha();
$key = app('session')->get('captcha.key');
$uni = $request->get('key');
if ($uni) {
CacheService::set('sms.key.cap.' . $uni, $key, 300);
}
return $rep;
}
/**
* 验证验证码是否正确
* @param $uni
* @param string $code
* @return bool
* @throws \Psr\SimpleCache\InvalidArgumentException
*/
protected function checkCaptcha($uni, string $code): bool
{
$cacheName = 'sms.key.cap.' . $uni;
if (!CacheService::has($cacheName)) {
return false;
}
$key = CacheService::get($cacheName);
$code = mb_strtolower($code, 'UTF-8');
$res = password_verify($code, $key);
if ($res) {
CacheService::delete($cacheName);
}
return $res;
}
/**
* 验证码发送
* @param Request $request
* @param SmsService $services
* @return mixed
*/
public function verify(Request $request, SmsService $services)
{
[$phone, $type, $key, $captchaType, $captchaVerification] = $request->postMore([
['phone', 0],
['type', ''],
['key', ''],
['captchaType', ''],
['captchaVerification', ''],
], true);
$keyName = 'sms.key.' . $key;
$nowKey = 'sms.' . date('YmdHi');
if (!CacheService::has($keyName)) {
return app('json')->fail(410003);
}
$total = 1;
if (CacheService::has($nowKey)) {
$total = CacheService::get($nowKey);
if ($total > Config::get('sms.maxMinuteCount', 20))
return app('json')->success(410006);
}
//二次验证
try {
aj_captcha_check_two($captchaType, $captchaVerification);
} catch (\Throwable $e) {
return app('json')->fail($e->getError());
}
try {
validate(RegisterValidates::class)->scene('code')->check(['phone' => $phone]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
$time = sys_config('verify_expire_time', 1);
$smsCode = $this->services->verify($services, $phone, $type, $time, app()->request->ip());
if ($smsCode) {
CacheService::set('code_' . $phone, $smsCode, $time * 60);
CacheService::set($nowKey, $total, 61);
return app('json')->success(410007);
} else {
return app('json')->fail(410008);
}
}
/**
* H5注册新用户
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function register(Request $request)
{
[$account, $captcha, $password, $spread] = $request->postMore([['account', ''], ['captcha', ''], ['password', ''], ['spread', 0]], true);
try {
validate(RegisterValidates::class)->scene('register')->check(['account' => $account, 'captcha' => $captcha, 'password' => $password]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
if (strlen(trim($password)) < 6 || strlen(trim($password)) > 32) {
return app('json')->fail(400762);
}
$verifyCode = CacheService::get('code_' . $account);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha)
return app('json')->fail(410010);
if (md5($password) == md5('123456')) return app('json')->fail(410012);
$registerStatus = $this->services->register($account, $password, $spread, 'h5');
if ($registerStatus) {
return app('json')->success(410013);
}
return app('json')->fail(410014);
}
/**
* 密码修改
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function reset(Request $request)
{
[$account, $captcha, $password] = $request->postMore([['account', ''], ['captcha', ''], ['password', '']], true);
try {
validate(RegisterValidates::class)->scene('register')->check(['account' => $account, 'captcha' => $captcha, 'password' => $password]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
if (strlen(trim($password)) < 6 || strlen(trim($password)) > 32) {
return app('json')->fail(400762);
}
$verifyCode = CacheService::get('code_' . $account);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha) {
return app('json')->fail(410010);
}
if ($password == '123456') return app('json')->fail(410012);
$resetStatus = $this->services->reset($account, $password);
if ($resetStatus) return app('json')->success(100001);
return app('json')->fail(100007);
}
/**
* 手机号登录
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function mobile(Request $request)
{
[$phone, $captcha, $spread] = $request->postMore([['phone', ''], ['captcha', ''], ['spread', 0]], true);
//验证手机号
try {
validate(RegisterValidates::class)->scene('code')->check(['phone' => $phone]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
//验证验证码
$verifyCode = CacheService::get('code_' . $phone);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha) {
return app('json')->fail(410010);
}
$user_type = $request->getFromType() ? $request->getFromType() : 'h5';
$token = $this->services->mobile($phone, $spread, $user_type);
if ($token) {
CacheService::delete('code_' . $phone);
return app('json')->success(410001, $token);
} else {
return app('json')->fail(410002);
}
}
/**
* H5切换登陆
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function switch_h5(Request $request)
{
$from = $request->post('from', 'wechat');
$user = $request->user();
$token = $this->services->switchAccount($user, $from);
if ($token) {
$token['userInfo'] = $user;
return app('json')->success(410001, $token);
} else
return app('json')->fail(410002);
}
/**
* 绑定手机号
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function binding_phone(Request $request)
{
list($phone, $captcha, $key) = $request->postMore([
['phone', ''],
['captcha', ''],
['key', '']
], true);
//验证手机号
try {
validate(RegisterValidates::class)->scene('code')->check(['phone' => $phone]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
if (!$key) {
return app('json')->fail(100100);
}
if (!$phone) {
return app('json')->fail(410015);
}
//验证验证码
$verifyCode = CacheService::get('code_' . $phone);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha) {
return app('json')->fail(410010);
}
$re = $this->services->bindind_phone($phone, $key);
if ($re) {
CacheService::delete('code_' . $phone);
return app('json')->success(410016, $re);
} else
return app('json')->fail(410017);
}
/**
* 绑定手机号
* @param Request $request
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function user_binding_phone(Request $request)
{
list($phone, $captcha, $step) = $request->postMore([
['phone', ''],
['captcha', ''],
['step', 0]
], true);
//验证手机号
try {
validate(RegisterValidates::class)->scene('code')->check(['phone' => $phone]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
if (!$step) {
//验证验证码
$verifyCode = CacheService::get('code_' . $phone);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha)
return app('json')->fail(410010);
}
$uid = (int)$request->uid();
$re = $this->services->userBindindPhone($uid, $phone, $step);
if ($re) {
CacheService::delete('code_' . $phone);
return app('json')->success($re['msg'] ?? 410016, $re['data'] ?? []);
} else
return app('json')->fail(410017);
}
public function update_binding_phone(Request $request)
{
[$phone, $captcha] = $request->postMore([
['phone', ''],
['captcha', ''],
], true);
//验证手机号
try {
validate(RegisterValidates::class)->scene('code')->check(['phone' => $phone]);
} catch (ValidateException $e) {
return app('json')->fail($e->getError());
}
//验证验证码
$verifyCode = CacheService::get('code_' . $phone);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha)
return app('json')->fail(410010);
$uid = (int)$request->uid();
$re = $this->services->updateBindindPhone($uid, $phone);
if ($re) {
CacheService::delete('code_' . $phone);
return app('json')->success($re['msg'] ?? 100001, $re['data'] ?? []);
} else
return app('json')->fail(100007);
}
/**
* 设置扫描二维码状态
* @param string $code
* @return mixed
*/
public function setLoginKey(string $code)
{
if (!$code) {
return app('json')->fail(410020);
}
$cacheCode = CacheService::get($code);
if ($cacheCode === false || $cacheCode === null) {
return app('json')->fail(410021);
}
CacheService::set($code, '0', 600);
return app('json')->success();
}
/**
* apple快捷登陆
* @param Request $request
* @param WechatServices $services
* @return mixed
* @throws \Psr\SimpleCache\InvalidArgumentException
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
*/
public function appleLogin(Request $request, WechatServices $services)
{
[$openId, $phone, $email, $captcha] = $request->postMore([
['openId', ''],
['phone', ''],
['email', ''],
['captcha', '']
], true);
if ($phone) {
if (!$captcha) {
return app('json')->fail(410004);
}
//验证验证码
$verifyCode = CacheService::get('code_' . $phone);
if (!$verifyCode)
return app('json')->fail(410009);
$verifyCode = substr($verifyCode, 0, 6);
if ($verifyCode != $captcha) {
CacheService::delete('code_' . $phone);
return app('json')->fail(410010);
}
}
if ($email == '') $email = substr(md5($openId), 0, 12);
$userInfo = [
'openId' => $openId,
'unionid' => '',
'avatarUrl' => sys_config('h5_avatar'),
'nickName' => $email,
];
$token = $services->appAuth($userInfo, $phone, 'apple');
if ($token) {
return app('json')->success(410001, $token);
} else if ($token === false) {
return app('json')->success(410001, ['isbind' => true]);
} else {
return app('json')->fail(410019);
}
}
/**
* 滑块验证
* @return mixed
*/
public function ajcaptcha(Request $request)
{
$captchaType = $request->get('captchaType');
return app('json')->success(aj_captcha_create($captchaType));
}
/**
* 一次验证
* @return mixed
*/
public function ajcheck(Request $request)
{
[$token, $pointJson, $captchaType] = $request->postMore([
['token', ''],
['pointJson', ''],
['captchaType', ''],
], true);
try {
aj_captcha_check_one($captchaType, $token, $pointJson);
return app('json')->success();
} catch (\Throwable $e) {
return app('json')->fail(400336);
}
}
}