feat: spu

This commit is contained in:
2025-01-06 18:02:57 +08:00
parent 0895955973
commit fd18cffeab
12 changed files with 505 additions and 6 deletions

View File

@@ -48,4 +48,12 @@ class CommonRedisKey
{ {
return '__system:wx:AccessToken:'; return '__system:wx:AccessToken:';
} }
/**
* @return string
*/
public static function getCycleList(): string
{
return '__cycle:list';
}
} }

View File

@@ -0,0 +1,46 @@
<?php
namespace App\Cache\Redis\Common;
use App\Cache\Redis\RedisCache;
use App\Model\Cycle;
use Hyperf\Di\Annotation\Inject;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
class CycleCache
{
/**
* @var RedisCache
*/
#[Inject]
protected RedisCache $redis;
/**
* @var Cycle $cycleModel
*/
#[Inject]
protected Cycle $cycleModel;
/**
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
public function setAllCycleCache(): void
{
$key = CommonRedisKey::getCycleList();
if ($this->redis->exists($key,'system')) return;
$allData = $this->cycleModel->get();
if (empty($allData)) return;
$allData = $allData->toArray();
foreach ($allData as $item) {
$this->redis->zAdd($key, $item['id'],$item['dates'],'system');
}
}
}

View File

@@ -540,6 +540,19 @@ class RedisCache
{ {
return $this->getRedis($poolName)->zRem($key, $value); return $this->getRedis($poolName)->zRem($key, $value);
} }
/**
* @param $key
* @param $value
* @param string $poolName
* @return false|int|Redis
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
public function zRank($key, $value, string $poolName = 'default')
{
return $this->getRedis($poolName)->zRank($key, $value);
}
// +-------------------------------------------------------------------------------------------------------------------------------------------- // +--------------------------------------------------------------------------------------------------------------------------------------------
// | geo // | geo
// +-------------------------------------------------------------------------------------------------------------------------------------------- // +--------------------------------------------------------------------------------------------------------------------------------------------

View File

@@ -7,6 +7,7 @@ namespace App\Controller\Admin;
use App\Middleware\Admin\JwtAuthMiddleware; use App\Middleware\Admin\JwtAuthMiddleware;
use App\Request\Admin\DriverRequest; use App\Request\Admin\DriverRequest;
use App\Request\Admin\GoodRequest; use App\Request\Admin\GoodRequest;
use App\Service\Admin\Good\SpuService;
use Hyperf\HttpServer\Annotation\Controller; use Hyperf\HttpServer\Annotation\Controller;
use Hyperf\HttpServer\Annotation\Middlewares; use Hyperf\HttpServer\Annotation\Middlewares;
use Hyperf\HttpServer\Annotation\RequestMapping; use Hyperf\HttpServer\Annotation\RequestMapping;
@@ -27,7 +28,7 @@ class GoodController
#[Scene(scene: "add_spu")] #[Scene(scene: "add_spu")]
public function add_spu(GoodRequest $request) public function add_spu(GoodRequest $request)
{ {
return (new SpuService())->add();
} }
/** /**

View File

@@ -10,9 +10,16 @@ declare(strict_types=1);
namespace App\Cron\Good; namespace App\Cron\Good;
use App\Cache\Redis\Common\CommonRedisKey;
use App\Cache\Redis\Common\CycleCache;
use App\Cache\Redis\RedisCache;
use App\Extend\DateUtil;
use App\Lib\Log;
use App\Model\Cycle; use App\Model\Cycle;
use Hyperf\Crontab\Annotation\Crontab; use Hyperf\Crontab\Annotation\Crontab;
use Hyperf\Di\Annotation\Inject; use Hyperf\Di\Annotation\Inject;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
#[Crontab(rule: "0 5 * * *", name: "CycleCreateTask", singleton: true , callback: "execute", memo: "创建新的周期任务")] #[Crontab(rule: "0 5 * * *", name: "CycleCreateTask", singleton: true , callback: "execute", memo: "创建新的周期任务")]
class CycleCreateTask class CycleCreateTask
@@ -23,11 +30,72 @@ class CycleCreateTask
#[Inject] #[Inject]
protected Cycle $cycleModel; protected Cycle $cycleModel;
public function execute() /**
{ * @var CycleCache $cycleCache
//todo Write logic */
var_dump(date('Y-m-d H:i:s', time())); #[Inject]
protected CycleCache $cycleCache;
$maxDate = $this->cycleModel->max('date'); /**
* @var RedisCache
*/
#[Inject]
protected RedisCache $redis;
/**
* @var Log
*/
#[Inject]
protected Log $log;
/**
* 非常重要的逻辑--生成周期 (不可删除)
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
public function execute(): void
{
$maxDate = $this->cycleModel->max('dates') ?? 0;
$nextMax = date('Ymd',strtotime('+3 day',time()));
if ($maxDate != 0 && $nextMax <= $maxDate) {
return;
}
$nextCycleRange = DateUtil::getDateFromRange(date('Ymd'),$nextMax);
$this->cycleCache->setAllCycleCache();
$key = CommonRedisKey::getCycleList();
$noDataInsertArr = [];
$noDataArr = [];
foreach ($nextCycleRange as $date) {
$one = $this->redis->zScore($key,$date,'system');
if ($one) continue;
$noDataInsertArr[] = [
'dates' => $date,
'create_time' => date('Y-m-d H:i:s'),
];
$noDataArr[] = $date;
}
$insertSql = (new Cycle)->insert($noDataInsertArr);
if (!$insertSql) $this->log->error('添加周期失败!!!');
$res = $this->cycleModel->whereIn('dates',$noDataArr)->get();
if (empty($res)) return;
$res = $res->toArray();
foreach ($res as $one) {
$this->redis->zAdd($key,$one['id'],$one['dates'],'system');
}
$this->log->notice('添加周期成功,添加的周期为'.json_encode($noDataArr,JSON_UNESCAPED_UNICODE));
} }
} }

View File

@@ -96,4 +96,32 @@ class DateUtil
$msTime = (float)sprintf('%.0f', (floatval($ms) + floatval($sec)) * 1000); $msTime = (float)sprintf('%.0f', (floatval($ms) + floatval($sec)) * 1000);
return substr($msTime,0,13); return substr($msTime,0,13);
} }
/**
* 获取指定日期段内每一天的日期 (开始时间必须小于结束时间)
* @param string $startDate
* @param string $endDate
* @param string $returnFormat
* @return array
*/
static function getDateFromRange(string $startDate, string $endDate, string $returnFormat = 'Y-m-d'): array
{
if (empty($startDate) || empty($endDate)) return [];
if ($endDate < $startDate) return [];
$sTimestamp = strtotime($startDate);
$eTimestamp = strtotime($endDate);
// 计算日期段内有多少天
$days = ($eTimestamp-$sTimestamp)/86400+1;
// 保存每天日期
$date = [];
for($i=0; $i<$days; $i++){
$date[] = date($returnFormat, $sTimestamp+(86400*$i));
}
return $date;
}
} }

