This commit is contained in:
mkm 2024-04-26 11:26:12 +08:00
parent 54a510494c
commit d6516e3621
110 changed files with 5711 additions and 145 deletions

24
app/ExceptionHandler.php Normal file
View File

@ -0,0 +1,24 @@
<?php
namespace App;
use Next\VarDumper\Dumper;
use Next\VarDumper\DumperHandler;
use support\exception\Handler;
use Throwable;
use Webman\Http\Request;
use Webman\Http\Response;
class ExceptionHandler extends Handler
{
use DumperHandler;
public function render(Request $request, Throwable $exception): Response
{
if ($exception instanceof Dumper) {
return \response(self::convertToHtml($exception));
}
return parent::render($request, $exception);
}
}

View File

@ -4,10 +4,10 @@
namespace app\admin\controller;
use app\common\controller\BaseLikeAdminController;
use app\common\controller\BaseLikeController;
use app\common\lists\BaseDataLists;
class BaseAdminController extends BaseLikeAdminController
class BaseAdminController extends BaseLikeController
{
protected $adminId = 0;
protected $adminInfo = [];

View File

@ -0,0 +1,95 @@
<?php
namespace app\admin\controller\order;
use app\admin\controller\BaseAdminController;
use app\admin\lists\order\CartLists;
use app\admin\logic\order\CartLogic;
use app\admin\validate\order\CartValidate;
/**
* 购物车表控制器
* Class CartController
* @package app\admin\controller\order
*/
class CartController extends BaseAdminController
{
/**
* @notes 获取购物车表列表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function lists()
{
return $this->dataLists(new CartLists());
}
/**
* @notes 添加购物车表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function add()
{
$params = (new CartValidate())->post()->goCheck('add');
$result = CartLogic::add($params);
if (true === $result) {
return $this->success('添加成功', [], 1, 1);
}
return $this->fail(CartLogic::getError());
}
/**
* @notes 编辑购物车表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function edit()
{
$params = (new CartValidate())->post()->goCheck('edit');
$result = CartLogic::edit($params);
if (true === $result) {
return $this->success('编辑成功', [], 1, 1);
}
return $this->fail(CartLogic::getError());
}
/**
* @notes 删除购物车表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function delete()
{
$params = (new CartValidate())->post()->goCheck('delete');
CartLogic::delete($params);
return $this->success('删除成功', [], 1, 1);
}
/**
* @notes 获取购物车表详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function detail()
{
$params = (new CartValidate())->goCheck('detail');
$result = CartLogic::detail($params);
return $this->data($result);
}
}

View File

@ -0,0 +1,97 @@
<?php
namespace app\admin\controller\retail;
use app\admin\controller\BaseAdminController;
use app\admin\lists\retail\CashierclassLists;
use app\admin\logic\retail\CashierclassLogic;
use app\admin\validate\retail\CashierclassValidate;
/**
* 零售订单控制器
* Class CashierclassController
* @package app\admin\controller\retail
*/
class CashierclassController extends BaseAdminController
{
/**
* @notes 获取零售订单列表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function lists()
{
$lists=new CashierclassLists();
$lists->setSearch(['merchant'=>222]);
return $this->dataLists($lists);
}
/**
* @notes 添加零售订单
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function add()
{
$params = (new CashierclassValidate())->post()->goCheck('add');
$result = CashierclassLogic::add($params);
if (true === $result) {
return $this->success('添加成功', [], 1, 1);
}
return $this->fail(CashierclassLogic::getError());
}
/**
* @notes 编辑零售订单
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function edit()
{
$params = (new CashierclassValidate())->post()->goCheck('edit');
$result = CashierclassLogic::edit($params);
if (true === $result) {
return $this->success('编辑成功', [], 1, 1);
}
return $this->fail(CashierclassLogic::getError());
}
/**
* @notes 删除零售订单
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function delete()
{
$params = (new CashierclassValidate())->post()->goCheck('delete');
CashierclassLogic::delete($params);
return $this->success('删除成功', [], 1, 1);
}
/**
* @notes 获取零售订单详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function detail()
{
$params = (new CashierclassValidate())->goCheck('detail');
$result = CashierclassLogic::detail($params);
return $this->data($result);
}
}

View File

@ -0,0 +1,95 @@
<?php
namespace app\admin\controller\retail;
use app\admin\controller\BaseAdminController;
use app\admin\lists\retail\CashierinfoLists;
use app\admin\logic\retail\CashierinfoLogic;
use app\admin\validate\retail\CashierinfoValidate;
/**
* 零售订单详情控制器
* Class CashierinfoController
* @package app\admin\controller\retail
*/
class CashierinfoController extends BaseAdminController
{
/**
* @notes 获取零售订单详情列表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function lists()
{
return $this->dataLists(new CashierinfoLists());
}
/**
* @notes 添加零售订单详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function add()
{
$params = (new CashierinfoValidate())->post()->goCheck('add');
$result = CashierinfoLogic::add($params);
if (true === $result) {
return $this->success('添加成功', [], 1, 1);
}
return $this->fail(CashierinfoLogic::getError());
}
/**
* @notes 编辑零售订单详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function edit()
{
$params = (new CashierinfoValidate())->post()->goCheck('edit');
$result = CashierinfoLogic::edit($params);
if (true === $result) {
return $this->success('编辑成功', [], 1, 1);
}
return $this->fail(CashierinfoLogic::getError());
}
/**
* @notes 删除零售订单详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function delete()
{
$params = (new CashierinfoValidate())->post()->goCheck('delete');
CashierinfoLogic::delete($params);
return $this->success('删除成功', [], 1, 1);
}
/**
* @notes 获取零售订单详情详情
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function detail()
{
$params = (new CashierinfoValidate())->goCheck('detail');
$result = CashierinfoLogic::detail($params);
return $this->data($result);
}
}

View File

@ -0,0 +1,59 @@
<?php
namespace app\admin\controller\setting\pay;
use app\admin\controller\BaseAdminController;
use app\admin\lists\setting\pay\PayConfigLists;
use app\admin\logic\setting\pay\PayConfigLogic;
use app\admin\validate\setting\PayConfigValidate;
/**
* 支付配置
* Class PayConfigController
* @package app\admin\controller\setting\pay
*/
class PayConfigController extends BaseAdminController
{
/**
* @notes 设置支付配置
* @return \think\response\Json
* @author 段誉
* @date 2023/2/23 16:14
*/
public function setConfig()
{
$params = (new PayConfigValidate())->post()->goCheck();
PayConfigLogic::setConfig($params);
return $this->success('设置成功', [], 1, 1);
}
/**
* @notes 获取支付配置
* @return \think\response\Json
* @author 段誉
* @date 2023/2/23 16:14
*/
public function getConfig()
{
$id = (new PayConfigValidate())->goCheck('get');
$result = PayConfigLogic::getConfig($id);
return $this->success('获取成功', $result);
}
/**
* @notes
* @return \think\response\Json
* @author 段誉
* @date 2023/2/23 16:15
*/
public function lists()
{
return $this->dataLists(new PayConfigLists());
}
}

View File

@ -0,0 +1,49 @@
<?php
namespace app\admin\controller\setting\pay;
use app\admin\controller\BaseAdminController;
use app\admin\logic\setting\pay\PayWayLogic;
/**
* 支付方式
* Class PayWayController
* @package app\adminapi\controller\setting\pay
*/
class PayWayController extends BaseAdminController
{
/**
* @notes 获取支付方式
* @return \think\response\Json
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:27
*/
public function getPayWay()
{
$result = PayWayLogic::getPayWay();
return $this->success('获取成功',$result);
}
/**
* @notes 设置支付方式
* @return \think\response\Json
* @throws \Exception
* @author 段誉
* @date 2023/2/23 16:27
*/
public function setPayWay()
{
$params = $this->request->post();
$result = (new PayWayLogic())->setPayWay($params);
if (true !== $result) {
return $this->fail($result);
}
return $this->success('操作成功',[],1, 1);
}
}

View File

@ -21,6 +21,21 @@ class UserController extends BaseAdminController
return $this->dataLists(new UserLists());
}
/**
* @notes 添加用户列表
* @return \think\response\Json
* @author likeadmin
* @date 2024/04/25 10:20
*/
public function add()
{
$params = (new UserValidate())->post()->goCheck('add');
$result = UserLogic::add($params);
if (true === $result) {
return $this->success('添加成功', [], 1, 1);
}
return $this->fail(UserLogic::getError());
}
/**
* @notes 获取用户详情

View File

@ -0,0 +1,65 @@
<?php
namespace app\admin\lists\order;
use app\admin\lists\BaseAdminDataLists;
use app\common\model\order\Cart;
use app\common\lists\ListsSearchInterface;
/**
* 购物车表列表
* Class CartLists
* @package app\admin\listsorder
*/
class CartLists extends BaseAdminDataLists implements ListsSearchInterface
{
/**
* @notes 设置搜索条件
* @return \string[][]
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function setSearch(): array
{
return [
];
}
/**
* @notes 获取购物车表列表
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function lists(): array
{
return Cart::where($this->searchWhere)
->field(['cart_id'])
->limit($this->limitOffset, $this->limitLength)
->order(['cart_id' => 'desc'])
->select()
->toArray();
}
/**
* @notes 获取购物车表数量
* @return int
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function count(): int
{
return Cart::where($this->searchWhere)->count();
}
}

View File

@ -0,0 +1,71 @@
<?php
namespace app\admin\lists\retail;
use app\admin\lists\BaseAdminDataLists;
use app\common\model\retail\Cashierclass;
use app\common\lists\ListsSearchInterface;
/**
* 零售订单列表
* Class CashierclassLists
* @package app\admin\listsretail
*/
class CashierclassLists extends BaseAdminDataLists implements ListsSearchInterface
{
/**
* @notes 设置搜索条件
* @return \string[][]
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function setSearch(): array
{
$search= [
'=' => ['merchant', 'store_id', 'customer', 'time', 'number', 'user', 'pay_type', 'type', 'auditinguser'],
];
return $search;
}
public function queryWhere(): array
{
$where=['ass'];
return $where;
}
/**
* @notes 获取零售订单列表
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function lists(): array
{
return Cashierclass::where($this->searchWhere)
->field(['id', 'merchant', 'store_id', 'customer', 'time', 'number', 'total', 'deduction_price', 'actual', 'money', 'user', 'pay_type', 'type', 'auditinguser', 'auditingtime'])
->limit($this->limitOffset, $this->limitLength)
->order(['id' => 'desc'])
->select()
->toArray();
}
/**
* @notes 获取零售订单数量
* @return int
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function count(): int
{
return Cashierclass::where($this->searchWhere)->count();
}
}

View File

@ -0,0 +1,65 @@
<?php
namespace app\admin\lists\retail;
use app\admin\lists\BaseAdminDataLists;
use app\common\model\retail\Cashierinfo;
use app\common\lists\ListsSearchInterface;
/**
* 零售订单详情列表
* Class CashierinfoLists
* @package app\admin\listsretail
*/
class CashierinfoLists extends BaseAdminDataLists implements ListsSearchInterface
{
/**
* @notes 设置搜索条件
* @return \string[][]
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function setSearch(): array
{
return [
'=' => ['pid', 'room', 'goods', 'warehouse', 'serial'],
];
}
/**
* @notes 获取零售订单详情列表
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function lists(): array
{
return Cashierinfo::where($this->searchWhere)
->field(['id', 'pid', 'room', 'goods', 'warehouse', 'serial', 'nums', 'price', 'discount', 'total', 'data', 'more', 'sort'])
->limit($this->limitOffset, $this->limitLength)
->order(['id' => 'desc'])
->select()
->toArray();
}
/**
* @notes 获取零售订单详情数量
* @return int
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function count(): int
{
return Cashierinfo::where($this->searchWhere)->count();
}
}

View File

@ -0,0 +1,50 @@
<?php
namespace app\admin\lists\setting\pay;
use app\admin\lists\BaseAdminDataLists;
use app\common\model\pay\PayConfig;
/**
* 支付配置列表
* Class PayConfigLists
* @package app\adminapi\lists\setting\pay
*/
class PayConfigLists extends BaseAdminDataLists
{
/**
* @notes 获取列表
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:15
*/
public function lists(): array
{
$lists = PayConfig::field('id,name,pay_way,icon,sort')
->append(['pay_way_name'])
->order('sort','asc')
->select()
->toArray();
return $lists;
}
/**
* @notes 获取数量
* @return int
* @author 段誉
* @date 2023/2/23 16:15
*/
public function count(): int
{
return PayConfig::count();
}
}

View File

@ -0,0 +1,96 @@
<?php
namespace app\admin\logic\order;
use app\common\model\order\Cart;
use app\common\logic\BaseLogic;
use think\facade\Db;
/**
* 购物车表逻辑
* Class CartLogic
* @package app\admin\logic\order
*/
class CartLogic extends BaseLogic
{
/**
* @notes 添加购物车表
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:37
*/
public static function add(array $params): bool
{
Db::startTrans();
try {
Cart::create([
'cart_num' => $params['cart_num'],
'goods_id' => $params['goods_id'],
'uid' => $params['uid'],
'source' => $params['source']??0,
'mer_id' => $params['mer_id']??0,
'staff_id' => $params['staff_id']??0,
'is_new' => $params['is_new']??0,
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 编辑购物车表
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:37
*/
public static function edit(array $params): bool
{
Db::startTrans();
try {
Cart::where(['uid'=>$params['uid'],'goods_id'=>$params['goods_id']])->inc('cart_num',$params['cart_num'])->update();
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 删除购物车表
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:37
*/
public static function delete(array $params): bool
{
return Cart::destroy($params['cart_id']);
}
/**
* @notes 获取购物车表详情
* @param $params
* @return array
* @author likeadmin
* @date 2024/04/24 10:37
*/
public static function detail($params): array
{
return Cart::findOrEmpty($params['cart_id'])->toArray();
}
}

View File

@ -0,0 +1,122 @@
<?php
namespace app\admin\logic\retail;
use app\common\model\retail\Cashierclass;
use app\common\logic\BaseLogic;
use think\facade\Db;
/**
* 零售订单逻辑
* Class CashierclassLogic
* @package app\admin\logic\retail
*/
class CashierclassLogic extends BaseLogic
{
/**
* @notes 添加零售订单
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 09:57
*/
public static function add(array $params): bool
{
Db::startTrans();
try {
Cashierclass::create([
'merchant' => $params['merchant'],
'store_id' => $params['store_id'],
'customer' => $params['customer'],
'time' => $params['time'],
'number' => $params['number'],
'total' => $params['total'],
'deduction_price' => $params['deduction_price'],
'actual' => $params['actual'],
'money' => $params['money'],
'pay_type' => $params['pay_type'],
'data' => $params['data'],
'type' => $params['type'],
'auditinguser' => $params['auditinguser'],
'auditingtime' => strtotime($params['auditingtime']),
'more' => $params['more']
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 编辑零售订单
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 09:57
*/
public static function edit(array $params): bool
{
Db::startTrans();
try {
Cashierclass::where('id', $params['id'])->update([
'merchant' => $params['merchant'],
'store_id' => $params['store_id'],
'customer' => $params['customer'],
'time' => $params['time'],
'number' => $params['number'],
'total' => $params['total'],
'deduction_price' => $params['deduction_price'],
'actual' => $params['actual'],
'money' => $params['money'],
'pay_type' => $params['pay_type'],
'data' => $params['data'],
'type' => $params['type'],
'auditinguser' => $params['auditinguser'],
'auditingtime' => strtotime($params['auditingtime']),
'more' => $params['more']
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 删除零售订单
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 09:57
*/
public static function delete(array $params): bool
{
return Cashierclass::destroy($params['id']);
}
/**
* @notes 获取零售订单详情
* @param $params
* @return array
* @author likeadmin
* @date 2024/04/24 09:57
*/
public static function detail($params): array
{
return Cashierclass::findOrEmpty($params['id'])->toArray();
}
}

View File

@ -0,0 +1,94 @@
<?php
namespace app\admin\logic\retail;
use app\common\model\retail\Cashierinfo;
use app\common\logic\BaseLogic;
use think\facade\Db;
/**
* 零售订单详情逻辑
* Class CashierinfoLogic
* @package app\admin\logic\retail
*/
class CashierinfoLogic extends BaseLogic
{
/**
* @notes 添加零售订单详情
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:11
*/
public static function add(array $params): bool
{
Db::startTrans();
try {
Cashierinfo::create([
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 编辑零售订单详情
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:11
*/
public static function edit(array $params): bool
{
Db::startTrans();
try {
Cashierinfo::where('id', $params['id'])->update([
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 删除零售订单详情
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/24 10:11
*/
public static function delete(array $params): bool
{
return Cashierinfo::destroy($params['id']);
}
/**
* @notes 获取零售订单详情详情
* @param $params
* @return array
* @author likeadmin
* @date 2024/04/24 10:11
*/
public static function detail($params): array
{
return Cashierinfo::findOrEmpty($params['id'])->toArray();
}
}

View File

@ -0,0 +1,91 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\admin\logic\setting\pay;
use app\common\enum\PayEnum;
use app\common\logic\BaseLogic;
use app\common\model\pay\PayConfig;
/**
* 支付配置
* Class PayConfigLogic
* @package app\adminapi\logic\setting\pay
*/
class PayConfigLogic extends BaseLogic
{
/**
* @notes 设置配置
* @param $params
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:16
*/
public static function setConfig($params)
{
$payConfig = PayConfig::find($params['id']);
$config = '';
if ($payConfig['pay_way'] == PayEnum::WECHAT_PAY) {
$config = [
'interface_version' => $params['config']['interface_version'],
'merchant_type' => $params['config']['merchant_type'],
'mch_id' => $params['config']['mch_id'],
'pay_sign_key' => $params['config']['pay_sign_key'],
'apiclient_cert' => $params['config']['apiclient_cert'],
'apiclient_key' => $params['config']['apiclient_key'],
];
}
if ($payConfig['pay_way'] == PayEnum::ALI_PAY) {
$config = [
'mode' => $params['config']['mode'],
'merchant_type' => $params['config']['merchant_type'],
'app_id' => $params['config']['app_id'],
'private_key' => $params['config']['private_key'],
'ali_public_key' => $params['config']['ali_public_key'],
];
}
$payConfig->name = $params['name'];
$payConfig->icon = $params['icon'];
$payConfig->sort = $params['sort'];
$payConfig->config = $config;
$payConfig->remark = $params['remark'] ?? '';
return $payConfig->save();
}
/**
* @notes 获取配置
* @param $params
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:16
*/
public static function getConfig($params)
{
$payConfig = PayConfig::find($params['id'])->toArray();
$payConfig['domain'] = '//'.request()->host();;
return $payConfig;
}
}

View File

@ -0,0 +1,111 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\admin\logic\setting\pay;
use app\common\enum\PayEnum;
use app\common\enum\YesNoEnum;
use app\common\logic\BaseLogic;
use app\common\model\pay\PayConfig;
use app\common\model\pay\PayWay;
use app\common\service\FileService;
/**
* 支付方式
* Class PayWayLogic
* @package app\adminapi\logic\setting\pay
*/
class PayWayLogic extends BaseLogic
{
/**
* @notes 获取支付方式
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:25
*/
public static function getPayWay()
{
$payWay = PayWay::select()->append(['pay_way_name'])
->toArray();
if (empty($payWay)) {
return [];
}
$lists = [];
for ($i = 1; $i <= max(array_column($payWay, 'scene')); $i++) {
foreach ($payWay as $val) {
if ($val['scene'] == $i) {
$val['icon'] = FileService::getFileUrl(PayConfig::where('id', $val['pay_config_id'])->value('icon'));
$lists[$i][] = $val;
}
}
}
return $lists;
}
/**
* @notes 设置支付方式
* @param $params
* @return bool|string
* @throws \Exception
* @author 段誉
* @date 2023/2/23 16:26
*/
public static function setPayWay($params)
{
$payWay = new PayWay;
$data = [];
foreach ($params as $key => $value) {
$isDefault = array_column($value, 'is_default');
$isDefaultNum = array_count_values($isDefault);
$status = array_column($value, 'status');
$sceneName = PayEnum::getPaySceneDesc($key);
if (!in_array(YesNoEnum::YES, $isDefault)) {
return $sceneName . '支付场景缺少默认支付';
}
if ($isDefaultNum[YesNoEnum::YES] > 1) {
return $sceneName . '支付场景的默认值只能存在一个';
}
if (!in_array(YesNoEnum::YES, $status)) {
return $sceneName . '支付场景至少开启一个支付状态';
}
foreach ($value as $val) {
$result = PayWay::where('id', $val['id'])->findOrEmpty();
if ($result->isEmpty()) {
continue;
}
if ($val['is_default'] == YesNoEnum::YES && $val['status'] == YesNoEnum::NO) {
return $sceneName . '支付场景的默认支付未开启支付状态';
}
$data[] = [
'id' => $val['id'],
'is_default' => $val['is_default'],
'status' => $val['status'],
];
}
}
$payWay->saveAll($data);
return true;
}
}

View File

@ -16,6 +16,9 @@ namespace app\admin\logic\user;
use app\common\enum\user\UserTerminalEnum;
use app\common\logic\BaseLogic;
use app\common\model\user\User;
use think\facade\Db;
use app\common\service\FileService;
use Webman\Config;
/**
* 用户逻辑层
@ -24,6 +27,73 @@ use app\common\model\user\User;
*/
class UserLogic extends BaseLogic
{
/**
* @notes 添加用户列表
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/25 10:20
*/
public static function add(array $params): bool
{
$passwordSalt = Config::get('project.unique_identification');
$password = create_password($params['password'], $passwordSalt);
$defaultAvatar = config('project.default_image.admin_avatar');
$avatar = !empty($params['avatar']) ? FileService::setFileUrl($params['avatar']) : $defaultAvatar;
Db::startTrans();
try {
User::create([
'avatar' => $avatar,
'real_name' => $params['real_name'],
'nickname' => $params['nickname'],
'account' => $params['account'],
'password' => $password,
'mobile' => $params['mobile'],
'sex' => $params['sex'],
'is_disable' => $params['is_disable'],
'user_money' => $params['user_money']
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 编辑用户列表
* @param array $params
* @return bool
* @author likeadmin
* @date 2024/04/25 10:20
*/
public static function edit(array $params): bool
{
Db::startTrans();
try {
User::where('id', $params['id'])->update([
'avatar' => $params['avatar'],
'real_name' => $params['real_name'],
'nickname' => $params['nickname'],
'account' => $params['account'],
'password' => $params['password'],
'mobile' => $params['mobile'],
'sex' => $params['sex'],
'is_disable' => $params['is_disable'],
'user_money' => $params['user_money']
]);
Db::commit();
return true;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 用户详情
@ -62,5 +132,4 @@ class UserLogic extends BaseLogic
$params['field'] => $params['value']
]);
}
}
}

View File

@ -0,0 +1,84 @@
<?php
namespace app\admin\validate\order;
use app\common\validate\BaseValidate;
/**
* 购物车表验证器
* Class CartValidate
* @package app\admin\validate\order
*/
class CartValidate extends BaseValidate
{
/**
* 设置校验规则
* @var string[]
*/
protected $rule = [
'cart_num' => 'require|number',
'goods_id' => 'require|number',
];
/**
* 参数描述
* @var string[]
*/
protected $field = [
'cart_num' => '数量',
'goods_id' => '商品id',
];
/**
* @notes 添加场景
* @return CartValidate
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function sceneAdd()
{
return $this->only(['cart_num','goods_id']);
}
/**
* @notes 编辑场景
* @return CartValidate
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function sceneEdit()
{
return $this->only(['']);
}
/**
* @notes 删除场景
* @return CartValidate
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function sceneDelete()
{
return $this->only(['']);
}
/**
* @notes 详情场景
* @return CartValidate
* @author likeadmin
* @date 2024/04/24 10:37
*/
public function sceneDetail()
{
return $this->only(['']);
}
}

View File

@ -0,0 +1,98 @@
<?php
namespace app\admin\validate\retail;
use app\common\validate\BaseValidate;
/**
* 零售订单验证器
* Class CashierclassValidate
* @package app\admin\validate\retail
*/
class CashierclassValidate extends BaseValidate
{
/**
* 设置校验规则
* @var string[]
*/
protected $rule = [
'id' => 'require',
'merchant' => 'require',
'customer' => 'require',
'time' => 'require',
'number' => 'require',
'total' => 'require',
'actual' => 'require',
'money' => 'require',
'pay_type' => 'require',
];
/**
* 参数描述
* @var string[]
*/
protected $field = [
'id' => 'id',
'merchant' => '所属商户',
'customer' => '客户ID',
'time' => '单据时间',
'number' => '单据编号',
'total' => '单据金额',
'actual' => '实际金额',
'money' => '实收金额',
'pay_type' => '支付方式 0余额 1微信 2小程序 3 h5 4支付宝 5 支付宝扫码 6 微信扫码 8先货后款 10对公转账',
];
/**
* @notes 添加场景
* @return CashierclassValidate
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function sceneAdd()
{
return $this->only(['merchant','customer','time','number','total','actual','money','pay_type']);
}
/**
* @notes 编辑场景
* @return CashierclassValidate
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function sceneEdit()
{
return $this->only(['id','merchant','customer','time','number','total','actual','money','pay_type']);
}
/**
* @notes 删除场景
* @return CashierclassValidate
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function sceneDelete()
{
return $this->only(['id']);
}
/**
* @notes 详情场景
* @return CashierclassValidate
* @author likeadmin
* @date 2024/04/24 09:57
*/
public function sceneDetail()
{
return $this->only(['id']);
}
}

View File

@ -0,0 +1,82 @@
<?php
namespace app\admin\validate\retail;
use app\common\validate\BaseValidate;
/**
* 零售订单详情验证器
* Class CashierinfoValidate
* @package app\admin\validate\retail
*/
class CashierinfoValidate extends BaseValidate
{
/**
* 设置校验规则
* @var string[]
*/
protected $rule = [
'id' => 'require',
];
/**
* 参数描述
* @var string[]
*/
protected $field = [
'id' => 'id',
];
/**
* @notes 添加场景
* @return CashierinfoValidate
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function sceneAdd()
{
return $this->remove('id', true);
}
/**
* @notes 编辑场景
* @return CashierinfoValidate
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function sceneEdit()
{
return $this->only(['id']);
}
/**
* @notes 删除场景
* @return CashierinfoValidate
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function sceneDelete()
{
return $this->only(['id']);
}
/**
* @notes 详情场景
* @return CashierinfoValidate
* @author likeadmin
* @date 2024/04/24 10:11
*/
public function sceneDetail()
{
return $this->only(['id']);
}
}

View File

@ -0,0 +1,128 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\admin\validate\setting;
use app\common\enum\PayEnum;
use app\common\model\pay\PayConfig;
use app\common\validate\BaseValidate;
class PayConfigValidate extends BaseValidate
{
protected $rule = [
'id' => 'require',
'name' => 'require|checkName',
'icon' => 'require',
'sort' => 'require|number|max:5',
'config' => 'require|checkConfig',
];
protected $message = [
'id.require' => 'id不能为空',
'name.require' => '支付名称不能为空',
'icon.require' => '支付图标不能为空',
'sort.require' => '排序不能为空',
'sort,number' => '排序必须是纯数字',
'sort.max' => '排序最大不能超过五位数',
'config.require' => '支付参数缺失',
];
public function sceneGet()
{
return $this->only(['id']);
}
/**
* @notes 校验支付配置记录
* @param $value
* @param $rule
* @param $data
* @return bool|string
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2023/2/23 16:19
*/
public function checkConfig($config, $rule, $data)
{
$result = PayConfig::where('id', $data['id'])->find();
if (empty($result)) {
return '支付方式不存在';
}
if ($result['pay_way'] == PayEnum::WECHAT_PAY) {
if (empty($config['interface_version'])) {
return '微信支付接口版本不能为空';
}
if (empty($config['merchant_type'])) {
return '商户类型不能为空';
}
if (empty($config['mch_id'])) {
return '微信支付商户号不能为空';
}
if (empty($config['pay_sign_key'])) {
return '商户API密钥不能为空';
}
if (empty($config['apiclient_cert'])) {
return '微信支付证书不能为空';
}
if (empty($config['apiclient_key'])) {
return '微信支付证书密钥不能为空';
}
}
if ($result['pay_way'] == PayEnum::ALI_PAY) {
if (empty($config['mode'])) {
return '模式不能为空';
}
if (empty($config['merchant_type'])) {
return '商户类型不能为空';
}
if (empty($config['app_id'])) {
return '应用ID不能为空';
}
if (empty($config['private_key'])) {
return '应用私钥不能为空';
}
if (empty($config['ali_public_key'])) {
return '支付宝公钥不能为空';
}
}
return true;
}
/**
* @notes 校验支付名
* @param $value
* @param $rule
* @param $data
* @return bool|string
* @author 段誉
* @date 2023/2/23 16:19
*/
public function checkName($value, $rule, $data)
{
$result = PayConfig::where('name', $value)
->where('id', '<>', $data['id'])
->findOrEmpty();
if (!$result->isEmpty()) {
return '支付名称已存在';
}
return true;
}
}

View File

@ -1,110 +1,92 @@
<?php
namespace app\admin\validate\user;
use app\common\model\user\User;
use app\common\validate\BaseValidate;
/**
* 用户列表验证器
* Class UserValidate
* @package app\admin\validate\user
*/
class UserValidate extends BaseValidate
{
protected $rule = [
'id' => 'require|checkUser',
'field' => 'require|checkField',
'value' => 'require',
];
protected $message = [
'id.require' => '请选择用户',
'field.require' => '请选择操作',
'value.require' => '请输入内容',
/**
* 设置校验规则
* @var string[]
*/
protected $rule = [
'id' => 'require',
'real_name' => 'require',
'nickname' => 'require',
'account' => 'require',
'password' => 'require',
'mobile' => 'require',
];
/**
* @notes 详情场景
* @return \app\admin\validate\user\UserValidate
* @author 乔峰
* @date 2022/9/22 16:35
* 参数描述
* @var string[]
*/
public function sceneDetail()
protected $field = [
'id' => 'id',
'real_name' => '真实姓名',
'nickname' => '用户昵称',
'account' => '用户账号',
'password' => '用户密码',
'mobile' => '用户电话',
];
/**
* @notes 添加场景
* @return UserValidate
* @author likeadmin
* @date 2024/04/25 10:20
*/
public function sceneAdd()
{
return $this->only(['real_name','nickname','account','password','mobile']);
}
/**
* @notes 编辑场景
* @return UserValidate
* @author likeadmin
* @date 2024/04/25 10:20
*/
public function sceneEdit()
{
return $this->only(['id','real_name','nickname','account','password','mobile']);
}
/**
* @notes 删除场景
* @return UserValidate
* @author likeadmin
* @date 2024/04/25 10:20
*/
public function sceneDelete()
{
return $this->only(['id']);
}
/**
* @notes 用户信息校验
* @param $value
* @param $rule
* @param $data
* @return bool|string
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 乔峰
* @date 2022/9/22 17:03
* @notes 详情场景
* @return UserValidate
* @author likeadmin
* @date 2024/04/25 10:20
*/
public function checkUser($value, $rule, $data)
public function sceneDetail()
{
$userIds = is_array($value) ? $value : [$value];
foreach ($userIds as $item) {
if (!User::find($item)) {
return '用户不存在!';
}
}
return true;
return $this->only(['id']);
}
/**
* @notes 校验是否可更新信息
* @param $value
* @param $rule
* @param $data
* @return bool|string
* @author 乔峰
* @date 2022/9/22 16:37
*/
public function checkField($value, $rule, $data)
{
$allowField = ['account', 'sex', 'mobile', 'real_name'];
if (!in_array($value, $allowField)) {
return '用户信息不允许更新';
}
switch ($value) {
case 'account':
//验证手机号码是否存在
$account = User::where([
['id', '<>', $data['id']],
['account', '=', $data['value']]
])->findOrEmpty();
if (!$account->isEmpty()) {
return '账号已被使用';
}
break;
case 'mobile':
if (false == $this->validate($data['value'], 'mobile', $data)) {
return '手机号码格式错误';
}
//验证手机号码是否存在
$mobile = User::where([
['id', '<>', $data['id']],
['mobile', '=', $data['value']]
])->findOrEmpty();
if (!$mobile->isEmpty()) {
return '手机号码已存在';
}
break;
}
return true;
}
}

View File

@ -0,0 +1,18 @@
<?php
namespace app\api\controller;
use app\common\controller\BaseLikeController;
class BaseApiController extends BaseLikeController
{
protected int $userId = 0;
protected array $userInfo = [];
public function initialize()
{
if (isset($this->request->userInfo) && $this->request->userInfo) {
$this->userInfo = $this->request->userInfo;
$this->userId = $this->request->userInfo['user_id'];
}
}
}

View File

@ -4,12 +4,14 @@ namespace app\api\controller;
use app\admin\validate\tools\GenerateTableValidate;
use app\admin\logic\tools\GeneratorLogic;
use app\common\controller\BaseLikeAdminController;
class IndexController extends BaseLikeAdminController
class IndexController extends BaseApiController
{
public $notNeedLogin = ['index'];
public function index()
{
d(2);
return json(['msg' =>create_password(123456, '11d3')]);
}
@ -19,7 +21,6 @@ class IndexController extends BaseLikeAdminController
public function download()
{
$params = (new GenerateTableValidate())->goCheck('download');
var_dump($params);
$result = GeneratorLogic::download($params['file']);
if (false === $result) {
return $this->fail(GeneratorLogic::getError() ?: '下载失败');

View File

@ -0,0 +1,22 @@
<?php
namespace app\api\controller;
use app\api\validate\LoginAccountValidate;
use app\api\logic\LoginLogic;
class LoginController extends BaseApiController
{
public $notNeedLogin = ['account'];
public function account()
{
$params=$this->request->post();
$params = (new LoginAccountValidate())->post()->goCheck();
$result = LoginLogic::login($params);
if (false === $result) {
return $this->fail(LoginLogic::getError());
}
return $this->data($result);
}
}

View File

@ -0,0 +1,28 @@
<?php
namespace app\api\controller\order;
use app\admin\logic\order\CartLogic;
use app\admin\validate\order\CartValidate;
use app\api\controller\BaseApiController;
use app\common\model\order\Cart;
class CartController extends BaseApiController
{
public function create(){
$params = (new CartValidate())->get()->goCheck('add');
$params['uid']=1;
$result=Cart::where(['uid'=>$params['uid'],'goods_id'=>$params['goods_id'],'is_fail'=>0,'is_pay'=>0])->find();
if($result){
$res=CartLogic::edit($params);
}else{
$res=CartLogic::add($params);
}
if($res){
return $this->success('添加成功');
}else{
return $this->fail(CartLogic::getError());
}
}
}

View File

@ -0,0 +1,50 @@
<?php
namespace app\api\controller\order;
use app\admin\logic\order\CartLogic;
use app\api\logic\order\OrderLogic;
use app\api\controller\BaseApiController;
use app\common\logic\order\RetailOrderLogic;
use app\common\enum\PayEnum;
use app\common\model\order\Cart;
use app\common\model\user\User;
/**
* 购货单控制器
*/
class PurchaseOrderController extends BaseApiController
{
public function checkOrder(){
$params=$this->request->get();
$params['store_id']=1;
$res=OrderLogic::cartIdByPurchaseOrderInfo($params);
return $this->data($res);
}
/**
* @notes 创建购货订单
*/
public function createOrder(){
$user=User::where('id',$this->request->userId)->find();
$cartId = (Array)$this->request->post('cart_id', []);
$addressId = (int)$this->request->post('address_id');
$pay_type = (int)$this->request->post('pay_type');
$auth_code = $this->request->post('auth_code'); //微信支付条码
$params=$this->request->post();
$order=OrderLogic::createOrder($cartId,$addressId,null,$params);
if(is_object($order)){
if($pay_type==PayEnum::BALANCE_PAY){
$user=User::where('id',$this->request->userId)->find();
RetailOrderLogic::payBalance($user,$order);
return $this->success('支付成功');
}
return $this->data(['order_id'=>$order->id]);
}else{
return $this->fail(OrderLogic::getError());
}
}
}

View File

@ -0,0 +1,49 @@
<?php
namespace app\api\controller\order;
use app\admin\logic\order\CartLogic;
use app\api\logic\order\OrderLogic;
use app\api\controller\BaseApiController;
use app\common\logic\order\RetailOrderLogic;
use app\common\enum\PayEnum;
use app\common\model\order\Cart;
use app\common\model\user\User;
class RetailOrderController extends BaseApiController
{
public function checkOrder(){
$cartId = (Array)$this->request->post('cart_id', []);
$addressId = (int)$this->request->post('address_id');
$pay_type = (int)$this->request->post('pay_type');
$auth_code = $this->request->post('auth_code'); //微信支付条码
$params=$this->request->post();
$res=OrderLogic::cartIdByOrderInfo($cartId,$addressId,null,$params);
return $this->data($res);
}
/**
* @notes 创建零售订单
*/
public function createOrder(){
$user=User::where('id',$this->request->userId)->find();
$cartId = (Array)$this->request->post('cart_id', []);
$addressId = (int)$this->request->post('address_id');
$pay_type = (int)$this->request->post('pay_type');
$auth_code = $this->request->post('auth_code'); //微信支付条码
$params=$this->request->post();
$order=OrderLogic::createOrder($cartId,$addressId,null,$params);
if(is_object($order)){
if($pay_type==PayEnum::BALANCE_PAY){
$user=User::where('id',$this->request->userId)->find();
RetailOrderLogic::payBalance($user,$order);
return $this->success('支付成功');
}
return $this->data(['order_id'=>$order->id]);
}else{
return $this->fail(OrderLogic::getError());
}
}
}

View File

@ -0,0 +1,34 @@
<?php
namespace app\api\http\middleware;
use app\common\exception\ControllerExtendException;
use app\api\controller\BaseApiController;
use think\exception\ClassNotFoundException;
use app\common\exception\HttpException;
use Webman\Http\Request;
use Webman\Http\Response;
use Webman\MiddlewareInterface;
class InitMiddleware implements MiddlewareInterface
{
public function process(Request $request, callable $handler): Response
{
//获取控制器
try {
$controller = str_replace('.', '\\', $request->controller);
$controllerClass = new $controller;
if (($controllerClass instanceof BaseApiController) === false) {
throw new ControllerExtendException($controller, '404');
}
} catch (ClassNotFoundException $e) {
throw new HttpException(404, 'controller not exists:' . $e->getClass());
}
//创建控制器对象
$request->controllerObject = new $controller;
return $handler($request);
}
}

View File

@ -0,0 +1,63 @@
<?php
namespace app\api\http\middleware;
use app\common\cache\UserTokenCache;
use app\common\service\JsonService;
use app\api\service\UserTokenService;
use Webman\Config;
use Webman\Http\Request;
use Webman\Http\Response;
use Webman\MiddlewareInterface;
class LoginMiddleware implements MiddlewareInterface
{
/**
* @notes 登录验证
* @param $request
* @param \Closure $next
* @return mixed|\think\response\Json
* @author 令狐冲
* @date 2021/7/1 17:33
*/
public function process(Request $request, callable $handler): Response
{
$token = $request->header('token');
//判断接口是否免登录
$isNotNeedLogin = $request->controllerObject->isNotNeedLogin();
//不直接判断$isNotNeedLogin结果使不需要登录的接口通过为了兼容某些接口可以登录或不登录访问
if (empty($token) && !$isNotNeedLogin) {
//没有token并且该地址需要登录才能访问, 指定show为0前端不弹出此报错
return JsonService::fail('请求参数缺token', [], 0, 0);
}
$userInfo = (new UserTokenCache())->getUserInfo($token);
if (empty($userInfo) && !$isNotNeedLogin) {
//token过期无效并且该地址需要登录才能访问
return JsonService::fail('登录超时,请重新登录', [], -1, 0);
}
//token临近过期自动续期
if ($userInfo) {
//获取临近过期自动续期时长
$beExpireDuration = Config::get('project.user_token.be_expire_duration');
//token续期
if (time() > ($userInfo['expire_time'] - $beExpireDuration)) {
$result = UserTokenService::overtimeToken($token);
//续期失败(数据表被删除导致)
if (empty($result)) {
return JsonService::fail('登录过期', [], -1);
}
}
}
//给request赋值用于控制器
$request->userInfo = $userInfo;
$request->userId = $userInfo['user_id'] ?? 0;
return $handler($request);
}
}

View File

@ -0,0 +1,431 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\api\logic;
use app\common\cache\WebScanLoginCache;
use app\common\logic\BaseLogic;
use app\api\service\{UserTokenService, WechatUserService};
use app\common\enum\{LoginEnum, user\UserTerminalEnum, YesNoEnum};
use app\common\service\{
ConfigService,
FileService,
};
use app\common\model\user\{User, UserAuth};
use app\common\service\wechat\WeChatMnpService;
use think\facade\Db;
use Webman\Config;
/**
* 登录逻辑
* Class LoginLogic
* @package app\api\logic
*/
class LoginLogic extends BaseLogic
{
/**
* @notes 账号密码注册
* @param array $params
* @return bool
* @author 段誉
* @date 2022/9/7 15:37
*/
public static function register(array $params)
{
try {
$userSn = User::createUserSn();
$passwordSalt = Config::get('project.unique_identification');
$password = create_password($params['password'], $passwordSalt);
$avatar = ConfigService::get('default_image', 'user_avatar');
User::create([
'sn' => $userSn,
'avatar' => $avatar,
'nickname' => '用户' . $userSn,
'account' => $params['account'],
'password' => $password,
'channel' => $params['channel'],
]);
return true;
} catch (\Exception $e) {
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 账号/手机号登录,手机号验证码
* @param $params
* @return array|false
* @author 段誉
* @date 2022/9/6 19:26
*/
public static function login($params)
{
try {
// 账号/手机号 密码登录
$where = ['account|mobile' => $params['account']];
if ($params['scene'] == LoginEnum::MOBILE_CAPTCHA) {
//手机验证码登录
$where = ['mobile' => $params['account']];
}
$user = User::where($where)->findOrEmpty();
if ($user->isEmpty()) {
throw new \Exception('用户不存在');
}
//更新登录信息
$user->login_time = time();
$user->login_ip = request()->getLocalIp();
$user->save();
//设置token
$userInfo = UserTokenService::setToken($user->id, $params['terminal']);
//返回登录信息
$avatar = $user->avatar ?: Config::get('project.default_image.user_avatar');
$avatar = FileService::getFileUrl($avatar);
return [
'nickname' => $userInfo['nickname'],
// 'sn' => $userInfo['sn'],
'mobile' => $userInfo['mobile'],
'avatar' => $avatar,
'token' => $userInfo['token'],
];
} catch (\Exception $e) {
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 退出登录
* @param $userInfo
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 17:56
*/
public static function logout($userInfo)
{
//token不存在不注销
if (!isset($userInfo['token'])) {
return false;
}
//设置token过期
return UserTokenService::expireToken($userInfo['token']);
}
/**
* @notes 获取微信请求code的链接
* @param string $url
* @return string
* @author 段誉
* @date 2022/9/20 19:47
*/
public static function codeUrl(string $url)
{
// return (new WeChatOaService())->getCodeUrl($url);
}
/**
* @notes 公众号登录
* @param array $params
* @return array|false
* @throws \GuzzleHttp\Exception\GuzzleException
* @author 段誉
* @date 2022/9/20 19:47
*/
public static function oaLogin(array $params)
{
// Db::startTrans();
// try {
// //通过code获取微信 openid
// $response = (new WeChatOaService())->getOaResByCode($params['code']);
// $userServer = new WechatUserService($response, UserTerminalEnum::WECHAT_OA);
// $userInfo = $userServer->getResopnseByUserInfo()->authUserLogin()->getUserInfo();
// // 更新登录信息
// self::updateLoginInfo($userInfo['id']);
// Db::commit();
// return $userInfo;
// } catch (\Exception $e) {
// Db::rollback();
// self::$error = $e->getMessage();
// return false;
// }
}
/**
* @notes 小程序-静默登录
* @param array $params
* @return array|false
* @author 段誉
* @date 2022/9/20 19:47
*/
public static function silentLogin(array $params)
{
try {
//通过code获取微信 openid
$response = (new WeChatMnpService())->getMnpResByCode($params['code']);
$userServer = new WechatUserService($response, UserTerminalEnum::WECHAT_MMP);
$userInfo = $userServer->getResopnseByUserInfo('silent')->getUserInfo();
if (!empty($userInfo)) {
// 更新登录信息
self::updateLoginInfo($userInfo['id']);
}
return $userInfo;
} catch (\Exception $e) {
self::$error = $e->getMessage();
return false;
}
}
/**
* @notes 小程序-授权登录
* @param array $params
* @return array|false
* @author 段誉
* @date 2022/9/20 19:47
*/
public static function mnpLogin(array $params)
{
Db::startTrans();
try {
//通过code获取微信 openid
$response = (new WeChatMnpService())->getMnpResByCode($params['code']);
$userServer = new WechatUserService($response, UserTerminalEnum::WECHAT_MMP);
$userInfo = $userServer->getResopnseByUserInfo()->authUserLogin()->getUserInfo();
// 更新登录信息
self::updateLoginInfo($userInfo['id']);
Db::commit();
return $userInfo;
} catch (\Exception $e) {
Db::rollback();
self::$error = $e->getMessage();
return false;
}
}
/**
* @notes 更新登录信息
* @param $userId
* @throws \Exception
* @author 段誉
* @date 2022/9/20 19:46
*/
public static function updateLoginInfo($userId)
{
$user = User::findOrEmpty($userId);
if ($user->isEmpty()) {
throw new \Exception('用户不存在');
}
$time = time();
$user->login_time = $time;
$user->login_ip = request()->getLocalIp();
$user->update_time = $time;
$user->save();
}
/**
* @notes 小程序端绑定微信
* @param array $params
* @return bool
* @author 段誉
* @date 2022/9/20 19:46
*/
public static function mnpAuthLogin(array $params)
{
// try {
// //通过code获取微信openid
// $response = (new WeChatMnpService())->getMnpResByCode($params['code']);
// $response['user_id'] = $params['user_id'];
// $response['terminal'] = UserTerminalEnum::WECHAT_MMP;
// return self::createAuth($response);
// } catch (\Exception $e) {
// self::$error = $e->getMessage();
// return false;
// }
}
/**
* @notes 公众号端绑定微信
* @param array $params
* @return bool
* @throws \GuzzleHttp\Exception\GuzzleException
* @author 段誉
* @date 2022/9/16 10:43
*/
public static function oaAuthLogin(array $params)
{
// try {
// //通过code获取微信openid
// $response = (new WeChatOaService())->getOaResByCode($params['code']);
// $response['user_id'] = $params['user_id'];
// $response['terminal'] = UserTerminalEnum::WECHAT_OA;
// return self::createAuth($response);
// } catch (\Exception $e) {
// self::$error = $e->getMessage();
// return false;
// }
}
/**
* @notes 生成授权记录
* @param $response
* @return bool
* @throws \Exception
* @author 段誉
* @date 2022/9/16 10:43
*/
public static function createAuth($response)
{
//先检查openid是否有记录
$isAuth = UserAuth::where('openid', '=', $response['openid'])->findOrEmpty();
if (!$isAuth->isEmpty()) {
throw new \Exception('该微信已被绑定');
}
if (isset($response['unionid']) && !empty($response['unionid'])) {
//在用unionid找记录防止生成两个账号同个unionid的问题
$userAuth = UserAuth::where(['unionid' => $response['unionid']])
->findOrEmpty();
if (!$userAuth->isEmpty() && $userAuth->user_id != $response['user_id']) {
throw new \Exception('该微信已被绑定');
}
}
//如果没有授权,直接生成一条微信授权记录
UserAuth::create([
'user_id' => $response['user_id'],
'openid' => $response['openid'],
'unionid' => $response['unionid'] ?? '',
'terminal' => $response['terminal'],
]);
return true;
}
/**
* @notes 获取扫码登录地址
* @return array|false
* @author 段誉
* @date 2022/10/20 18:23
*/
public static function getScanCode($redirectUri)
{
// try {
// $config = WeChatConfigService::getOpConfig();
// $appId = $config['app_id'];
// $redirectUri = UrlEncode($redirectUri);
// // 设置有效时间标记状态, 超时扫码不可登录
// $state = MD5(time().rand(10000, 99999));
// (new WebScanLoginCache())->setScanLoginState($state);
// // 扫码地址
// $url = WeChatRequestService::getScanCodeUrl($appId, $redirectUri, $state);
// return ['url' => $url];
// } catch (\Exception $e) {
// self::$error = $e->getMessage();
// return false;
// }
}
/**
* @notes 网站扫码登录
* @param $params
* @return array|false
* @author 段誉
* @date 2022/10/21 10:28
*/
public static function scanLogin($params)
{
// Db::startTrans();
// try {
// // 通过code 获取 access_token,openid,unionid等信息
// $userAuth = WeChatRequestService::getUserAuthByCode($params['code']);
// if (empty($userAuth['openid']) || empty($userAuth['access_token'])) {
// throw new \Exception('获取用户授权信息失败');
// }
// // 获取微信用户信息
// $response = WeChatRequestService::getUserInfoByAuth($userAuth['access_token'], $userAuth['openid']);
// // 生成用户或更新用户信息
// $userServer = new WechatUserService($response, UserTerminalEnum::PC);
// $userInfo = $userServer->getResopnseByUserInfo()->authUserLogin()->getUserInfo();
// // 更新登录信息
// self::updateLoginInfo($userInfo['id']);
// Db::commit();
// return $userInfo;
// } catch (\Exception $e) {
// Db::rollback();
// self::$error = $e->getMessage();
// return false;
// }
}
/**
* @notes 更新用户信息
* @param $params
* @param $userId
* @return User
* @author 段誉
* @date 2023/2/22 11:19
*/
public static function updateUser($params, $userId)
{
return User::where(['id' => $userId])->update([
'nickname' => $params['nickname'],
'avatar' => FileService::setFileUrl($params['avatar']),
'is_new_user' => YesNoEnum::NO
]);
}
}

View File

@ -0,0 +1,176 @@
<?php
namespace app\api\logic\order;
use app\common\logic\BaseLogic;
use app\common\model\goods\Goods;
use app\common\model\order\Cart;
use app\common\model\retail\Cashierclass;
use app\common\model\retail\Cashierinfo;
use think\facade\Db;
/**
* 订单逻辑
* Class OrderLogic
* @package app\api\logic\order
*/
class OrderLogic extends BaseLogic
{
/**
* @notes 获取购物车商品信息
* @param $params
* @return array
*/
static public function cartIdByOrderInfo($cartId, $addressId, $user = null, $params = []): array
{
$where = ['is_pay' => 0, 'is_fail' => 0];
$cart_select = Cart::whereIn('cart_id', $cartId)->where($where)->field('goods_id as goods,cart_num')->select()->toArray();
if (empty($cart_select)) {
self::setError('购物车为空');
return false;
}
/** 计算价格 */
foreach ($cart_select as $k => $v) {
$sell = Goods::where(['id' => $v['goods']])->value('sell');
$cart_select[$k]['total'] = bcmul($v['cart_num'], $sell, 2);
$cart_select[$k]['price'] = $sell;
}
$order = [
'time' => time(),
'number' => static::getNewOrderId('PF'),
'total' => array_sum(array_column($cart_select, 'total')),
'pay_type' => $params['pay_type'],
'cart_id'=>explode(',',$cartId)
];
return ['order' => $order, 'cart_list' => $cart_select];
}
/**
* 创建新订单
* @return Object
*/
static public function createOrder($cartId, $addressId, $user = null, $params = []): Object
{
$orderInfo = self::cartIdByOrderInfo($cartId, $addressId, $user, $params);
$_order = $orderInfo['order'];
$_order['deduction_price'] = 0;
$_order['merchant'] = 0;
$_order['customer'] = request()->userId;
$_order['money'] = 0;
$_order['user'] = request()->userId;
$_order['account'] = 0;
$_order['payinfo'] = '';
$_order['type'] = 0;
$_order['actual'] = $_order['total'];
Db::startTrans();
try {
$order = Cashierclass::create($_order);
$goods_list = $orderInfo['cart_list'];
foreach ($goods_list as $k => $v) {
$goods_list[$k]['pid'] = $order->id;
$goods_list[$k]['merchant'] = 0;
$goods_list[$k]['room'] = 0;
$goods_list[$k]['discount'] = 0;
$goods_list[$k]['warehouse'] = 0;
$goods_list[$k]['nums'] = $v['cart_num'];
}
(new Cashierinfo())->saveAll($goods_list);
Db::commit();
return $order;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 获取购货订单购物车商品信息
* @param $params
* @return array
*/
static public function cartIdByPurchaseOrderInfo($params = [])
{
$where1=['paid'=>1];
d(222);
$cartId=Cashierinfo::where('store_id',$params['store_id'])->where($where1)->select();
throw new \RuntimeException("Command has no defaultName");
$cartId=implode(',',$cartId);
var_dump($cartId);
$where = ['is_pay' => 0, 'is_fail' => 0];
$cart_select = Cart::whereIn('cart_id', $cartId)->where($where)->field('goods_id as goods,cart_num')->select()->toArray();
if (empty($cart_select)) {
self::setError('购物车为空');
return false;
}
/** 计算价格 */
foreach ($cart_select as $k => $v) {
$sell = Goods::where(['id' => $v['goods']])->value('sell');
$cart_select[$k]['total'] = bcmul($v['cart_num'], $sell, 2);
$cart_select[$k]['price'] = $sell;
}
$order = [
'time' => time(),
'number' => static::getNewOrderId('PF'),
'total' => array_sum(array_column($cart_select, 'total')),
'pay_type' => $params['pay_type'],
'cart_id'=>explode(',',$cartId)
];
return ['order' => $order, 'cart_list' => $cart_select];
}
/**
* 创建购货订单
* @return Object
*/
static public function createPurchaseOrder($cartId, $addressId, $user = null, $params = []): Object
{
$orderInfo = self::cartIdByOrderInfo($cartId, $addressId, $user, $params);
$_order = $orderInfo['order'];
$_order['deduction_price'] = 0;
$_order['merchant'] = 0;
$_order['customer'] = request()->userId;
$_order['money'] = 0;
$_order['user'] = request()->userId;
$_order['account'] = 0;
$_order['payinfo'] = '';
$_order['type'] = 0;
$_order['actual'] = $_order['total'];
Db::startTrans();
try {
$order = Cashierclass::create($_order);
$goods_list = $orderInfo['cart_list'];
foreach ($goods_list as $k => $v) {
$goods_list[$k]['pid'] = $order->id;
$goods_list[$k]['merchant'] = 0;
$goods_list[$k]['room'] = 0;
$goods_list[$k]['discount'] = 0;
$goods_list[$k]['warehouse'] = 0;
$goods_list[$k]['nums'] = $v['cart_num'];
}
(new Cashierinfo())->saveAll($goods_list);
Db::commit();
return $order;
} catch (\Exception $e) {
Db::rollback();
self::setError($e->getMessage());
return false;
}
}
/**
* @notes 获取订单号
* @param $type
* @return string
* @author likeadmin
* @date 2021/7/28 17:05
*/
static public function getNewOrderId($type)
{
list($msec, $sec) = explode(' ', microtime());
$msectime = number_format((floatval($msec) + floatval($sec)) * 1000, 0, '', '');
$orderId = $type . $msectime . mt_rand(10000, max(intval($msec * 10000) + 10000, 98369));
return $orderId;
}
}

View File

@ -0,0 +1,103 @@
<?php
namespace app\api\service;
use app\common\cache\UserTokenCache;
use app\common\model\user\UserSession;
use Webman\Config;
class UserTokenService
{
/**
* @notes 设置或更新用户token
* @param $userId
* @param $terminal
* @return array|false|mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 10:10
*/
public static function setToken($userId, $terminal)
{
$time = time();
$userSession = UserSession::where([['user_id', '=', $userId], ['terminal', '=', $terminal]])->find();
//获取token延长过期的时间
$expireTime = $time + Config::get('project.user_token.expire_duration');
$userTokenCache = new UserTokenCache();
//token处理
if ($userSession) {
//清空缓存
$userTokenCache->deleteUserInfo($userSession->token);
//重新获取token
$userSession->token = create_token($userId);
$userSession->expire_time = $expireTime;
$userSession->update_time = $time;
$userSession->save();
} else {
//找不到在该终端的token记录创建token记录
$userSession = UserSession::create([
'user_id' => $userId,
'terminal' => $terminal,
'token' => create_token($userId),
'expire_time' => $expireTime
]);
}
return $userTokenCache->setUserInfo($userSession->token);
}
/**
* @notes 延长token过期时间
* @param $token
* @return array|false|mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 10:10
*/
public static function overtimeToken($token)
{
$time = time();
$adminSession = UserSession::where('token', '=', $token)->find();
//延长token过期时间
$adminSession->expire_time = $time + Config::get('project.user_token.expire_duration');
$adminSession->update_time = $time;
$adminSession->save();
return (new UserTokenCache())->setUserInfo($adminSession->token);
}
/**
* @notes 设置token为过期
* @param $token
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 10:10
*/
public static function expireToken($token)
{
$userSession = UserSession::where('token', '=', $token)
->find();
if (empty($userSession)) {
return false;
}
$time = time();
$userSession->expire_time = $time;
$userSession->update_time = $time;
$userSession->save();
return (new UserTokenCache())->deleteUserInfo($token);
}
}

View File

@ -0,0 +1,254 @@
<?php
namespace app\api\service;
use app\common\enum\YesNoEnum;
use app\common\model\user\{User, UserAuth};
use app\common\enum\user\UserTerminalEnum;
use app\common\service\{ConfigService, storage\Driver as StorageDriver};
use think\Exception;
/**
* 用户功能类(主要微信登录后创建和更新用户)
* Class WechatUserService
* @package app\api\service
*/
class WechatUserService
{
protected int $terminal = UserTerminalEnum::WECHAT_MMP;
protected array $response = [];
protected ?string $code = null;
protected ?string $openid = null;
protected ?string $unionid = null;
protected ?string $nickname = null;
protected ?string $headimgurl = null;
protected User $user;
public function __construct(array $response, int $terminal)
{
$this->terminal = $terminal;
$this->setParams($response);
}
/**
* @notes 设置微信返回的用户信息
* @param $response
* @author cjhao
* @date 2021/8/2 11:49
*/
private function setParams($response): void
{
$this->response = $response;
$this->openid = $response['openid'];
$this->unionid = $response['unionid'] ?? '';
$this->nickname = $response['nickname'] ?? '';
$this->headimgurl = $response['headimgurl'] ?? '';
}
/**
* @notes 根据opendid或unionid获取系统用户信息
* @return $this
* @author 段誉
* @date 2022/9/23 16:09
*/
public function getResopnseByUserInfo(): self
{
$openid = $this->openid;
$unionid = $this->unionid;
$user = User::alias('u')
->field('u.id,u.sn,u.mobile,u.nickname,u.avatar,u.mobile,u.is_disable,u.is_new_user')
->join('user_auth au', 'au.user_id = u.id')
->where(function ($query) use ($openid, $unionid) {
$query->whereOr(['au.openid' => $openid]);
if (isset($unionid) && $unionid) {
$query->whereOr(['au.unionid' => $unionid]);
}
})
->findOrEmpty();
$this->user = $user;
return $this;
}
/**
* @notes 获取用户信息
* @param bool $isCheck 是否验证账号是否可用
* @return array
* @throws Exception
* @author cjhao
* @date 2021/8/3 11:42
*/
public function getUserInfo($isCheck = true): array
{
if (!$this->user->isEmpty() && $isCheck) {
$this->checkAccount();
}
if (!$this->user->isEmpty()) {
$this->getToken();
}
return $this->user->toArray();
}
/**
* @notes 校验账号
* @throws Exception
* @author 段誉
* @date 2022/9/16 10:14
*/
private function checkAccount()
{
if ($this->user->is_disable) {
throw new Exception('您的账号异常,请联系客服。');
}
}
/**
* @notes 创建用户
* @throws Exception
* @author 段誉
* @date 2022/9/16 10:06
*/
private function createUser(): void
{
//设置头像
if (empty($this->headimgurl)) {
// 默认头像
$defaultAvatar = config('project.default_image.user_avatar');
$avatar = ConfigService::get('default_image', 'user_avatar', $defaultAvatar);
} else {
// 微信获取到的头像信息
$avatar = $this->getAvatarByWechat();
}
$userSn = User::createUserSn();
$this->user->sn = $userSn;
$this->user->account = 'u' . $userSn;
$this->user->nickname = "用户" . $userSn;
$this->user->avatar = $avatar;
$this->user->channel = $this->terminal;
$this->user->is_new_user = YesNoEnum::YES;
if ($this->terminal != UserTerminalEnum::WECHAT_MMP && !empty($this->nickname)) {
$this->user->nickname = $this->nickname;
}
$this->user->save();
UserAuth::create([
'user_id' => $this->user->id,
'openid' => $this->openid,
'unionid' => $this->unionid,
'terminal' => $this->terminal,
]);
}
/**
* @notes 更新用户信息
* @throws Exception
* @author 段誉
* @date 2022/9/16 10:06
* @remark 该端没授权信息,重新写入一条该端的授权信息
*/
private function updateUser(): void
{
// 无头像需要更新头像
if (empty($this->user->avatar)) {
$this->user->avatar = $this->getAvatarByWechat();
$this->user->save();
}
$userAuth = UserAuth::where(['user_id' => $this->user->id, 'openid' => $this->openid])
->findOrEmpty();
// 无该端授权信息,新增一条
if ($userAuth->isEmpty()) {
$userAuth->user_id = $this->user->id;
$userAuth->openid = $this->openid;
$userAuth->unionid = $this->unionid;
$userAuth->terminal = $this->terminal;
$userAuth->save();
} else {
if (empty($userAuth['unionid']) && !empty($this->unionid)) {
$userAuth->unionid = $this->unionid;
$userAuth->save();
}
}
}
/**
* @notes 获取token
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author cjhao
* @date 2021/8/2 16:45
*/
private function getToken(): void
{
$user = UserTokenService::setToken($this->user->id, $this->terminal);
$this->user->token = $user['token'];
}
/**
* @notes 用户授权登录,
* 如果用户不存在,创建用户;用户存在,更新用户信息,并检查该端信息是否需要写入
* @return WechatUserService
* @throws Exception
* @author cjhao
* @date 2021/8/2 16:35
*/
public function authUserLogin(): self
{
if ($this->user->isEmpty()) {
$this->createUser();
} else {
$this->updateUser();
}
return $this;
}
/**
* @notes 处理从微信获取到的头像信息
* @return string
* @throws Exception
* @author 段誉
* @date 2022/9/16 9:50
*/
public function getAvatarByWechat(): string
{
// 存储引擎
$config = [
'default' => ConfigService::get('storage', 'default', 'local'),
'engine' => ConfigService::get('storage')
];
if ($config['default'] == 'local') {
// 本地存储
$file_name = md5($this->openid . time()) . '.jpeg';
$avatar = download_file($this->headimgurl, 'uploads/user/avatar/', $file_name);
} else {
// 第三方存储
$avatar = 'uploads/user/avatar/' . md5($this->openid . time()) . '.jpeg';
$StorageDriver = new StorageDriver($config);
if (!$StorageDriver->fetch($this->headimgurl, $avatar)) {
throw new Exception('头像保存失败:' . $StorageDriver->getError());
}
}
return $avatar;
}
}

View File

@ -0,0 +1,148 @@
<?php
namespace app\api\validate;
use app\common\cache\UserAccountSafeCache;
use app\common\enum\LoginEnum;
use app\common\enum\notice\NoticeEnum;
use app\common\enum\user\UserTerminalEnum;
use app\common\enum\YesNoEnum;
use app\common\service\ConfigService;
use app\common\service\sms\SmsDriver;
use app\common\validate\BaseValidate;
use app\common\model\user\User;
use Webman\Config;
/**
* 账号密码登录校验
* Class LoginValidate
* @package app\api\validate
*/
class LoginAccountValidate extends BaseValidate
{
protected $rule = [
'terminal' => 'require|in:' . UserTerminalEnum::WECHAT_MMP . ',' . UserTerminalEnum::WECHAT_OA . ','
. UserTerminalEnum::H5 . ',' . UserTerminalEnum::PC . ',' . UserTerminalEnum::IOS .
',' . UserTerminalEnum::ANDROID,
'scene' => 'require|in:' . LoginEnum::ACCOUNT_PASSWORD . ',' . LoginEnum::MOBILE_CAPTCHA . '|checkConfig',
'account' => 'require',
];
protected $message = [
'terminal.require' => '终端参数缺失',
'terminal.in' => '终端参数状态值不正确',
'scene.require' => '场景不能为空',
'scene.in' => '场景值错误',
'account.require' => '请输入账号',
'password.require' => '请输入密码',
];
/**
* @notes 登录场景相关校验
* @param $scene
* @param $rule
* @param $data
* @return bool|string
* @author 段誉
* @date 2022/9/15 14:37
*/
public function checkConfig($scene, $rule, $data)
{
$config = ConfigService::get('login', 'login_way');
if (!in_array($scene, $config)) {
return '不支持的登录方式';
}
// 账号密码登录
if (LoginEnum::ACCOUNT_PASSWORD == $scene) {
if (!isset($data['password'])) {
return '请输入密码';
}
return $this->checkPassword($data['password'], [], $data);
}
// 手机验证码登录
if (LoginEnum::MOBILE_CAPTCHA == $scene) {
if (!isset($data['code'])) {
return '请输入手机验证码';
}
return $this->checkCode($data['code'], [], $data);
}
return true;
}
/**
* @notes 登录密码校验
* @param $password
* @param $other
* @param $data
* @return bool|string
* @author 段誉
* @date 2022/9/15 14:39
*/
public function checkPassword($password, $other, $data)
{
//账号安全机制,连续输错后锁定,防止账号密码暴力破解
$userAccountSafeCache = new UserAccountSafeCache();
if (!$userAccountSafeCache->isSafe()) {
return '密码连续' . $userAccountSafeCache->count . '次输入错误,请' . $userAccountSafeCache->minute . '分钟后重试';
}
$where = [];
if ($data['scene'] == LoginEnum::ACCOUNT_PASSWORD) {
// 手机号密码登录
$where = ['account|mobile' => $data['account']];
}
$userInfo = User::where($where)
->field(['password,is_disable'])
->findOrEmpty();
if ($userInfo->isEmpty()) {
return '用户不存在';
}
if ($userInfo['is_disable'] === YesNoEnum::YES) {
return '用户已禁用';
}
if (empty($userInfo['password'])) {
$userAccountSafeCache->record();
return '用户不存在';
}
$passwordSalt = Config::get('project.unique_identification');
if ($userInfo['password'] !== create_password($password, $passwordSalt)) {
$userAccountSafeCache->record();
return '密码错误';
}
$userAccountSafeCache->relieve();
return true;
}
/**
* @notes 校验验证码
* @param $code
* @param $rule
* @param $data
* @return bool|string
* @author Tab
* @date 2021/8/25 15:43
*/
public function checkCode($code, $rule, $data)
{
// $smsDriver = new SmsDriver();
// $result = $smsDriver->verify($data['account'], $code, NoticeEnum::LOGIN_CAPTCHA);
// if ($result) {
// return true;
// }
// return '验证码错误';
}
}

View File

@ -0,0 +1,29 @@
<?php
namespace app\ceshi\controller;
use support\Request;
class IndexController
{
public function index(Request $request)
{
d(231232);
static $readme;
if (!$readme) {
$readme = file_get_contents(base_path('README.md'));
}
return $readme;
}
public function view(Request $request)
{
return view('index/view', ['name' => 'webman']);
}
public function json(Request $request)
{
return json(['code' => 0, 'msg' => 'ok']);
}
}

View File

@ -0,0 +1,69 @@
<?php
namespace app\common\cache;
use support\Cache;
/**
* //后台账号安全机制,连续输错后锁定,防止账号密码暴力破解
* Class AdminAccountSafeCache
* @package app\common\cache
*/
class UserAccountSafeCache extends BaseCache
{
private $key;//缓存次数名称
public $minute = 15;//缓存设置为15分钟即密码错误次数达到锁定15分钟
public $count = 15; //设置连续输错次数即15分钟内连续输错误15次后锁定
public function __construct()
{
parent::__construct();
$ip = \request()->getLocalIp();
$this->key = $this->tagName . $ip;
}
/**
* @notes 记录登录错误次数
* @author 令狐冲
* @date 2021/6/30 01:51
*/
public function record()
{
if (Cache::get($this->key)) {
//缓存存在,记录错误次数
Cache::set($this->key, Cache::get($this->key) + 1);
} else {
//缓存不存在,第一次设置缓存
Cache::set($this->key, 1, $this->minute * 60);
}
}
/**
* @notes 判断是否安全
* @return bool
* @author 令狐冲
* @date 2021/6/30 01:53
*/
public function isSafe()
{
$count = Cache::get($this->key);
if ($count >= $this->count) {
return false;
}
return true;
}
/**
* @notes 删除该ip记录错误次数
* @author 令狐冲
* @date 2021/6/30 01:55
*/
public function relieve()
{
Cache::delete($this->key);
}
}

107
app/common/cache/UserTokenCache.php vendored Normal file
View File

@ -0,0 +1,107 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\common\cache;
use app\common\model\user\User;
use app\common\model\user\UserSession;
use support\Cache;
class UserTokenCache extends BaseCache
{
private $prefix = 'token_user_';
/**
* @notes 通过token获取缓存用户信息
* @param $token
* @return array|false|mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 10:11
*/
public function getUserInfo($token)
{
//直接从缓存获取
$userInfo = Cache::get($this->prefix . $token);
if ($userInfo) {
return $userInfo;
}
//从数据获取信息被设置缓存(可能后台清除缓存)
$userInfo = $this->setUserInfo($token);
if ($userInfo) {
return $userInfo;
}
return false;
}
/**
* @notes 通过有效token设置用户信息缓存
* @param $token
* @return array|false|mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @author 段誉
* @date 2022/9/16 10:11
*/
public function setUserInfo($token)
{
$userSession = UserSession::where([['token', '=', $token], ['expire_time', '>', time()]])->find();
if (empty($userSession)) {
return [];
}
$user = User::where('id', '=', $userSession->user_id)
->find();
$userInfo = [
'user_id' => $user->id,
'nickname' => $user->nickname,
'token' => $token,
'sn' => $user->sn,
'mobile' => $user->mobile,
'avatar' => $user->avatar,
'terminal' => $userSession->terminal,
'expire_time' => $userSession->expire_time,
];
// $ttl = new \DateTime(Date('Y-m-d H:i:s', $userSession->expire_time));
$this->set($this->prefix . $token, $userInfo, $userSession->expire_time);
return $this->getUserInfo($token);
}
/**
* @notes 删除缓存
* @param $token
* @return bool
* @author 段誉
* @date 2022/9/16 10:13
*/
public function deleteUserInfo($token)
{
return Cache::delete($this->prefix . $token);
}
}

View File

@ -8,7 +8,7 @@ use app\BaseController;
use app\common\lists\BaseDataLists;
use app\common\service\JsonService;
class BaseLikeAdminController extends BaseController
class BaseLikeController extends BaseController
{
public $notNeedLogin = [];

View File

@ -0,0 +1,40 @@
<?php
namespace app\common\enum;
class OrderEnum
{
/**
* 流水类型
* @USER_ORDER_PAY 用户订单支付
* @MERCHANT_ORDER_OBTAINS 商户订单获得
* @ORDER_HANDLING_FEES 订单手续费
*/
const USER_ORDER_PAY = 1;
const MERCHANT_ORDER_OBTAINS = 2;
const ORDER_HANDLING_FEES = 3;
/**
* 收入支出类型
* @EXPENDITURE 支出
* @INCOME 收入
*/
const EXPENDITURE =0;
const INCOME =1;
/**
* 账户类型
* @USER 用户
* @MERCHANT 商户
* @PLATFORM 平台
*/
const USER =0;
const MERCHANT =1;
const PLATFORM =2;
}

128
app/common/enum/PayEnum.php Normal file
View File

@ -0,0 +1,128 @@
<?php
namespace app\common\enum;
/**
* 支付
* Class PayrEnum
* @package app\common\enum
*/
class PayEnum
{
/**
* 支付类型
* @BALANCE 余额
* @WECHAT_PAY 微信支付
* @ALIPAY 支付宝
* @WECHAT_PAY_APP 微信APP
* @ALIPAY_APP 支付宝APP
* @WECHAT_PAY_H5 微信H5
* @ALIPAY_H5 支付宝H5
* @WECHAT_PAY_MINI 微信小程序
* @ALIPAY_MINI 支付宝小程序
* @WECHAT_PAY_JSAPI 微信公众号
* @ALIPAY_PAY_JSAPI 支付宝公众号
* @WECHAT_PAY_BARCODE 微信条码
* @WECHAT_PAY_QRCODE 微信二维码
* @ALIPAY_BARCODE 支付宝条码
* @ALIPAY_QRCODE 支付宝二维码
* @GOODS_FIRST_PAYMENT_LATER 先货后款
* @CORPORATE_TRANSFER 对公转账
*/
const BALANCE_PAY = 3;
const WECHAT_PAY = 1;
const WECHAT_PAY_APP = 4;
const WECHAT_PAY_H5 = 5;
const WECHAT_PAY_JSAPI = 6;
const WECHAT_PAY_MINI = 7;
const WECHAT_PAY_BARCODE = 9;
const WECHAT_PAY_QRCODE = 10;
const ALIPAY = 2;
const ALIPAY_APP = 8;
const ALIPAY_H5 = 11;
const ALIPAY_MINI = 12;
const ALIPAY_BARCODE = 13;
const ALIPAY_QRCODE = 14;
const GOODS_FIRST_PAYMENT_LATER = 15;
const CORPORATE_TRANSFER = 16;
//支付状态
const UNPAID = 0; //未支付
const ISPAID = 1; //已支付
//支付场景
const SCENE_H5 = 1; //H5
const SCENE_OA = 2; //微信公众号
const SCENE_MNP = 3; //微信小程序
const SCENE_APP = 4; //APP
const SCENE_PC = 5; //PC商城
/**
* @notes 获取支付类型
* @param bool $value
* @return string|string[]
* @author 段誉
* @date 2023/2/23 15:36
*/
public static function getPayDesc($value = true)
{
$data = [
self::BALANCE_PAY => '余额支付',
self::WECHAT_PAY => '微信支付',
self::ALIPAY => '支付宝支付',
];
if ($value === true) {
return $data;
}
return $data[$value] ?? '';
}
/**
* @notes 支付状态
* @param bool $value
* @return string|string[]
* @author 段誉
* @date 2023/2/23 15:36
*/
public static function getPayStatusDesc($value = true)
{
$data = [
self::UNPAID => '未支付',
self::ISPAID => '已支付',
];
if ($value === true) {
return $data;
}
return $data[$value] ?? '';
}
/**
* @notes 支付场景
* @param bool $value
* @return string|string[]
* @author 段誉
* @date 2023/2/23 15:36
*/
public static function getPaySceneDesc($value = true)
{
$data = [
self::SCENE_H5 => 'H5',
self::SCENE_OA => '微信公众号',
self::SCENE_MNP => '微信小程序',
self::SCENE_APP => 'APP',
self::SCENE_PC => 'PC',
];
if ($value === true) {
return $data;
}
return $data[$value] ?? '';
}
}

View File

@ -0,0 +1,105 @@
<?php
namespace app\common\logic\order;
use app\common\enum\OrderEnum;
use app\common\logic\BaseLogic;
use app\common\model\order\Cart;
use app\common\model\order\FinancialRecord;
use app\common\model\user\User;
use Exception;
use support\Log;
use taoser\exception\ValidateException;
use think\facade\Db;
/**
* 零售订单逻辑处理
*/
class RetailOrderLogic extends BaseLogic
{
/**
* 余额订单支付
* @param User $user
* @param $order
* @return bool
* @throws Exception
* @throws ValidateException
*/
static public function payBalance(User $user, $order)
{
if ($user['user_money'] < $order['actual']){
self::setError('余额不足,请更换支付方式');
return false;
}
Db::startTrans();
try {
$user->user_money = bcsub($user->user_money, $order['actual'], 2);
$user->save();
//用户支出流水
$record[] = [
'financial_record_sn' => time(),
'order_id' => $order['id'],
'number_sn' => $order['number'],
'user_id' => $order['customer'],
'financial_type' => OrderEnum::USER_ORDER_PAY,
'financial_pm' => OrderEnum::EXPENDITURE,
'number' => $order['actual'],
'status' => 1,
'type' => OrderEnum::USER,
'mer_id' => $order['merchant'],
];
//商户获得流水
$record[] = [
'financial_record_sn' => time(),
'order_id' => $order['id'],
'number_sn' => $order['number'],
'user_id' => $order['customer'],
'financial_type' => OrderEnum::MERCHANT_ORDER_OBTAINS,
'financial_pm' => OrderEnum::INCOME,
'number' => $order['actual'],
'status' => 0,
'type' => OrderEnum::MERCHANT,
'mer_id' => $order['merchant'],
];
(new FinancialRecord())->saveAll($record);
$CallbackData = ['money' => $order['actual']];
self::paySuccess($order, $CallbackData);
Db::commit();
return true;
} catch (Exception $e) {
Db::rollback();
Log::error('余额支付失败' . $e->getMessage() . '。line:' . $e->getLine() . '。file:' . $e->getFile());
self::setError('余额支付失败' . $e->getMessage());
return false;
}
}
/**
* @notes 订单支付成功
* @param $order 订单
* @param $CallbackData 回调数据
* @date 2021/7/8 00:40
*/
static function paySuccess($order, $CallbackData = [])
{
Db::startTrans();
try {
$order->money = $CallbackData['money'];
$order->paid = 1;
$order->save();
if($order['cart_id']){
$cart_arr=implode(',',$order['cart_id']);
Cart::whereIn('id',$cart_arr)->update(['is_pay'=>1]);
}
// 提交事务
Db::commit();
} catch (\Exception $e) {
// 回滚事务
Db::rollback();
}
// TODO: Implement paySuccess() method.
}
}

View File

@ -0,0 +1,22 @@
<?php
namespace app\common\model\order;
use app\common\model\BaseModel;
use think\model\concern\SoftDelete;
/**
* 购物车表模型
* Class Cart
* @package app\common\model\order
*/
class Cart extends BaseModel
{
use SoftDelete;
protected $name = 'cart';
protected $deleteTime = 'delete_time';
}

View File

@ -0,0 +1,22 @@
<?php
namespace app\common\model\order;
use app\common\model\BaseModel;
/**
* 财务流水模型
* Class FinancialRecord
* @package app\common\model\order
*/
class FinancialRecord extends BaseModel
{
protected $name = 'financial_record';
}

View File

@ -0,0 +1,45 @@
<?php
namespace app\common\model\pay;
use app\common\enum\PayEnum;
use app\common\model\BaseModel;
use app\common\service\FileService;
class PayConfig extends BaseModel
{
protected $name = 'dev_pay_config';
// 设置json类型字段
protected $json = ['config'];
// 设置JSON数据返回数组
protected $jsonAssoc = true;
/**
* @notes 支付图标获取器 - 路径添加域名
* @param $value
* @return string
* @author ljj
* @date 2021/7/28 2:12 下午
*/
public function getIconAttr($value)
{
return empty($value) ? '' : FileService::getFileUrl($value);
}
/**
* @notes 支付方式名称获取器
* @param $value
* @param $data
* @return string|string[]
* @author ljj
* @date 2021/7/31 2:24 下午
*/
public function getPayWayNameAttr($value,$data)
{
return PayEnum::getPayDesc($data['pay_way']);
}
}

View File

@ -0,0 +1,54 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\common\model\pay;
use app\common\model\BaseModel;
use app\common\service\FileService;
class PayWay extends BaseModel
{
protected $name = 'dev_pay_way';
public function getIconAttr($value,$data)
{
return FileService::getFileUrl($value);
}
/**
* @notes 支付方式名称获取器
* @param $value
* @param $data
* @return mixed
* @author ljj
* @date 2021/7/28 4:02 下午
*/
public static function getPayWayNameAttr($value,$data)
{
return PayConfig::where('id',$data['pay_config_id'])->value('name');
}
/**
* @notes 关联支配配置模型
* @return \think\model\relation\HasOne
* @author ljj
* @date 2021/10/11 3:04 下午
*/
public function payConfig()
{
return $this->hasOne(PayConfig::class,'id','pay_config_id');
}
}

View File

@ -0,0 +1,22 @@
<?php
namespace app\common\model\retail;
use app\common\model\BaseModel;
use think\model\concern\SoftDelete;
/**
* 零售订单模型
* Class Cashierclass
* @package app\common\model\retail
*/
class Cashierclass extends BaseModel
{
use SoftDelete;
protected $name = 'cashierclass';
protected $deleteTime = 'delete_time';
}

View File

@ -0,0 +1,22 @@
<?php
namespace app\common\model\retail;
use app\common\model\BaseModel;
use think\model\concern\SoftDelete;
/**
* 零售订单详情模型
* Class Cashierinfo
* @package app\common\model\retail
*/
class Cashierinfo extends BaseModel
{
use SoftDelete;
protected $name = 'cashierinfo';
protected $deleteTime = 'delete_time';
}

View File

@ -0,0 +1,15 @@
<?php
namespace app\common\model\user;
use app\common\model\BaseModel;
/**
* 用户登录token信息
* Class UserSession
* @package app\common\model\user
*/
class UserSession extends BaseModel
{
}

View File

@ -96,7 +96,7 @@ class ControllerGenerator extends BaseGenerator implements GenerateInterface
if ($this->moduleName == 'admin') {
$tpl = "use app\\" . $this->moduleName . "\\controller\\BaseAdminController;" . PHP_EOL;
} else {
$tpl = "use app\\common\\controller\\BaseLikeAdminController;" . PHP_EOL;
$tpl = "use app\\common\\controller\\BaseLikeController;" . PHP_EOL;
}
if (!empty($this->classDir)) {
@ -151,7 +151,7 @@ class ControllerGenerator extends BaseGenerator implements GenerateInterface
{
$tpl = 'BaseAdminController';
if ($this->moduleName != 'admin') {
$tpl = 'BaseLikeAdminController';
$tpl = 'BaseLikeController';
}
return $tpl;
}

View File

@ -2,20 +2,20 @@
<div class="edit-popup">
<popup
ref="popupRef"
:title="popupTitle"
title="详情"
:async="true"
width="550px"
:cancelButtonText="false"
:confirmButtonText="false"
>
<el-form ref="formRef" :model="formData" label-width="90px" :rules="formRules">
<el-form ref="formRef" :model="formData" label-width="90px">
{FORM_VIEW}
</el-form>
</popup>
</div>
</template>
<script lang="ts" setup name="{SETUP_NAME}Edit">
<script lang="ts" setup name="{SETUP_NAME}DETAILS">
import type { FormInstance } from 'element-plus'
import Popup from '@/components/popup/index.vue'
import {{IMPORT_LISTS} api{UPPER_CAMEL_NAME}Add, api{UPPER_CAMEL_NAME}Edit, api{UPPER_CAMEL_NAME}Detail } from '@/api/{API_DIR}'
@ -61,6 +61,10 @@ const getDetail = async (row: Record<string, any>) => {
setFormData(data)
}
//打开弹窗
const open = () => {
popupRef.value?.open()
}
// 关闭回调
const handleClose = () => {

View File

@ -0,0 +1,93 @@
<?php
namespace app\common\service\wechat;
use app\common\enum\PayEnum;
use app\common\enum\user\UserTerminalEnum;
use app\common\model\pay\PayConfig;
use app\common\service\ConfigService;
/**
* 微信配置类
* Class WeChatConfigService
* @package app\common\service
*/
class WeChatConfigService
{
/**
* @notes 获取小程序配置
* @return array
* @author 段誉
* @date 2022/9/6 19:49
*/
public static function getMnpConfig()
{
return [
'app_id' => ConfigService::get('mnp_setting', 'app_id'),
'secret' => ConfigService::get('mnp_setting', 'app_secret'),
'response_type' => 'array',
'log' => [
'level' => 'debug',
'file' => runtime_path(). '/wechat/' . date('Ym') . '/' . date('d') . '.log'
],
];
}
/**
* @notes 获取微信公众号配置
* @return array
* @author 段誉
* @date 2022/9/6 19:49
*/
public static function getOaConfig()
{
return [
'app_id' => ConfigService::get('oa_setting', 'app_id'),
'secret' => ConfigService::get('oa_setting', 'app_secret'),
'token' => ConfigService::get('oa_setting', 'token'),
'response_type' => 'array',
'log' => [
'level' => 'debug',
'file' => runtime_path(). '/wechat/' . date('Ym') . '/' . date('d') . '.log'
],
];
}
/**
* @notes 获取微信开放平台配置
* @return array
* @author 段誉
* @date 2022/10/20 15:51
*/
public static function getOpConfig()
{
return [
'app_id' => ConfigService::get('open_platform', 'app_id'),
'secret' => ConfigService::get('open_platform', 'app_secret'),
'response_type' => 'array',
'log' => [
'level' => 'debug',
'file' => runtime_path(). '/wechat/' . date('Ym') . '/' . date('d') . '.log'
],
];
}
/**
* @notes 临时写入证书
* @param $path
* @param $cert
* @author 段誉
* @date 2023/2/27 15:48
*/
public static function setCert($path, $cert)
{
$fopenPath = fopen($path, 'w');
fwrite($fopenPath, $cert);
fclose($fopenPath);
}
}

View File

@ -0,0 +1,101 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\common\service\wechat;
use EasyWeChat\Kernel\Exceptions\Exception;
use EasyWeChat\MiniApp\Application;
/**
* 微信功能类
* Class WeChatMnpService
* @package app\common\service
*/
class WeChatMnpService
{
protected $app;
protected $config;
public function __construct()
{
$this->config = $this->getConfig();
$this->app = new Application($this->config);
}
/**
* @notes 配置
* @return array
* @throws \Exception
* @author 段誉
* @date 2023/2/27 12:03
*/
protected function getConfig()
{
$config = WeChatConfigService::getMnpConfig();
if (empty($config['app_id']) || empty($config['secret'])) {
throw new \Exception('请先设置小程序配置');
}
return $config;
}
/**
* @notes 小程序-根据code获取微信信息
* @param string $code
* @return array
* @throws Exception
* @throws \EasyWeChat\Kernel\Exceptions\HttpException
* @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
* @throws \Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\DecodingExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\RedirectionExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\ServerExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface
* @author 段誉
* @date 2023/2/27 11:03
*/
public function getMnpResByCode(string $code)
{
$utils = $this->app->getUtils();
$response = $utils->codeToSession($code);
if (!isset($response['openid']) || empty($response['openid'])) {
throw new Exception('获取openID失败');
}
return $response;
}
/**
* @notes 获取手机号
* @param string $code
* @return \EasyWeChat\Kernel\HttpClient\Response|\Symfony\Contracts\HttpClient\ResponseInterface
* @throws \Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface
* @author 段誉
* @date 2023/2/27 11:46
*/
public function getUserPhoneNumber(string $code)
{
return $this->app->getClient()->postJson('wxa/business/getuserphonenumber', [
'code' => $code,
]);
}
}

View File

@ -0,0 +1,157 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\common\service\wechat;
use EasyWeChat\Kernel\Exceptions\Exception;
use EasyWeChat\OfficialAccount\Application;
/**
* 公众号相关
* Class WeChatOaService
* @package app\common\service\wechat
*/
class WeChatOaService
{
protected $app;
protected $config;
public function __construct()
{
$this->config = $this->getConfig();
$this->app = new Application($this->config);
}
/**
* @notes easywechat服务端
* @return \EasyWeChat\Kernel\Contracts\Server|\EasyWeChat\OfficialAccount\Server
* @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
* @throws \ReflectionException
* @throws \Throwable
* @author 段誉
* @date 2023/2/27 14:22
*/
public function getServer()
{
return $this->app->getServer();
}
/**
* @notes 配置
* @return array
* @throws \Exception
* @author 段誉
* @date 2023/2/27 12:03
*/
protected function getConfig()
{
$config = WeChatConfigService::getOaConfig();
if (empty($config['app_id']) || empty($config['secret'])) {
throw new \Exception('请先设置公众号配置');
}
return $config;
}
/**
* @notes 公众号-根据code获取微信信息
* @param string $code
* @return mixed
* @throws Exception
* @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
* @author 段誉
* @date 2023/2/27 11:04
*/
public function getOaResByCode(string $code)
{
$response = $this->app->getOAuth()
->scopes(['snsapi_userinfo'])
->userFromCode($code)
->getRaw();
if (!isset($response['openid']) || empty($response['openid'])) {
throw new Exception('获取openID失败');
}
return $response;
}
/**
* @notes 公众号跳转url
* @param string $url
* @return mixed
* @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
* @author 段誉
* @date 2023/2/27 10:35
*/
public function getCodeUrl(string $url)
{
return $this->app->getOAuth()
->scopes(['snsapi_userinfo'])
->redirect($url);
}
/**
* @notes 创建公众号菜单
* @param array $buttons
* @param array $matchRule
* @return \EasyWeChat\Kernel\HttpClient\Response|\Symfony\Contracts\HttpClient\ResponseInterface
* @throws \Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface
* @author 段誉
* @date 2023/2/27 12:07
*/
public function createMenu(array $buttons, array $matchRule = [])
{
if (!empty($matchRule)) {
return $this->app->getClient()->postJson('cgi-bin/menu/addconditional', [
'button' => $buttons,
'matchrule' => $matchRule,
]);
}
return $this->app->getClient()->postJson('cgi-bin/menu/create', ['button' => $buttons]);
}
/**
* @notes 获取jssdkConfig
* @param $url
* @param $jsApiList
* @param array $openTagList
* @param false $debug
* @return mixed[]
* @throws \EasyWeChat\Kernel\Exceptions\HttpException
* @throws \Psr\SimpleCache\InvalidArgumentException
* @throws \Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\DecodingExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\RedirectionExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\ServerExceptionInterface
* @throws \Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface
* @author 段誉
* @date 2023/3/1 11:46
*/
public function getJsConfig($url, $jsApiList, $openTagList = [], $debug = false)
{
return $this->app->getUtils()->buildJsSdkConfig($url, $jsApiList, $openTagList, $debug);
}
}

View File

@ -0,0 +1,82 @@
<?php
// +----------------------------------------------------------------------
// | likeadmin快速开发前后端分离管理后台PHP版
// +----------------------------------------------------------------------
// | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
// | 开源版本可自由商用可去除界面版权logo
// | gitee下载https://gitee.com/likeshop_gitee/likeadmin
// | github下载https://github.com/likeshop-github/likeadmin
// | 访问官网https://www.likeadmin.cn
// | likeadmin团队 版权所有 拥有最终解释权
// +----------------------------------------------------------------------
// | author: likeadminTeam
// +----------------------------------------------------------------------
namespace app\common\service\wechat;
use app\common\logic\BaseLogic;
use WpOrg\Requests\Requests;
/**
* 自定义微信请求
* Class WeChatRequestService
* @package app\common\service\wechat
*/
class WeChatRequestService extends BaseLogic
{
/**
* @notes 获取网站扫码登录地址
* @param $appId
* @param $redirectUri
* @param $state
* @return string
* @author 段誉
* @date 2022/10/20 18:20
*/
public static function getScanCodeUrl($appId, $redirectUri, $state)
{
$url = 'https://open.weixin.qq.com/connect/qrconnect?';
$url .= 'appid=' . $appId . '&redirect_uri=' . $redirectUri . '&response_type=code&scope=snsapi_login';
$url .= '&state=' . $state . '#wechat_redirect';
return $url;
}
/**
* @notes 通过code获取用户信息(access_token,openid,unionid等)
* @param $code
* @return mixed
* @author 段誉
* @date 2022/10/21 10:16
*/
public static function getUserAuthByCode($code)
{
$config = WeChatConfigService::getOpConfig();
$url = 'https://api.weixin.qq.com/sns/oauth2/access_token';
$url .= '?appid=' . $config['app_id'] . '&secret=' . $config['secret'] . '&code=' . $code;
$url .= '&grant_type=authorization_code';
$requests = Requests::get($url);
return json_decode($requests->body, true);
}
/**
* @notes 通过授权信息获取用户信息
* @param $accessToken
* @param $openId
* @return mixed
* @author 段誉
* @date 2022/10/21 10:21
*/
public static function getUserInfoByAuth($accessToken, $openId)
{
$url = 'https://api.weixin.qq.com/sns/userinfo';
$url .= '?access_token=' . $accessToken . '&openid=' . $openId;
$response = Requests::get($url);
return json_decode($response->body, true);
}
}

View File

@ -206,4 +206,43 @@ function get_file_domain($content)
$preg = '/(<img .*?src=")[^https|^http](.*?)(".*?>)/is';
$fileUrl = FileService::getFileUrl();
return preg_replace($preg, "\${1}$fileUrl\${2}\${3}", $content);
}
/**
* @notes 下载文件
* @param $url
* @param $saveDir
* @param $fileName
* @return string
* @author 段誉
* @date 2022/9/16 9:53
*/
function download_file($url, $saveDir, $fileName)
{
if (!file_exists($saveDir)) {
mkdir($saveDir, 0775, true);
}
$fileSrc = $saveDir . $fileName;
file_exists($fileSrc) && unlink($fileSrc);
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 30);
$file = curl_exec($ch);
curl_close($ch);
$resource = fopen($fileSrc, 'a');
fwrite($resource, $file);
fclose($resource);
if (filesize($fileSrc) == 0) {
unlink($fileSrc);
return '';
}
return $fileSrc;
}
function halt(String|Array|Float|Int $trage)
{
$str = is_array($trage) ? var_export($trage,1) : trim($trage);
$hears = ['Content-Type' => 'text/html;charset=utf-8'];
return new \support\Response('200',$hears,'<pre>'.$str);
}

View File

@ -45,7 +45,10 @@
"illuminate/redis": "^10.22",
"max/var-dumper": "^0.1.1",
"symfony/cache": "^6.4",
"yansongda/pay": "^3.6"
"yansongda/pay": "^3.6",
"tinywan/jwt": "^1.9",
"overtrue/wechat": "^6.7",
"next/var-dumper": "^0.1.0"
},
"suggest": {
"ext-event": "For better performance. "

1350
composer.lock generated

File diff suppressed because it is too large Load Diff

6
config/event.php Normal file
View File

@ -0,0 +1,6 @@
<?php
// return [
// 'order.paySuccess' => [
// // ...其它事件处理函数...
// ]
// ];

View File

@ -14,5 +14,6 @@
return [
// '' => support\exception\Handler::class,
'' => app\common\exception\Handler::class,
// '' => app\common\exception\Handler::class,
'' => \App\ExceptionHandler::class,
];

View File

@ -1,4 +1,5 @@
<?php
/**
* This file is part of webman.
*
@ -13,7 +14,7 @@
*/
return [
'admin'=>[
'admin' => [
// 跨域中间件
app\common\http\middleware\AdminAllowMiddleware::class,
// 初始化
@ -22,5 +23,14 @@ return [
app\admin\middleware\LoginMiddleware::class,
// 权限认证
app\admin\middleware\AuthMiddleware::class,
]
];
],
// 'api' => [
// // 跨域中间件
// app\common\http\middleware\AdminAllowMiddleware::class,
// app\api\http\middleware\InitMiddleware::class,
// app\api\http\middleware\LoginMiddleware::class,
// ]
];

View File

@ -0,0 +1,83 @@
<?php
return [
'enable' => true,
'jwt' => [
/** 算法类型 HS256、HS384、HS512、RS256、RS384、RS512、ES256、ES384、Ed25519 */
'algorithms' => 'HS256',
/** access令牌秘钥 */
'access_secret_key' => '2022d3d3LmJq',
/** access令牌过期时间单位秒。默认 2 小时 */
'access_exp' => 7200,
/** refresh令牌秘钥 */
'refresh_secret_key' => '2022KTxigxc9o50c',
/** refresh令牌过期时间单位秒。默认 7 天 */
'refresh_exp' => 604800,
/** refresh 令牌是否禁用,默认不禁用 false */
'refresh_disable' => false,
/** 令牌签发者 */
'iss' => 'webman.tinywan.cn',
/** 某个时间点后才能访问单位秒。30 表示当前时间30秒后才能使用 */
'nbf' => 0,
/** 时钟偏差冗余时间,单位秒。建议这个余地应该不大于几分钟 */
'leeway' => 60,
/** 是否允许单设备登录,默认不允许 false */
'is_single_device' => false,
/** 缓存令牌时间,单位:秒。默认 7 天 */
'cache_token_ttl' => 604800,
/** 缓存令牌前缀,默认 JWT:TOKEN: */
'cache_token_pre' => 'JWT:TOKEN:',
/** 缓存刷新令牌前缀,默认 JWT:REFRESH_TOKEN: */
'cache_refresh_token_pre' => 'JWT:REFRESH_TOKEN:',
/** 用户信息模型 */
'user_model' => function ($uid) {
return [];
},
/** 是否支持 get 请求获取令牌 */
'is_support_get_token' => false,
/** GET 请求获取令牌请求key */
'is_support_get_token_key' => 'authorization',
/** access令牌私钥 */
'access_private_key' => <<<EOD
-----BEGIN RSA PRIVATE KEY-----
...
-----END RSA PRIVATE KEY-----
EOD,
/** access令牌公钥 */
'access_public_key' => <<<EOD
-----BEGIN PUBLIC KEY-----
...
-----END PUBLIC KEY-----
EOD,
/** refresh令牌私钥 */
'refresh_private_key' => <<<EOD
-----BEGIN RSA PRIVATE KEY-----
...
-----END RSA PRIVATE KEY-----
EOD,
/** refresh令牌公钥 */
'refresh_public_key' => <<<EOD
-----BEGIN PUBLIC KEY-----
...
-----END PUBLIC KEY-----
EOD,
],
];

Binary file not shown.

After

Width:  |  Height:  |  Size: 292 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 154 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 84 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1022 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 247 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 178 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 84 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 26 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 42 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 831 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 839 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.7 KiB

Some files were not shown because too many files have changed in this diff Show More