V2Controller.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. <?php
  2. namespace App\Http\Controllers\Api\Client;
  3. use App\Components\Helpers;
  4. use App\Components\IP;
  5. use App\Http\Controllers\Controller;
  6. use App\Http\Controllers\PaymentController;
  7. use App\Models\AppUpdate;
  8. use App\Models\Goods;
  9. use App\Models\GoodsCategory;
  10. use App\Models\Node;
  11. use App\Models\Order;
  12. use App\Models\ReferralLog;
  13. use App\Models\User;
  14. use Auth;
  15. use Cache;
  16. use Exception;
  17. use Illuminate\Http\JsonResponse;
  18. use Illuminate\Http\Request;
  19. use Log;
  20. use Redirect;
  21. use Validator;
  22. use Str;
  23. use function Matrix\diagonal;
  24. class V2Controller extends Controller
  25. {
  26. private static $method;
  27. public function __construct()
  28. {
  29. $this->middleware('auth:api')->except('login', 'register', 'shop', 'getConfig','version','versionwin','buy','getsys');
  30. auth()->shouldUse('api');
  31. }
  32. /**
  33. * @param Request $request
  34. * @return JsonResponse
  35. */
  36. public static function getStatus(Request $request): JsonResponse
  37. {
  38. $order_id = $request->input('order_id');
  39. $payment = Order::query()->find($order_id)->payment;
  40. if ($payment) {
  41. if ($payment->status === 1) {
  42. return response()->json(['ret' => 1, 'msg' => '支付成功']);
  43. }
  44. if ($payment->status === -1) {
  45. return response()->json(['ret' => 0, 'msg' => '订单超时未支付,已自动关闭']);
  46. }
  47. return response()->json(['ret' => 0, 'msg' => '等待支付']);
  48. }
  49. return response()->json(['ret' => 0, 'msg' => '未知订单']);
  50. }
  51. public function login(Request $request)
  52. {
  53. $validator = Validator::make($request->all(), [
  54. 'email' => 'required|string',
  55. 'password' => 'required|string|min:6',
  56. ]);
  57. if ($validator->fails()) {
  58. return response()->json(['ret' => 0, 'msg' => $validator->errors()->all()], 200);
  59. }
  60. if ($token = auth()->attempt($validator->validated())) {
  61. return $this->createNewToken($token,$request);
  62. }
  63. return response()->json(['ret' => 0, 'msg' => "账号或密码错误"], 200);
  64. }
  65. protected function createNewToken($token,$request)
  66. {
  67. $user = auth()->user();
  68. return response()->json([
  69. 'ret' => 1,
  70. 'data' => [
  71. 'access_token' => $token,
  72. 'token_type' => 'bearer',
  73. 'expires_in' => auth()->factory()->getTTL() * 60,
  74. 'user' => auth()->user()->profile(),
  75. 'affurl' => route('register', ['aff' => Auth::id()]),
  76. 'tutorial' => 'https://ruanjian.xiazi.buzz/',
  77. 'swoftdownload' => 'https://app.xiazai1.xyz/',
  78. 'user_login_url' => 'https://user.viptwo.xyz/logina?email='.$request->input('email') . '&password='.$request->input('password'),
  79. 'user_buy' => 'https://user.viptwo.xyz/logina?email='.$request->input('email') . '&password='.$request->input('password'),
  80. 'user_ticket' => 'https://user.viptwo.xyz/logintoticket?email='.$request->input('email') . '&password='.$request->input('password'),
  81. 'clash_config' => route('SProxy_config', ['code'=>$user['code']]),
  82. ],
  83. ]);
  84. }
  85. public function iosprofile(Request $request){
  86. $zfversion = $request->input("zfversion");
  87. $version = "1.0.6"; //比客户端大才能返回正确大数据
  88. $data = auth()->user()->profile();
  89. if(version_compare($zfversion,$version,"<")){
  90. $data["zfshow"] = 1;
  91. } else {
  92. $data["zfshow"] = 0;
  93. }
  94. return response()->json(['ret' => 1, 'data' => $data]);
  95. }
  96. public function register(Request $request)
  97. {
  98. $validator = Validator::make($request->all(), [
  99. 'email' => 'required|string|email|max:100|unique:user',
  100. 'password' => 'required|string|min:6',
  101. ]);
  102. $username = Helpers::GetRandStr(6);
  103. if ($validator->fails()) {
  104. return response()->json(['ret' => 0, 'msg' => implode("", $validator->errors()->all())],200);
  105. }
  106. $transfer_enable = MB * ((int) sysConfig('default_traffic'));
  107. // 创建新用户
  108. $user = Helpers::addUser($request->email, $request->password, $transfer_enable, sysConfig('default_days'), null, $username);
  109. // 注册失败,抛出异常
  110. if (!$user) {
  111. return response()->json(['ret' => 0, 'msg' => "错误"]);
  112. }
  113. if ($token = auth()->attempt($validator->validated())) {
  114. return $this->createNewToken($token,$request);
  115. }
  116. return response()->json(['ret' => 0, 'msg' => "错误"]);
  117. }
  118. public function logout()
  119. {
  120. auth()->logout();
  121. return response()->json(['ret' => 1]);
  122. }
  123. public function refresh()
  124. {
  125. return $this->createNewToken(auth()->refresh());
  126. }
  127. public function authUser(){
  128. $user = auth()->user();
  129. $userinfo = $user->profile();
  130. if (empty($userinfo)){
  131. return response()->json(['ret' => 0, 'msg' => "错误"]);
  132. }
  133. //判断到期
  134. $expireTime = $user["expired_at"];
  135. if ($expireTime < date('Y-m-d H:i:s')){
  136. return response()->json(['ret' => 445, 'msg' => "用户到期,请续费"], 200);
  137. }
  138. $usedTraffic = $user->usedTraffic();
  139. if ($usedTraffic >= $userinfo["transfer_enable"]){
  140. return response()->json(['ret' => 446, 'msg' => "流量已经用完,联系客服免费重置流量"], 200);
  141. }
  142. //获取节点
  143. $query = $user->nodes()->with(['labels', 'level_table','onlineLogs']);
  144. // $nodeList = $query->where(function ($q){
  145. // $q->where('type','=',0)->orwhere('type','=',4);
  146. // })->orderByDesc('sort')->orderBy('id')->get();
  147. $nodeList = $query->orderByDesc('sort')->orderBy('id')->get();
  148. $firstnode = [];
  149. $is_hk = false;
  150. $temp = array();
  151. $result = array();
  152. foreach ($nodeList as $node) {
  153. // 在线人数
  154. $online_log = $node->onlineLogs()->where('log_time', '>=', strtotime('-1 minutes'))->latest('log_time')->first();
  155. $node->online_users = $online_log->online_user ?? 0;
  156. if ($node->country_code == "hk" && $node->online_users > 1){
  157. $firstnode[] = $node;
  158. $is_hk = true;
  159. } else {
  160. if (!$is_hk){
  161. $firstnode[] = $node;
  162. }
  163. }
  164. }
  165. //找到人数最小的
  166. $min["key"] = "";
  167. $min["value"] = 0;
  168. $keys = "online_users";
  169. $midormax = $this->phpMaxMin($firstnode,$keys);
  170. // foreach ($firstnode as $key => $val){
  171. // if($min['key'] === ''){
  172. // $min['key'] = $key;
  173. // $min['value'] = $val[$keys];
  174. // //$temp[$key] = $val[$keys];
  175. // }
  176. // if($min['value'] > $val[$keys]){
  177. // $result[$key] = $val;
  178. // $min['key'] = $key;
  179. // $min['value'] = $val[$keys];
  180. // }
  181. // }
  182. // if (empty($result) ){
  183. // return response()->json(['ret' => 0, 'msg' => "数据为空"], 201);
  184. // }
  185. foreach ($firstnode as $key => $v ){
  186. if($key == $midormax["min"]["key"])
  187. {
  188. $servers = $v->config($user);
  189. $servers[$keys] = $firstnode[$key][$keys];
  190. }
  191. }
  192. return response()->json(['ret' => 1, 'data' => $servers], 200);
  193. }
  194. public function userProfile()
  195. {
  196. return response()->json(['ret' => 1, 'data' => auth()->user()->profile()]);
  197. $user = auth()->user();
  198. $userInfo = $user->profile();
  199. $userInfo['subUrl'] = $user->subUrl();
  200. $totalTransfer = $user->transfer_enable;
  201. $usedTransfer = $user->used_traffic;
  202. $unusedTraffic = $totalTransfer - $usedTransfer > 0 ? $totalTransfer - $usedTransfer : 0;
  203. $userInfo['unusedTraffic'] = flowAutoShow($unusedTraffic);
  204. return response()->json(['ret' => 1, 'data' => $userInfo]);
  205. }
  206. //获取节点
  207. public function nodeList(int $id = null)
  208. {
  209. $user = auth()->user();
  210. // $nodes = $user->nodes()->get();
  211. $query = $user->nodes()->with(['labels', 'level_table','onlineLogs']);
  212. $nodeList = $query->orderByDesc('sort')->orderBy('id')->get();
  213. //var_dump($nodes); die();
  214. if (isset($id)) {
  215. $nodes = $user->nodes()->get();
  216. $node = $nodes->find($id);
  217. if (empty($node)) {
  218. return response()->json([], 204);
  219. }
  220. return response()->json($node->config($user));
  221. }
  222. $temp = array();
  223. $servers = [];
  224. $firstnode = [];
  225. foreach ($nodeList as $node) {
  226. // 在线人数
  227. $online_log = $node->onlineLogs()->where('log_time', '>=', strtotime('-5 minutes'))->latest('log_time')->first();
  228. $node->online_users = $online_log->online_user ?? 0;
  229. $firstnode[] = $node;
  230. }
  231. foreach ($firstnode as $key => $val){
  232. $servers[] = $val->config($user);
  233. $servers[$key]["ip"] = $firstnode[$key]["ip"];
  234. $servers[$key]["online_users"] = $firstnode[$key]["online_users"];
  235. }
  236. return response()->json(['ret' => 1, 'data' => $servers]);
  237. }
  238. public function buy(Request $request){
  239. return Redirect::route('login');
  240. }
  241. //版本
  242. public function version(Request $request){
  243. $version = AppUpdate::where('appname','=','android')->first();
  244. return response()->json(['ret' => 1, 'data' => $version], 200);
  245. }
  246. public function versionwin(Request $request){
  247. $version = AppUpdate::where('appname','=','win')->first();
  248. return response()->json(['ret' => 1, 'data' => $version], 200);
  249. }
  250. //商品关系
  251. public function shop()
  252. {
  253. $shops = [
  254. 'keys' => [],
  255. 'data' => [],
  256. ];
  257. foreach (GoodsCategory::query()->whereStatus(1)->get() as $item) {
  258. $shops['keys'][] = $item['name'];
  259. $shops['data'][$item['name']] = $item->goods()->get()->append('traffic_label')->toArray();
  260. }
  261. return response()->json(['ret' => 1, 'data' => $shops]);
  262. }
  263. public function getConfig()
  264. {
  265. $config = config('bobclient');
  266. $config['website_name'] = sysConfig('website_name');
  267. $config['website_url'] = sysConfig('website_url');
  268. $config['payment'] = [
  269. 'alipay' => sysConfig('is_AliPay'),
  270. 'wechat' => sysConfig('is_WeChatPay'),
  271. ];
  272. return response()->json(['ret' => 1, 'data' => $config]);
  273. }
  274. public function purchase(Request $request)
  275. {
  276. $goods_id = $request->input('goods_id');
  277. $coupon_sn = $request->input('coupon_sn');
  278. self::$method = $request->input('method');
  279. $credit = $request->input('amount');
  280. $pay_type = $request->input('pay_type');
  281. $amount = 0;
  282. if ($credit) { // 充值余额
  283. if (! is_numeric($credit) || $credit <= 0) {
  284. return response()->json(['ret' => 0, 'msg' => trans('user.payment.error')]);
  285. }
  286. $amount = $credit;
  287. } elseif ($goods_id && self::$method) { // 购买服务
  288. $goods = Goods::find($goods_id);
  289. if (! $goods || ! $goods->status) {
  290. return response()->json(['ret' => 0, 'msg' => '订单创建失败:商品已下架']);
  291. }
  292. $amount = $goods->price;
  293. // 是否有生效的套餐
  294. $activePlan = Order::userActivePlan()->doesntExist();
  295. // 无生效套餐,禁止购买加油包
  296. if ($goods->type === 1 && $activePlan) {
  297. return response()->json(['ret' => 0, 'msg' => '购买加油包前,请先购买套餐']);
  298. }
  299. // 单个商品限购
  300. if ($goods->limit_num) {
  301. $count = Order::uid()->where('status', '>=', 0)->whereGoodsId($goods_id)->count();
  302. if ($count >= $goods->limit_num) {
  303. return response()->json(['ret' => 0, 'msg' => '此商品限购'.$goods->limit_num.'次,您已购买'.$count.'次']);
  304. }
  305. }
  306. // 使用优惠券
  307. if ($coupon_sn) {
  308. $coupon = Coupon::whereStatus(0)->whereIn('type', [1, 2])->whereSn($coupon_sn)->first();
  309. if (! $coupon) {
  310. return response()->json(['ret' => 0, 'msg' => '订单创建失败:优惠券不存在']);
  311. }
  312. // 计算实际应支付总价
  313. $amount = $coupon->type === 2 ? $goods->price * $coupon->value / 100 : $goods->price - $coupon->value;
  314. $amount = $amount > 0 ? round($amount, 2) : 0; // 四舍五入保留2位小数,避免无法正常创建订单
  315. }
  316. //非余额付款下,检查在线支付是否开启
  317. if (self::$method !== 'credit') {
  318. // 判断是否开启在线支付
  319. if (! sysConfig('is_onlinePay')) {
  320. return response()->json(['ret' => 0, 'msg' => '订单创建失败:系统并未开启在线支付功能']);
  321. }
  322. // 判断是否存在同个商品的未支付订单
  323. if (Order::uid()->whereStatus(0)->exists()) {
  324. return response()->json(['ret' => 0, 'msg' => '订单创建失败:尚有未支付的订单,请先去支付']);
  325. }
  326. } elseif (Auth::getUser()->credit < $amount) { // 验证账号余额是否充足
  327. return response()->json(['ret' => 0, 'msg' => '您的余额不足,请先充值']);
  328. }
  329. // 价格异常判断
  330. if ($amount < 0) {
  331. return response()->json(['ret' => 0, 'msg' => '订单创建失败:订单总价异常']);
  332. }
  333. if ($amount === 0 && self::$method !== 'credit') {
  334. return response()->json(['ret' => 0, 'msg' => '订单创建失败:订单总价为0,无需使用在线支付']);
  335. }
  336. }
  337. // 生成订单
  338. try {
  339. $newOrder = Order::create([
  340. 'sn' => date('ymdHis').random_int(100000, 999999),
  341. 'user_id' => auth()->id(),
  342. 'goods_id' => $credit ? null : $goods_id,
  343. 'coupon_id' => $coupon->id ?? null,
  344. 'origin_amount' => $credit ?: ($goods->price ?? 0),
  345. 'amount' => $amount,
  346. 'pay_type' => $pay_type,
  347. 'pay_way' => self::$method,
  348. ]);
  349. // 使用优惠券,减少可使用次数
  350. if (! empty($coupon)) {
  351. if ($coupon->usable_times > 0) {
  352. $coupon->decrement('usable_times', 1);
  353. }
  354. Helpers::addCouponLog('订单支付使用', $coupon->id, $goods_id, $newOrder->id);
  355. }
  356. $request->merge(['id' => $newOrder->id, 'type' => $pay_type, 'amount' => $amount]);
  357. PaymentController::$method = self::$method;
  358. // 生成支付单
  359. $data = PaymentController::getClient()->purchase($request);
  360. $data = $data->getData(true);
  361. $data['order_id'] = $newOrder->id;
  362. return response()->json($data);
  363. } catch (Exception $e) {
  364. Log::error('订单生成错误:'.$e->getMessage());
  365. }
  366. return response()->json(['ret' => 0, 'msg' => '订单创建失败']);
  367. }
  368. public function gift(Request $request)
  369. {
  370. $user = $request->user('api');
  371. $referral_traffic = flowAutoShow(sysConfig('referral_traffic') * MB);
  372. $referral_percent = sysConfig('referral_percent');
  373. // 邀请码
  374. $code = $user->invites()->whereStatus(1)->value('code');
  375. $data['invite_gift'] = trans('user.invite.promotion', [
  376. 'traffic' => $referral_traffic,
  377. 'referral_percent' => $referral_percent * 100,
  378. ]);
  379. $affSalt = sysConfig('aff_salt');
  380. if (isset($affSalt)) {
  381. $aff_link = route('register', ['aff' => (new Hashids($affSalt, 8))->encode($user->id)]);
  382. } else {
  383. $aff_link = route('register', ['aff' => $user->id]);
  384. }
  385. $data['invite_url'] = $aff_link;
  386. $data['invite_text'] = $aff_link.'&(复制整段文字到浏览器打开即可访问),找梯子最重要的就是稳定,这个已经上线三年了,一直稳定没有被封过,赶紧下载备用吧!安装后打开填写我的邀请码【'.$code.'】,你还能多得3天会员.';
  387. // 累计数据
  388. $data['back_sum'] = ReferralLog::query()->where('inviter_id', $user->id)->sum('commission') / 100;
  389. $data['user_sum'] = $user->invitees()->count();
  390. $data['list'] = $user->invitees()->selectRaw('username, UNIX_TIMESTAMP(created_at) as created_at')->limit(10)->get();
  391. return response()->json(['ret' => 1, 'data' => $data]);
  392. }
  393. public function checkIn(Request $request): JsonResponse
  394. {
  395. $user = $request->user();
  396. // 系统开启登录加积分功能才可以签到
  397. if (! sysConfig('is_checkin')) {
  398. return response()->json(['ret' => 0, 'title' => trans('common.failed'), 'msg' => trans('user.home.attendance.disable')]);
  399. }
  400. // 已签到过,验证是否有效
  401. if (Cache::has('userCheckIn_'.$user->id)) {
  402. return response()->json(['ret' => 0, 'title' => trans('common.success'), 'msg' => trans('user.home.attendance.done')]);
  403. }
  404. $traffic = random_int((int) sysConfig('min_rand_traffic'), (int) sysConfig('max_rand_traffic')) * MB;
  405. if (! $user->incrementData($traffic)) {
  406. return response()->json(['ret' => 0, 'title' => trans('common.failed'), 'msg' => trans('user.home.attendance.failed')]);
  407. }
  408. // 写入用户流量变动记录
  409. Helpers::addUserTrafficModifyLog($user->id, null, $user->transfer_enable, $user->transfer_enable + $traffic, trans('user.home.attendance.attribute'));
  410. // 多久后可以再签到
  411. $ttl = sysConfig('traffic_limit_time') ? sysConfig('traffic_limit_time') * Minute : Day;
  412. Cache::put('userCheckIn_'.$user->id, '1', $ttl);
  413. return response()->json(['ret' => 1, 'msg' => trans('user.home.attendance.success', ['data' => flowAutoShow($traffic)])]);
  414. }
  415. public function phpMaxMin($arr = [],$keys = ''){
  416. $max['key'] = '';
  417. $max['value'] = '';
  418. $min['key'] = '';
  419. $min['value'] = '';
  420. foreach ($arr as $key => $val){
  421. if($max['key'] === ''){
  422. $max['key'] = $key;
  423. $max['value'] = $val[$keys];
  424. }
  425. if((int)$max['value'] < $val[$keys]){
  426. $max['key'] = $key;
  427. $max['value'] = $val[$keys];
  428. }
  429. if($min['key'] === ''){
  430. $min['key'] = $key;
  431. $min['value'] = $val[$keys];
  432. }
  433. if((int)$min['value'] > $val[$keys]){
  434. $min['key'] = $key;
  435. $min['value'] = $val[$keys];
  436. }
  437. }
  438. $array['max'] = $max;
  439. $array['min'] = $min;
  440. return $array;
  441. }
  442. }