View File

@@ -4,6 +4,7 @@ declare(strict_types=1);
namespace App\Model; namespace App\Model;
use Hyperf\Database\Model\Builder;
use Hyperf\DbConnection\Model\Model; use Hyperf\DbConnection\Model\Model;
/** /**
@@ -24,8 +25,22 @@ class Cycle extends Model
*/ */
protected array $fillable = []; protected array $fillable = [];
protected array $guarded = [];
/** /**
* The attributes that should be cast to native types. * The attributes that should be cast to native types.
*/ */
protected array $casts = ['id' => 'integer', 'is_use' => 'integer']; protected array $casts = ['id' => 'integer', 'is_use' => 'integer'];
const string CREATED_AT = 'create_time';
const null UPDATED_AT = null;
/**
* @param string $date
* @return Builder|\Hyperf\Database\Model\Model|null
*/
public function getInfoByDate(string $date): \Hyperf\Database\Model\Model|Builder|null
{
return $this->where('dates',$date)->first();
}
} }

44
app/Model/Sku.php Normal file
View File

@@ -0,0 +1,44 @@
<?php
declare(strict_types=1);
namespace App\Model;
use Hyperf\DbConnection\Model\Model;
/**
* @property int $id
* @property int $spu_id
* @property string $title
* @property string $image_ids
* @property string $price
* @property string $param
* @property string $extra
* @property int $total_stock
* @property int $surplus_stock
* @property int $sales_num
* @property int $order_num
* @property int $cancel_num
* @property int $ahead_refund_num
* @property int $behind_refund_num
* @property int $saleable
* @property string $create_time
* @property string $update_time
*/
class Sku extends Model
{
/**
* The table associated with the model.
*/
protected ?string $table = 'sku';
/**
* The attributes that are mass assignable.
*/
protected array $fillable = [];
/**
* The attributes that should be cast to native types.
*/
protected array $casts = ['id' => 'integer', 'spu_id' => 'integer', 'total_stock' => 'integer', 'surplus_stock' => 'integer', 'sales_num' => 'integer', 'order_num' => 'integer', 'cancel_num' => 'integer', 'ahead_refund_num' => 'integer', 'behind_refund_num' => 'integer', 'saleable' => 'integer'];
}

