diff --git a/.env.example b/.env.example
index cd9d87d4..e2571180 100644
--- a/.env.example
+++ b/.env.example
@@ -48,4 +48,6 @@ MAILGUN_DOMAIN=
MAILGUN_SECRET=
REDIRECT_HTTPS=true
-BAIDU_APP_AK =
\ No newline at end of file
+BAIDU_APP_AK =
+
+JWT_SECRET=
diff --git a/app/Http/Controllers/Admin/Config/CategoryController.php b/app/Http/Controllers/Admin/Config/CategoryController.php
new file mode 100644
index 00000000..d90393b9
--- /dev/null
+++ b/app/Http/Controllers/Admin/Config/CategoryController.php
@@ -0,0 +1,72 @@
+all(), [
+ 'name' => 'required',
+ ]);
+
+ if ($validator->fails()) {
+ return Response::json(['status' => 'fail', 'message' => $validator->errors()->all()]);
+ }
+
+ if (GoodsCategory::create($validator->validated())) {
+ return Response::json(['status' => 'success', 'message' => '提交成功']);
+ }
+
+ return Response::json(['status' => 'fail', 'message' => '操作失败']);
+ }
+
+ // 编辑等级
+ public function update(Request $request, GoodsCategory $category)
+ {
+ $validator = Validator::make($request->all(), [
+ 'name' => 'required',
+ 'sort' => 'required|numeric',
+ ]);
+
+ if ($validator->fails()) {
+ return Response::json(['status' => 'fail', 'message' => $validator->errors()->all()]);
+ }
+ if ($category->update($validator->validated())) {
+ return Response::json(['status' => 'success', 'message' => '操作成功']);
+ }
+
+ return Response::json(['status' => 'fail', 'message' => '操作失败']);
+ }
+
+ // 删除等级
+ public function destroy(GoodsCategory $category)
+ {
+ // 校验该等级下是否存在关联账号
+ if ($category->goods()->exists()) {
+ return Response::json(['status' => 'fail', 'message' => '该分类下存在关联账号,请先取消关联']);
+ }
+
+ try {
+ if ($category->delete()) {
+ return Response::json(['status' => 'success', 'message' => '删除成功']);
+ }
+ } catch (Exception $e) {
+ Log::error('删除时报错:'.$e->getMessage());
+
+ return Response::json(['status' => 'fail', 'message' => '删除失败:'.$e->getMessage()]);
+ }
+
+ return Response::json(['status' => 'fail', 'message' => '删除失败']);
+ }
+}
diff --git a/app/Http/Controllers/Admin/ShopController.php b/app/Http/Controllers/Admin/ShopController.php
index 1160a0da..3dd36d82 100644
--- a/app/Http/Controllers/Admin/ShopController.php
+++ b/app/Http/Controllers/Admin/ShopController.php
@@ -6,6 +6,7 @@ use App\Http\Controllers\Controller;
use App\Http\Requests\Admin\ShopStoreRequest;
use App\Http\Requests\Admin\ShopUpdateRequest;
use App\Models\Goods;
+use App\Models\GoodsCategory;
use App\Models\Level;
use Arr;
use Exception;
@@ -37,7 +38,7 @@ class ShopController extends Controller
// 添加商品页面
public function create()
{
- return view('admin.shop.info', ['levels' => Level::orderBy('level')->get()]);
+ return view('admin.shop.info', ['levels' => Level::orderBy('level')->get(), 'categories' => GoodsCategory::all()]);
}
// 添加商品
@@ -92,6 +93,7 @@ class ShopController extends Controller
return view('admin.shop.info', [
'good' => $good,
'levels' => Level::orderBy('level')->get(),
+ 'categories' => GoodsCategory::all(),
]);
}
@@ -109,6 +111,7 @@ class ShopController extends Controller
return $path;
}
}
+
try {
$data['is_hot'] = array_key_exists('is_hot', $data) ? 1 : 0;
$data['status'] = array_key_exists('status', $data) ? 1 : 0;
diff --git a/app/Http/Controllers/AdminController.php b/app/Http/Controllers/AdminController.php
index 02c020a7..20c60fab 100644
--- a/app/Http/Controllers/AdminController.php
+++ b/app/Http/Controllers/AdminController.php
@@ -3,6 +3,7 @@
namespace App\Http\Controllers;
use App\Models\Country;
+use App\Models\GoodsCategory;
use App\Models\Invite;
use App\Models\Label;
use App\Models\Level;
@@ -118,6 +119,7 @@ class AdminController extends Controller
return view('admin.config.config', [
'methods' => SsConfig::type(1)->get(),
'protocols' => SsConfig::type(2)->get(),
+ 'categories' => GoodsCategory::all(),
'obfsList' => SsConfig::type(3)->get(),
'countries' => Country::all(),
'levels' => Level::all(),
diff --git a/app/Http/Controllers/Api/Client/V1Controller.php b/app/Http/Controllers/Api/Client/V1Controller.php
index bfeef54e..90579437 100644
--- a/app/Http/Controllers/Api/Client/V1Controller.php
+++ b/app/Http/Controllers/Api/Client/V1Controller.php
@@ -2,17 +2,30 @@
namespace App\Http\Controllers\Api\Client;
+use App\Components\Helpers;
use App\Http\Controllers\Controller;
+use App\Http\Controllers\PaymentController;
+use App\Models\Coupon;
use App\Models\Goods;
+use App\Models\GoodsCategory;
+use App\Models\Order;
+use App\Models\Payback;
+use App\Models\Payment;
+use App\Models\ReferralLog;
use App\Models\User;
+use Hashids\Hashids;
+use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
+use Illuminate\Support\Facades\Cache;
use Validator;
class V1Controller extends Controller
{
+ private static $method;
+
public function __construct()
{
- $this->middleware('auth:api', ['except' => ['login', 'register', 'shop']]);
+ $this->middleware('auth:api', ['except' => ['login', 'register', 'shop', 'config', 'getConfig']]);
auth()->shouldUse('api');
}
@@ -27,7 +40,7 @@ class V1Controller extends Controller
return response()->json(['ret' => 0, 'msg' => $validator->errors()->all()], 422);
}
- if ($token = auth()->attempt($validator->validated())) {
+ if ($token = auth('api')->attempt($validator->validated())) {
return $this->createNewToken($token);
}
@@ -37,18 +50,20 @@ class V1Controller extends Controller
protected function createNewToken($token)
{
return response()->json([
- 'ret' => 1,
- 'access_token' => $token,
- 'token_type' => 'bearer',
- 'expires_in' => auth()->factory()->getTTL() * 60,
- 'user' => auth()->user()->profile(),
+ 'ret' => 1,
+ 'data' => [
+ 'access_token' => $token,
+ 'token_type' => 'bearer',
+ 'expires_in' => auth('api')->factory()->getTTL() * 60,
+ 'user' => auth('api')->user()->profile(),
+ ],
]);
}
public function register(Request $request)
{
$validator = Validator::make($request->all(), [
- 'name' => 'required|string|between:2,100',
+ 'name' => 'required|string|between:2,100',
'username' => 'required|'.(sysConfig('username_type') ?? 'email').'|max:100|unique:user,username',
'password' => 'required|string|confirmed|min:6',
]);
@@ -67,46 +82,256 @@ class V1Controller extends Controller
public function logout()
{
- auth()->logout();
+ auth('api')->logout();
return response()->json(['ret' => 1]);
}
public function refresh()
{
- return $this->createNewToken(auth()->refresh());
+ return $this->createNewToken(auth('api')->refresh());
}
public function userProfile()
{
- return response()->json(auth()->user()->profile());
+ $user = auth('api')->user();
+ $userInfo = $user->profile();
+ $userInfo['subUrl'] = $user->subUrl();
+ $totalTransfer = $user->transfer_enable;
+ $usedTransfer = $user->used_traffic;
+ $unusedTraffic = $totalTransfer - $usedTransfer > 0 ? $totalTransfer - $usedTransfer : 0;
+ $userInfo['unusedTraffic'] = flowAutoShow($unusedTraffic);
+
+ return response()->json(['ret' => 1, 'data' => $userInfo]);
}
public function nodeList(int $id = null)
{
- $user = auth()->user();
+ $user = auth('api')->user();
$nodes = $user->nodes()->get();
- if (isset($id)) {
- $node = $nodes->find($id);
- if (empty($node)) {
- return response()->json([], 204);
- }
-
- return response()->json($node->config($user));
- }
- $servers = [];
- foreach ($nodes as $node) {
- $servers[] = $node->config($user);
- }
-
- return response()->json($servers);
+ return response()->json(['ret' => 1, 'data' => $nodes]);
}
public function shop()
{
- $shop = Goods::whereStatus(1)->where('type', '<=', '2')->orderByDesc('type')->orderByDesc('sort')->get();
+ $shops = [
+ 'keys' => [],
+ 'data' => [],
+ ];
+ $shop_plan = GoodsCategory::query()->where('status', 1)->get();
+ foreach ($shop_plan as $item) {
+ array_push($shops['keys'], $item['name']);
+ $shops['data'][$item['name']] = $item->goods()->get()->append('traffic_label')->toArray();
+ }
- return response()->json($shop);
+ return response()->json(['ret' => 1, 'data' => $shops]);
+ }
+
+ public function getConfig()
+ {
+ $config = config('bobclient');
+ $config['website_name'] = sysConfig('website_name');
+ $config['website_url'] = sysConfig('website_url');
+ $config['payment'] = [
+ 'alipay' => sysConfig('is_AliPay'),
+ 'wechat' => sysConfig('is_WeChatPay'),
+ ];
+
+ return response()->json(['ret' => 1, 'data' => $config]);
+ }
+
+ // 创建支付订单
+ public function purchase(Request $request)
+ {
+ $goods_id = $request->input('goods_id');
+ $coupon_sn = $request->input('coupon_sn');
+ self::$method = $request->input('method');
+ $credit = $request->input('amount');
+ $pay_type = $request->input('pay_type');
+ $amount = 0;
+
+ // 充值余额
+ if ($credit) {
+ if (! is_numeric($credit) || $credit <= 0) {
+ return response()->json(['ret' => 0, 'msg' => trans('user.payment.error')]);
+ }
+ $amount = $credit;
+ // 购买服务
+ } elseif ($goods_id && self::$method) {
+ $goods = Goods::find($goods_id);
+ if (! $goods || ! $goods->status) {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:商品已下架']);
+ }
+ $amount = $goods->price;
+
+ // 是否有生效的套餐
+ $activePlan = Order::userActivePlan()->doesntExist();
+
+ // 无生效套餐,禁止购买加油包
+ if ($goods->type === 1 && $activePlan) {
+ return response()->json(['ret' => 0, 'msg' => '购买加油包前,请先购买套餐']);
+ }
+
+ // 单个商品限购
+ if ($goods->limit_num) {
+ $count = Order::uid()->where('status', '>=', 0)->whereGoodsId($goods_id)->count();
+ if ($count >= $goods->limit_num) {
+ return response()->json(['ret' => 0, 'msg' => '此商品限购'.$goods->limit_num.'次,您已购买'.$count.'次']);
+ }
+ }
+
+ // 使用优惠券
+ if ($coupon_sn) {
+ $coupon = Coupon::whereStatus(0)->whereIn('type', [1, 2])->whereSn($coupon_sn)->first();
+ if (! $coupon) {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:优惠券不存在']);
+ }
+
+ // 计算实际应支付总价
+ $amount = $coupon->type === 2 ? $goods->price * $coupon->value / 100 : $goods->price - $coupon->value;
+ $amount = $amount > 0 ? round($amount, 2) : 0; // 四舍五入保留2位小数,避免无法正常创建订单
+ }
+
+ //非余额付款下,检查在线支付是否开启
+ if (self::$method !== 'credit') {
+ // 判断是否开启在线支付
+ if (! sysConfig('is_onlinePay')) {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:系统并未开启在线支付功能']);
+ }
+
+ // 判断是否存在同个商品的未支付订单
+ if (Order::uid()->whereStatus(0)->exists()) {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:尚有未支付的订单,请先去支付']);
+ }
+ } elseif (Auth::getUser()->credit < $amount) { // 验证账号余额是否充足
+ return response()->json(['ret' => 0, 'msg' => '您的余额不足,请先充值']);
+ }
+
+ // 价格异常判断
+ if ($amount < 0) {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:订单总价异常']);
+ }
+
+ if ($amount === 0 && self::$method !== 'credit') {
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败:订单总价为0,无需使用在线支付']);
+ }
+ }
+
+ // 生成订单
+ try {
+ $newOrder = Order::create([
+ 'sn' => date('ymdHis').random_int(100000, 999999),
+ 'user_id' => auth()->id(),
+ 'goods_id' => $credit ? null : $goods_id,
+ 'coupon_id' => $coupon->id ?? null,
+ 'origin_amount' => $credit ?: $goods->price ?? 0,
+ 'amount' => $amount,
+ 'pay_type' => $pay_type,
+ 'pay_way' => self::$method,
+ ]);
+
+ // 使用优惠券,减少可使用次数
+ if (! empty($coupon)) {
+ if ($coupon->usable_times > 0) {
+ $coupon->decrement('usable_times', 1);
+ }
+
+ Helpers::addCouponLog('订单支付使用', $coupon->id, $goods_id, $newOrder->id);
+ }
+
+ $request->merge(['id' => $newOrder->id, 'type' => $pay_type, 'amount' => $amount]);
+ PaymentController::$method = self::$method;
+ // 生成支付单
+ $data = PaymentController::getClient()->purchase($request);
+ $data = $data->getData(true);
+ $data['order_id'] = $newOrder->id;
+
+ return response()->json($data);
+ } catch (Exception $e) {
+ Log::error('订单生成错误:'.$e->getMessage());
+ }
+
+ return response()->json(['ret' => 0, 'msg' => '订单创建失败']);
+ }
+
+ /**
+ * @param Request $request
+ * @return JsonResponse
+ */
+ public static function getStatus(Request $request): JsonResponse
+ {
+ $order_id = $request->input('order_id');
+ $payment = Order::query()->find($order_id)->payment;
+ if ($payment) {
+ if ($payment->status === 1) {
+ return response()->json(['ret' => 1, 'msg' => '支付成功']);
+ }
+
+ if ($payment->status === -1) {
+ return response()->json(['ret' => 0, 'msg' => '订单超时未支付,已自动关闭']);
+ }
+
+ return response()->json(['ret' => 0, 'msg' => '等待支付']);
+ }
+
+ return response()->json(['ret' => 0, 'msg' => '未知订单']);
+ }
+
+ public function gift(Request $request)
+ {
+ $user = $request->user('api');
+ $referral_traffic = flowAutoShow(sysConfig('referral_traffic') * MB);
+ $referral_percent = sysConfig('referral_percent');
+ // 邀请码
+ $code = $user->invites()->whereStatus(1)->value('code');
+
+ $data['invite_gift'] = trans('user.invite.promotion', [
+ 'traffic' => $referral_traffic,
+ 'referral_percent' => $referral_percent * 100,
+ ]);
+ $affSalt = sysConfig('aff_salt');
+ if (isset($affSalt)) {
+ $aff_link = route('register', ['aff' => (new Hashids($affSalt, 8))->encode($user->id)]);
+ } else {
+ $aff_link = route('register', ['aff' => $user->id]);
+ }
+ $data['invite_url'] = $aff_link;
+ $data['invite_text'] = $aff_link.'&(复制整段文字到浏览器打开即可访问),找梯子最重要的就是稳定,这个已经上线三年了,一直稳定没有被封过,赶紧下载备用吧!安装后打开填写我的邀请码【'.$code.'】,你还能多得3天会员.';
+ // 累计数据
+ $data['back_sum'] = ReferralLog::query()->where('inviter_id', $user->id)->sum('commission') / 100;
+ $data['user_sum'] = $user->invitees()->count();
+ $data['list'] = $user->invitees()->selectRaw('username, UNIX_TIMESTAMP(created_at) as created_at')->limit(10)->get();
+
+ return response()->json(['ret' => 1, 'data' => $data]);
+ }
+
+ public function checkIn(Request $request): JsonResponse
+ {
+ $user = $request->user();
+ // 系统开启登录加积分功能才可以签到
+ if (! sysConfig('is_checkin')) {
+ return response()->json(['ret' => 0, 'title' => trans('common.failed'), 'msg' => trans('user.home.attendance.disable')]);
+ }
+
+ // 已签到过,验证是否有效
+ if (Cache::has('userCheckIn_'.$user->id)) {
+ return response()->json(['ret' => 0, 'title' => trans('common.success'), 'msg' => trans('user.home.attendance.done')]);
+ }
+
+ $traffic = random_int((int) sysConfig('min_rand_traffic'), (int) sysConfig('max_rand_traffic')) * MB;
+
+ if (! $user->incrementData($traffic)) {
+ return response()->json(['ret' => 0, 'title' => trans('common.failed'), 'msg' => trans('user.home.attendance.failed')]);
+ }
+
+ // 写入用户流量变动记录
+ Helpers::addUserTrafficModifyLog($user->id, null, $user->transfer_enable, $user->transfer_enable + $traffic, trans('user.home.attendance.attribute'));
+
+ // 多久后可以再签到
+ $ttl = sysConfig('traffic_limit_time') ? sysConfig('traffic_limit_time') * Minute : Day;
+ Cache::put('userCheckIn_'.$user->id, '1', $ttl);
+
+ return response()->json(['ret' => 1, 'msg' => trans('user.home.attendance.success', ['data' => flowAutoShow($traffic)])]);
}
}
diff --git a/app/Http/Controllers/PaymentController.php b/app/Http/Controllers/PaymentController.php
index 1982fb03..143f3319 100644
--- a/app/Http/Controllers/PaymentController.php
+++ b/app/Http/Controllers/PaymentController.php
@@ -26,7 +26,7 @@ use Response;
class PaymentController extends Controller
{
- private static $method;
+ public static $method;
public static function notify(Request $request): void
{
diff --git a/app/Models/GoodsCategory.php b/app/Models/GoodsCategory.php
new file mode 100644
index 00000000..2c49e813
--- /dev/null
+++ b/app/Models/GoodsCategory.php
@@ -0,0 +1,17 @@
+hasMany(Goods::class, 'category_id');
+ }
+}
diff --git a/config/bobclient.php b/config/bobclient.php
new file mode 100644
index 00000000..24b3dfbf
--- /dev/null
+++ b/config/bobclient.php
@@ -0,0 +1,53 @@
+ [
+ 'telegram_url' => 'https://t.me/Bobs9', // 留空的话则不展示telegram群
+ 'qq_url' => 'https://t.me/Bobs9', // 留空的话则不展示QQ群
+ 'background_img' => 'https://shige.group/such/pic.php/forum/pic/item/00e93901213fb80e3d28759b21d12f2eb8389484/mlike.jpg', // 背景图片地址,图片宽高不超过 860px * 544px 就行 (留空为默认的背景图)
+ 'text' => '一键开启
极速上网体验',
+ 'text_color' => 'rgba(255, 255, 255, 0.8);', // 文字和按钮颜色 默认颜色 rgba(255, 255, 255, 0.8);
+ 'button_color' => '#8077f1', // 文字和按钮颜色 默认颜色:#8077f1(v2版本配置)
+ ],
+
+ // PC端消息中心图片和跳转链接
+ 'message' => [
+ 'background_img' => 'https://malus.s3cdn.net/uploads/malus_user-guide.jpg', // 背景图片地址
+ 'url' => 'https://www.goole.com', // 跳转链接
+ ],
+
+ // Crisp在线客服
+ 'crisp_enable' => false, // 是否开启
+ 'crisp_id' => '2c3c28c2-9265-45ea-8e85-0xxxxx', // Crisp 的网站ID
+
+ // 弹窗公告
+ 'notice' => [
+ 'is_start' => true, // 是否开启弹窗公告
+ 'title' => '最新公告', // 标题
+ 'content' => '这是最新 公告 内容', // 公告内容,可以为html格式,也可以纯文本
+ ],
+
+ // PC端菜单栏显示控制
+ 'menu' => [
+ 'shop' => true, // 会员
+ 'user' => true, // 我的
+ 'gift' => true, // 邀请
+ ],
+
+ // 检查用户计算机时间
+ 'check_time' => [
+ 'is_check' => true, // 是否开启检查
+ 'differ_time' => 90, // 相差多少秒提示
+ 'warning_text' => '请校准系统时间为北京时间,否则会导致无法上网!', // 提示内容
+ ],
+
+ // 个人中心头像
+ 'user_avatar' => 'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=4109802972,297162689&fm=11&gp=0.jpg',
+];
diff --git a/database/migrations/2021_07_24_214642_create_goods_category_table.php b/database/migrations/2021_07_24_214642_create_goods_category_table.php
new file mode 100644
index 00000000..29e69c99
--- /dev/null
+++ b/database/migrations/2021_07_24_214642_create_goods_category_table.php
@@ -0,0 +1,44 @@
+id();
+ $table->string('name')->default('')->comment('分类名称');
+ $table->tinyInteger('status')->default('1')->comment('状态 0:隐藏 1:显示');
+ $table->integer('sort')->default('0')->comment('排序');
+ $table->timestamps();
+ });
+ Schema::table('goods', function (Blueprint $table) {
+ $table->integer('category_id')->default(1)->nullable()->comment('分类ID');
+ });
+ \App\Models\GoodsCategory::query()->create(['name' => '黄金套餐']);
+ \App\Models\GoodsCategory::query()->create(['name' => '白金套餐']);
+ \App\Models\GoodsCategory::query()->create(['name' => '钻石套餐']);
+ }
+
+ /**
+ * Reverse the migrations.
+ *
+ * @return void
+ */
+ public function down()
+ {
+ Schema::dropIfExists('goods_category');
+ Schema::table('goods', function (Blueprint $table) {
+ $table->dropColumn(['category_id']);
+ });
+ }
+}
diff --git a/resources/views/admin/config/config.blade.php b/resources/views/admin/config/config.blade.php
index cb431726..8b270548 100644
--- a/resources/views/admin/config/config.blade.php
+++ b/resources/views/admin/config/config.blade.php
@@ -30,6 +30,9 @@
| 名称 | +排序 | +{{trans('common.action')}} | +
|---|---|---|
| + + | ++ + | +
+
+
+
+
+ |
+