NodeController.php 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Components\Helpers;
  4. use App\Components\NetworkDetection;
  5. use App\Jobs\VNet\reloadNode;
  6. use App\Models\Country;
  7. use App\Models\Label;
  8. use App\Models\Level;
  9. use App\Models\Node;
  10. use App\Models\NodeAuth;
  11. use App\Models\NodeCertificate;
  12. use App\Models\NodePing;
  13. use App\Services\NodeService;
  14. use DB;
  15. use Exception;
  16. use Illuminate\Http\JsonResponse;
  17. use Illuminate\Http\Request;
  18. use Log;
  19. use Redirect;
  20. use Response;
  21. use Session;
  22. use Str;
  23. use Validator;
  24. class NodeController extends Controller
  25. {
  26. // 节点列表
  27. public function nodeList(Request $request)
  28. {
  29. $status = $request->input('status');
  30. $query = Node::with(['onlineLogs', 'dailyDataFlows']);
  31. if (isset($status)) {
  32. $query->whereStatus($status);
  33. }
  34. $nodeList = $query->orderByDesc('sort')
  35. ->orderBy('id')
  36. ->paginate(15)
  37. ->appends($request->except('page'));
  38. foreach ($nodeList as $node) {
  39. // 在线人数
  40. $online_log = $node->onlineLogs()
  41. ->where(
  42. 'log_time',
  43. '>=',
  44. strtotime("-5 minutes")
  45. )
  46. ->latest('log_time')
  47. ->first();
  48. $node->online_users = empty($online_log) ? 0 : $online_log->online_user;
  49. // 已产生流量
  50. $node->transfer = flowAutoShow(
  51. $node->dailyDataFlows()->sum('total')
  52. );
  53. // 负载(10分钟以内)
  54. $node_info = $node->heartBeats()->recently()->first();
  55. $node->isOnline = empty($node_info) || empty($node_info->load) ? 0 : 1;
  56. $node->load = $node->isOnline ? $node_info->load : '离线';
  57. $node->uptime = empty($node_info) ? 0 : seconds2time(
  58. $node_info->uptime
  59. );
  60. }
  61. $view['nodeList'] = $nodeList;
  62. return view('admin.node.nodeList', $view);
  63. }
  64. public function checkNode($id): JsonResponse
  65. {
  66. $node = Node::find($id);
  67. // 使用DDNS的node先获取ipv4地址
  68. if ($node->is_ddns) {
  69. $ip = gethostbyname($node->server);
  70. if (strcmp($ip, $node->server) != 0) {
  71. $node->ip = $ip;
  72. } else {
  73. return Response::json(
  74. [
  75. 'status' => 'fail',
  76. 'title' => 'IP获取错误',
  77. 'message' => $node->name . 'IP获取失败',
  78. ]
  79. );
  80. }
  81. }
  82. $data[0] = NetworkDetection::networkCheck($node->ip, true); //ICMP
  83. $data[1] = NetworkDetection::networkCheck(
  84. $node->ip,
  85. false,
  86. $node->single ? $node->port : null
  87. ); //TCP
  88. return Response::json(
  89. [
  90. 'status' => 'success',
  91. 'title' => '[' . $node->name . ']阻断信息',
  92. 'message' => $data,
  93. ]
  94. );
  95. }
  96. // 添加节点
  97. public function addNode(Request $request)
  98. {
  99. if ($request->isMethod('POST')) {
  100. $validator = $this->nodeValidation($request);
  101. if ($validator) {
  102. return $validator;
  103. }
  104. // TODO:判断是否已存在绑定了相同域名的节点,提示是否要强制替换,或者不提示之前强制将其他节点的绑定域名置为空,然后发起域名绑定请求,或者请求进入队列
  105. try {
  106. DB::beginTransaction();
  107. $node = new Node();
  108. $node->type = $request->input('type');
  109. $node->name = $request->input('name');
  110. $node->country_code = $request->input('country_code');
  111. $node->server = $request->input('server');
  112. $node->ip = $request->input('ip');
  113. $node->ipv6 = $request->input('ipv6');
  114. $node->relay_server = $request->input('relay_server');
  115. $node->relay_port = $request->input('relay_port');
  116. $node->level = $request->input('level');
  117. $node->speed_limit = (int)$request->input(
  118. 'speed_limit'
  119. ) * Mbps;
  120. $node->client_limit = $request->input('client_limit');
  121. $node->description = $request->input('description');
  122. $node->method = $request->input('method');
  123. $node->protocol = $request->input('protocol');
  124. $node->protocol_param = $request->input('protocol_param');
  125. $node->obfs = $request->input('obfs');
  126. $node->obfs_param = $request->input('obfs_param');
  127. $node->traffic_rate = $request->input('traffic_rate');
  128. $node->is_subscribe = (int)$request->input('is_subscribe');
  129. $node->is_ddns = (int)$request->input('is_ddns');
  130. $node->is_relay = (int)$request->input('is_relay');
  131. $node->is_udp = (int)$request->input('is_udp');
  132. $node->push_port = $request->input('push_port');
  133. $node->detection_type = $request->input('detection_type');
  134. $node->compatible = (int)$request->input('compatible');
  135. $node->single = (int)$request->input('single');
  136. $node->port = $request->input('port');
  137. $node->passwd = $request->input('passwd');
  138. $node->sort = $request->input('sort');
  139. $node->status = (int)$request->input('status');
  140. $node->v2_alter_id = $request->input('v2_alter_id');
  141. $node->v2_port = $request->input('v2_port');
  142. $node->v2_method = $request->input('v2_method');
  143. $node->v2_net = $request->input('v2_net');
  144. $node->v2_type = $request->input('v2_type');
  145. $node->v2_host = $request->input('v2_host') ?: '';
  146. $node->v2_path = $request->input('v2_path');
  147. $node->v2_tls = (int)$request->input('v2_tls');
  148. $node->tls_provider = $request->input('tls_provider');
  149. $node->save();
  150. DB::commit();
  151. // 生成节点标签
  152. (new NodeService())->makeLabels(
  153. $node->id,
  154. $request->input('labels')
  155. );
  156. return Response::json(
  157. ['status' => 'success', 'message' => '添加成功']
  158. );
  159. } catch (Exception $e) {
  160. DB::rollBack();
  161. Log::error('添加节点信息异常:' . $e->getMessage());
  162. return Response::json(
  163. [
  164. 'status' => 'fail',
  165. 'message' => '添加失败:' . $e->getMessage(),
  166. ]
  167. );
  168. }
  169. } else {
  170. $view['methodList'] = Helpers::methodList();
  171. $view['protocolList'] = Helpers::protocolList();
  172. $view['obfsList'] = Helpers::obfsList();
  173. $view['countryList'] = Country::orderBy('code')->get();
  174. $view['levelList'] = Level::orderBy('level')->get();
  175. $view['labelList'] = Label::orderByDesc('sort')
  176. ->orderBy('id')
  177. ->get();
  178. $view['dvList'] = NodeCertificate::orderBy('id')->get();
  179. return view('admin.node.nodeInfo', $view);
  180. }
  181. }
  182. // 节点信息验证
  183. private function nodeValidation(Request $request)
  184. {
  185. if ($request->input('server')) {
  186. $domain = $request->input('server');
  187. $domain = explode('.', $domain);
  188. $domainSuffix = end($domain); // 取得域名后缀
  189. if ( ! in_array($domainSuffix, config('domains'), true)) {
  190. return Response::json(
  191. ['status' => 'fail', 'message' => '绑定域名不合法']
  192. );
  193. }
  194. }
  195. $validator = Validator::make(
  196. $request->all(),
  197. [
  198. 'type' => 'required|between:1,3',
  199. 'name' => 'required',
  200. 'country_code' => 'required',
  201. 'server' => 'required_if:is_ddns,1',
  202. 'push_port' => 'numeric|between:0,65535',
  203. 'traffic_rate' => 'required|numeric|min:0',
  204. 'level' => 'required|numeric|between:0,255',
  205. 'speed_limit' => 'required|numeric|min:0',
  206. 'client_limit' => 'required|numeric|min:0',
  207. 'port' => 'nullable|numeric|between:0,65535',
  208. 'ip' => 'ipv4',
  209. 'ipv6' => 'nullable|ipv6',
  210. 'relay_server' => 'required_if:is_relay,1',
  211. 'relay_port' => 'required_if:is_relay,1|numeric|between:0,65535',
  212. 'method' => 'required_if:type,1',
  213. 'protocol' => 'required_if:type,1',
  214. 'obfs' => 'required_if:type,1',
  215. 'is_subscribe' => 'boolean',
  216. 'is_ddns' => 'boolean',
  217. 'is_relay' => 'boolean',
  218. 'is_udp' => 'boolean',
  219. 'detection_type' => 'between:0,3',
  220. 'compatible' => 'boolean',
  221. 'single' => 'boolean',
  222. 'sort' => 'required|numeric|between:0,255',
  223. 'status' => 'boolean',
  224. 'v2_alter_id' => 'required_if:type,2|numeric|between:0,65535',
  225. 'v2_port' => 'required_if:type,2|numeric|between:0,65535',
  226. 'v2_method' => 'required_if:type,2',
  227. 'v2_net' => 'required_if:type,2',
  228. 'v2_type' => 'required_if:type,2',
  229. 'v2_tls' => 'boolean',
  230. ],
  231. [
  232. 'server.required_unless' => '开启DDNS, 域名不能为空',
  233. ]
  234. );
  235. if ($validator->fails()) {
  236. return Response::json(
  237. ['status' => 'fail', 'message' => $validator->errors()->all()]
  238. );
  239. }
  240. return false;
  241. }
  242. // 刷新节点地理位置
  243. public function refreshGeo(Request $request): JsonResponse
  244. {
  245. if ((new NodeService())->getNodeGeo($request->input('id', 0))) {
  246. return Response::json(
  247. ['status' => 'success', 'message' => '获取地理位置更新成功!']
  248. );
  249. }
  250. return Response::json(['status' => 'fail', 'message' => '获取地理位置更新失败!']);
  251. }
  252. // 重载节点
  253. public function reload($id): JsonResponse
  254. {
  255. if (reloadNode::dispatchNow(Node::whereId($id)->get())) {
  256. return Response::json(
  257. ['status' => 'success', 'message' => '重载成功!']
  258. );
  259. }
  260. return Response::json(['status' => 'fail', 'message' => '重载失败!']);
  261. }
  262. // 编辑节点
  263. public function editNode(Request $request)
  264. {
  265. $id = $request->input('id');
  266. if ($request->isMethod('POST')) {
  267. $validator = $this->nodeValidation($request);
  268. if ($validator) {
  269. return $validator;
  270. }
  271. $node = Node::find($id);
  272. try {
  273. DB::beginTransaction();
  274. // 生成节点标签
  275. (new NodeService())->makeLabels(
  276. $node->id,
  277. $request->input('labels')
  278. );
  279. $node->update(
  280. [
  281. 'type' => $request->input('type'),
  282. 'name' => $request->input('name'),
  283. 'country_code' => $request->input('country_code'),
  284. 'server' => $request->input('server'),
  285. 'ip' => $request->input('ip'),
  286. 'ipv6' => $request->input('ipv6'),
  287. 'relay_server' => $request->input('relay_server'),
  288. 'relay_port' => $request->input('relay_port'),
  289. 'level' => $request->input('level'),
  290. 'speed_limit' => (int)$request->input(
  291. 'speed_limit'
  292. ) * Mbps,
  293. 'client_limit' => $request->input('client_limit'),
  294. 'description' => $request->input('description'),
  295. 'method' => $request->input('method'),
  296. 'protocol' => $request->input('protocol'),
  297. 'protocol_param' => $request->input('protocol_param'),
  298. 'obfs' => $request->input('obfs'),
  299. 'obfs_param' => $request->input('obfs_param'),
  300. 'traffic_rate' => $request->input('traffic_rate'),
  301. 'is_subscribe' => (int)$request->input(
  302. 'is_subscribe'
  303. ),
  304. 'is_ddns' => (int)$request->input('is_ddns'),
  305. 'is_relay' => (int)$request->input('is_relay'),
  306. 'is_udp' => (int)$request->input('is_udp'),
  307. 'push_port' => $request->input('push_port'),
  308. 'detection_type' => $request->input('detection_type'),
  309. 'compatible' => (int)$request->input('compatible'),
  310. 'single' => (int)$request->input('single'),
  311. 'port' => $request->input('port'),
  312. 'passwd' => $request->input('passwd'),
  313. 'sort' => $request->input('sort'),
  314. 'status' => (int)$request->input('status'),
  315. 'v2_alter_id' => $request->input('v2_alter_id'),
  316. 'v2_port' => $request->input('v2_port'),
  317. 'v2_method' => $request->input('v2_method'),
  318. 'v2_net' => $request->input('v2_net'),
  319. 'v2_type' => $request->input('v2_type'),
  320. 'v2_host' => $request->input('v2_host') ?: '',
  321. 'v2_path' => $request->input('v2_path'),
  322. 'v2_tls' => (int)$request->input('v2_tls'),
  323. 'tls_provider' => $request->input('tls_provider'),
  324. ]
  325. );
  326. // TODO:更新节点绑定的域名DNS(将节点IP更新到域名DNS 的A记录)
  327. DB::commit();
  328. return Response::json(
  329. ['status' => 'success', 'message' => '编辑成功']
  330. );
  331. } catch (Exception $e) {
  332. DB::rollBack();
  333. Log::error('编辑节点信息异常:' . $e->getMessage());
  334. return Response::json(
  335. [
  336. 'status' => 'fail',
  337. 'message' => '编辑失败:' . $e->getMessage(),
  338. ]
  339. );
  340. }
  341. }
  342. $view['node'] = Node::with('labels')->find($id);
  343. $view['methodList'] = Helpers::methodList();
  344. $view['protocolList'] = Helpers::protocolList();
  345. $view['obfsList'] = Helpers::obfsList();
  346. $view['countryList'] = Country::orderBy('code')->get();
  347. $view['levelList'] = Level::orderBy('level')->get();
  348. $view['labelList'] = Label::orderByDesc('sort')->orderBy('id')->get(
  349. );
  350. $view['dvList'] = NodeCertificate::orderBy('id')->get();
  351. return view('admin.node.nodeInfo', $view);
  352. }
  353. // 删除节点
  354. public function delNode(Request $request): ?JsonResponse
  355. {
  356. $id = $request->input('id');
  357. $node = Node::find($id);
  358. if ( ! $node) {
  359. return Response::json(
  360. ['status' => 'fail', 'message' => '节点不存在,请重试']
  361. );
  362. }
  363. try {
  364. DB::beginTransaction();
  365. $node->delete();
  366. DB::commit();
  367. return Response::json(['status' => 'success', 'message' => '删除成功']);
  368. } catch (Exception $e) {
  369. DB::rollBack();
  370. Log::error('删除节点信息异常:' . $e->getMessage());
  371. return Response::json(
  372. ['status' => 'fail', 'message' => '删除失败:' . $e->getMessage()]
  373. );
  374. }
  375. }
  376. // 节点流量监控
  377. public function nodeMonitor(Request $request)
  378. {
  379. $node = Node::find($request->input('id'));
  380. if ( ! $node) {
  381. Session::flash('errorMsg', '节点不存在,请重试');
  382. return Redirect::back();
  383. }
  384. $view['nodeName'] = $node->name;
  385. $view['nodeServer'] = $node->server;
  386. $view = array_merge(
  387. $view,
  388. $this->DataFlowChart($node->id, 1)
  389. );
  390. return view('admin.node.nodeMonitor', $view);
  391. }
  392. // Ping节点延迟
  393. public function pingNode($id): ?JsonResponse
  394. {
  395. $node = Node::find($id);
  396. if ( ! $node) {
  397. return Response::json(
  398. ['status' => 'fail', 'message' => '节点不存在,请重试']
  399. );
  400. }
  401. $result = NetworkDetection::ping(
  402. $node->is_ddns ? $node->server : $node->ip
  403. );
  404. if ($result) {
  405. return Response::json(
  406. [
  407. 'status' => 'success',
  408. 'message' => [
  409. $result['telecom']['time'] ?: '无',//电信
  410. $result['Unicom']['time'] ?: '无',// 联通
  411. $result['move']['time'] ?: '无',// 移动
  412. $result['HongKong']['time'] ?: '无'// 香港
  413. ],
  414. ]
  415. );
  416. }
  417. return Response::json(['status' => 'fail', 'message' => 'Ping访问失败']);
  418. }
  419. // Ping节点延迟日志
  420. public function pingLog(Request $request)
  421. {
  422. $node_id = $request->input('nodeId');
  423. $query = NodePing::query();
  424. if (isset($node_id)) {
  425. $query->whereNodeId($node_id);
  426. }
  427. $view['nodeList'] = Node::orderBy('id')->get();
  428. $view['pingLogs'] = $query->latest()->paginate(15)->appends(
  429. $request->except('page')
  430. );
  431. return view('admin.logs.nodePingLog', $view);
  432. }
  433. // 节点授权列表
  434. public function authList(Request $request)
  435. {
  436. $view['list'] = NodeAuth::orderBy('node_id')->paginate(15)->appends(
  437. $request->except('page')
  438. );
  439. return view('admin.node.authList', $view);
  440. }
  441. // 添加节点授权
  442. public function addAuth(): JsonResponse
  443. {
  444. $nodeArray = Node::whereStatus(1)->orderBy('id')->pluck('id')->toArray(
  445. );
  446. $authArray = NodeAuth::orderBy('id')->pluck('node_id')->toArray();
  447. if ($nodeArray == $authArray) {
  448. return Response::json(
  449. ['status' => 'success', 'message' => '没有需要生成授权的节点']
  450. );
  451. }
  452. foreach (array_diff($nodeArray, $authArray) as $nodeId) {
  453. $obj = new NodeAuth();
  454. $obj->node_id = $nodeId;
  455. $obj->key = Str::random();
  456. $obj->secret = Str::random(8);
  457. $obj->save();
  458. }
  459. return Response::json(['status' => 'success', 'message' => '生成成功']);
  460. }
  461. // 删除节点授权
  462. public function delAuth(Request $request): JsonResponse
  463. {
  464. try {
  465. NodeAuth::whereId($request->input('id'))->delete();
  466. } catch (Exception $e) {
  467. return Response::json(
  468. ['status' => 'fail', 'message' => '错误:' . var_export($e, true)]
  469. );
  470. }
  471. return Response::json(['status' => 'success', 'message' => '操作成功']);
  472. }
  473. // 重置节点授权
  474. public function refreshAuth(Request $request): ?JsonResponse
  475. {
  476. $ret = NodeAuth::whereId($request->input('id'))->update(
  477. [
  478. 'key' => Str::random(),
  479. 'secret' => Str::random(8),
  480. ]
  481. );
  482. if ($ret) {
  483. return Response::json(['status' => 'success', 'message' => '操作成功']);
  484. }
  485. return Response::json(['status' => 'fail', 'message' => '操作失败']);
  486. }
  487. // 域名证书列表
  488. public function certificateList(Request $request)
  489. {
  490. $DvList = NodeCertificate::orderBy('id')->paginate(15)->appends(
  491. $request->except('page')
  492. );
  493. foreach ($DvList as $Dv) {
  494. if ($Dv->key && $Dv->pem) {
  495. $DvInfo = openssl_x509_parse($Dv->pem);
  496. $Dv->issuer = $DvInfo['issuer']['O'];
  497. $Dv->from = $DvInfo['validFrom_time_t'] ? date(
  498. 'Y-m-d',
  499. $DvInfo['validFrom_time_t']
  500. ) : null;
  501. $Dv->to = $DvInfo['validTo'] ? date(
  502. 'Y-m-d',
  503. $DvInfo['validTo_time_t']
  504. ) : null;
  505. }
  506. }
  507. $view['list'] = $DvList;
  508. return view('admin.node.certificateList', $view);
  509. }
  510. // 添加域名证书
  511. public function addCertificate(Request $request)
  512. {
  513. if ($request->isMethod('POST')) {
  514. $obj = new NodeCertificate();
  515. $obj->domain = $request->input('domain');
  516. $obj->key = str_replace(
  517. ["\r", "\n"],
  518. '',
  519. $request->input('key')
  520. );
  521. $obj->pem = str_replace(
  522. ["\r", "\n"],
  523. '',
  524. $request->input('pem')
  525. );
  526. $obj->save();
  527. if ($obj->id) {
  528. return Response::json(
  529. ['status' => 'success', 'message' => '生成成功']
  530. );
  531. }
  532. return Response::json(['status' => 'fail', 'message' => '生成失败']);
  533. }
  534. return view('admin.node.certificateInfo');
  535. }
  536. // 编辑域名证书
  537. public function editCertificate(Request $request)
  538. {
  539. $Dv = NodeCertificate::find($request->input('id'));
  540. if ($request->isMethod('POST')) {
  541. if ($Dv) {
  542. $ret = NodeCertificate::whereId($Dv->id)->update(
  543. [
  544. 'domain' => $request->input('domain'),
  545. 'key' => $request->input('key'),
  546. 'pem' => $request->input('pem'),
  547. ]
  548. );
  549. if ($ret) {
  550. return Response::json(
  551. ['status' => 'success', 'message' => '修改成功']
  552. );
  553. }
  554. }
  555. return Response::json(['status' => 'fail', 'message' => '修改失败']);
  556. }
  557. $view['Dv'] = $Dv;
  558. return view('admin.node.certificateInfo', $view);
  559. }
  560. // 删除域名证书
  561. public function delCertificate(Request $request): JsonResponse
  562. {
  563. try {
  564. NodeCertificate::whereId($request->input('id'))->delete();
  565. } catch (Exception $e) {
  566. return Response::json(
  567. ['status' => 'fail', 'message' => '错误:' . var_export($e, true)]
  568. );
  569. }
  570. return Response::json(['status' => 'success', 'message' => '操作成功']);
  571. }
  572. }