63
app/Model/Spu.php Normal file
View File

@@ -0,0 +1,63 @@
<?php
declare(strict_types=1);
namespace App\Model;
use Hyperf\Database\Model\Builder;
use Hyperf\DbConnection\Model\Model;
/**
* @property int $id
* @property int $cycle_id
* @property int $city_id
* @property int $kitchen_id
* @property int $chef_id
* @property string $title
* @property string $sub_title
* @property int $category_id
* @property int $saleable
* @property string $create_time
* @property string $update_time
*/
class Spu extends Model
{
/**
* The table associated with the model.
*/
protected ?string $table = 'spu';
/**
* The attributes that are mass assignable.
*/
protected array $fillable = [];
protected array $guarded = [];
/**
* The attributes that should be cast to native types.
*/
protected array $casts = ['id' => 'integer', 'cycle_id' => 'integer', 'city_id' => 'integer', 'kitchen_id' => 'integer','chef_id' => 'integer', 'category_id' => 'integer', 'saleable' => 'integer'];
const string CREATED_AT = 'create_time';
const string UPDATED_AT = 'update_time';
/**
* @param int $cityId
* @param int $cycleId
* @return Builder|\Hyperf\Database\Model\Model|null
*/
public function getInfoByCityIdAndCycleId(int $cityId, int $cycleId): \Hyperf\Database\Model\Model|Builder|null
{
return $this->where('city_id', $cityId)->where('cycle_id', $cycleId)->first();
}
/**
* @param int $id
* @return Builder|\Hyperf\Database\Model\Model|null
*/
public function getInfoById(int $id): \Hyperf\Database\Model\Model|Builder|null
{
return $this->where('id', $id)->first();
}
}

View File

@@ -0,0 +1,21 @@
<?php
/**
* This service file is part of item.
*
* @author ctexthuang
* @contact ctexthuang@qq.com
*/
declare(strict_types=1);
namespace App\Service\Admin\Good;
use App\Service\Admin\BaseService;
class SkuService extends BaseService
{
public function handle()
{
//todo Write logic
}
}

View File

@@ -0,0 +1,187 @@
<?php
/**
* This service file is part of item.
*
* @author ctexthuang
* @contact ctexthuang@qq.com
*/
declare(strict_types=1);
namespace App\Service\Admin\Good;
use App\Constants\Admin\UserCode;
use App\Constants\Common\SiteCode;
use App\Exception\ErrException;
use App\Model\AdminUser;
use App\Model\Cycle;
use App\Model\Kitchen;
use App\Model\Sku;
use App\Model\Spu;
use App\Service\Admin\BaseService;
use App\Service\ServiceTrait\Admin\GetUserInfoTrait;
use App\Service\ServiceTrait\Common\OssTrait;
use Hyperf\Di\Annotation\Inject;
class SpuService extends BaseService
{
use GetUserInfoTrait,OssTrait;
private $userInfo;
private int $cityId;
/**
* @var Spu
*/
#[Inject]
protected Spu $spuModel;
/**
* @var Sku
*/
#[Inject]
protected Sku $skuModel;
public function __construct()
{
parent::__construct();
$this->userInfo = $this->getUserInfo($this->adminId);
$this->cityId = (int)$this->userInfo['city_id'];
}
public function handle()
{
}
/**
* @var Cycle
*/
#[Inject]
protected Cycle $cycleModel;
/**
* 添加 spu
* @return array
*/
public function add(): array
{
$date = $this->request->input('date',date('Y-m-d'));
$cycleInfo = $this->cycleModel->getInfoByDate($date);
if (empty($cycleInfo)) throw new ErrException('没有该周期,请刷新后重新上传');
$info = $this->spuModel->getInfoByCityIdAndCycleId($this->cityId, $cycleInfo->id);
if (!empty($info)) throw new ErrException('该菜品在当前城市已存在');
$this->checkInfo();
$insertModel = new Spu();
$insertModel->city_id = $this->cityId;
$insertModel->cycle_id = $cycleInfo->id;
$insertModel->kitchen_id = $this->request->input('kitchen_id');
$insertModel->chef_id = $this->request->input('chef_id');
$insertModel->title = $this->request->input('title');
$insertModel->sub_title = $this->request->input('sub_title','');
$insertModel->category_id = $this->request->input('category_id');
$insertModel->saleable = $this->request->input('saleable');
if (!$insertModel->save()) throw new ErrException('添加菜品失败');
return $this->return->success();
}
/**
* @var Kitchen
*/
#[Inject]
protected Kitchen $kitchenModel;
/**
* @var AdminUser
*/
#[Inject]
protected AdminUser $adminUserModel;
/**
* 信息检测
* @return void
*/
private function checkInfo(): void
{
$kitchenId = $this->request->input('kitchen_id');
$kitchenInfo = $this->kitchenModel->getInfoById($kitchenId);
if ($kitchenInfo->status == SiteCode::KITCHEN_DISABLE) throw new ErrException('该厨房已禁用');
$chefId = $this->request->input('chef_id');
$chefInfo = $this->adminUserModel->getAdminInfoById($chefId);
if ($chefInfo->status == UserCode::DISABLE) throw new ErrException('该厨师已禁用');
}
/**
* 修改
* @return array
*/
public function edit(): array
{
$editId = $this->request->input('edit_id');
$info = $this->spuModel->getInfoById($editId);
if (empty($info)) throw new ErrException('数据不存在');
$this->checkInfo();
$info->kitchen_id = $this->request->input('kitchen_id');
$info->chef_id = $this->request->input('chef_id');
$info->title = $this->request->input('title');
$info->sub_title = $this->request->input('sub_title','');
$info->category_id = $this->request->input('category_id');
$info->saleable = $this->request->input('saleable');
if (!$info->save()) throw new ErrException('修改菜品失败');
return $this->return->success();
}
/**
* spu 删除
* @return array
* @throws \Exception
*/
public function del(): array
{
$id = $this->request->input('id');
$info = $this->spuModel->getInfoById($id);
if (empty($info)) throw new ErrException('数据已删除');
//todo 需要联动删除sku 已经有用户下单sku该怎么办
//没有直接删除菜
$info->delete();
return $this->return->success();
}
/**
* spu 详情
* @return array
*/
public function view(): array
{
$id = $this->request->input('id');
$info = $this->spuModel->getInfoById($id);
if (empty($info)) throw new ErrException('数据不存在');
return $this->return->success('success',$info->toArray());
}
}

View File

@@ -8,6 +8,11 @@ account=13632877014&password=123456
client.global.set("admin_token", response.body.data.token); client.global.set("admin_token", response.body.data.token);
%} %}
### test
GET {{host}}/admin/third/sts/test
Content-Type: application/x-www-form-urlencoded
Authorization: Bearer {{admin_token}}
###添加城市 ###添加城市
POST {{host}}/admin/city/add POST {{host}}/admin/city/add